From 76b485d9dcdce082c70776447612fb10f530e74a Mon Sep 17 00:00:00 2001 From: 0xFEEDC0DE64 Date: Sun, 19 Jul 2020 01:01:04 +0200 Subject: [PATCH] Implemented basic chat functionality --- chathelper.cpp | 41 +++++++++++++-------- chathelper.h | 19 +++++++--- closedclient.cpp | 2 -- handshakingclient.cpp | 2 -- loginclient.cpp | 4 +-- packets.cpp | 5 +++ packets.h | 8 ++++- playclient.cpp | 84 ++++++++++++++++++++++++------------------- playclient.h | 14 +++++--- server.cpp | 13 +++---- statusclient.cpp | 4 +-- 11 files changed, 118 insertions(+), 78 deletions(-) diff --git a/chathelper.cpp b/chathelper.cpp index cde2cdd..1f6ff0a 100644 --- a/chathelper.cpp +++ b/chathelper.cpp @@ -1,25 +1,36 @@ #include "chathelper.h" #include +#include -QJsonObject normalTextObj(const QString &text) +QJsonObject ChatPart::toObject() const { - return QJsonObject { { "text", text } }; -} - -QJsonObject boldTextObj(const QString &text) -{ - QJsonObject obj = normalTextObj(text); - obj["bold"] = true; + QJsonObject obj; + if (text.has_value()) + obj["text"] = *text; + if (bold.has_value()) + obj["bold"] = *bold; + if (italic.has_value()) + obj["italic"] = *italic; + if (underlined.has_value()) + obj["underlined"] = *underlined; + if (strikethrough.has_value()) + obj["strikethrough"] = *strikethrough; + if (obfuscated.has_value()) + obj["obfuscated"] = *obfuscated; + if (color.has_value()) + obj["color"] = *color; + if (!extra.empty()) + { + QJsonArray arr; + for (const auto &sub : extra) + arr.append(sub.toObject()); + obj["extra"] = arr; + } return obj; } -QString normalText(const QString &text) +QString ChatPart::toString() const { - return QJsonDocument{normalTextObj(text)}.toJson(); -} - -QString boldText(const QString &text) -{ - return QJsonDocument{boldTextObj(text)}.toJson(); + return QJsonDocument{toObject()}.toJson(QJsonDocument::Compact); } diff --git a/chathelper.h b/chathelper.h index eac4728..e60d344 100644 --- a/chathelper.h +++ b/chathelper.h @@ -1,10 +1,21 @@ #pragma once +#include +#include + #include #include -QJsonObject normalTextObj(const QString &text); -QJsonObject boldTextObj(const QString &text); +struct ChatPart { + std::optional text; + std::optional bold; + std::optional italic; + std::optional underlined; + std::optional strikethrough; + std::optional obfuscated; + std::optional color; + std::vector extra; -QString normalText(const QString &text); -QString boldText(const QString &text); + QJsonObject toObject() const; + QString toString() const; +}; diff --git a/closedclient.cpp b/closedclient.cpp index 1ab660f..786fc96 100644 --- a/closedclient.cpp +++ b/closedclient.cpp @@ -8,8 +8,6 @@ ClosedClient::ClosedClient(std::unique_ptr &&socket, Server &server) : QObject{&server}, m_socket{std::move(socket)}, m_server{server}, m_dataStream{m_socket.get()} { - m_socket->setParent(this); - connect(m_socket.get(), &QIODevice::readyRead, this, &ClosedClient::readyRead); connect(m_socket.get(), &QAbstractSocket::disconnected, this, &QObject::deleteLater); diff --git a/handshakingclient.cpp b/handshakingclient.cpp index 9f7bc57..a4fcdec 100644 --- a/handshakingclient.cpp +++ b/handshakingclient.cpp @@ -9,8 +9,6 @@ HandshakingClient::HandshakingClient(std::unique_ptr &&socket, Server &server) : QObject{&server}, m_socket{std::move(socket)}, m_server{server}, m_dataStream{m_socket.get()} { - m_socket->setParent(this); - connect(m_socket.get(), &QIODevice::readyRead, this, &HandshakingClient::readyRead); connect(m_socket.get(), &QAbstractSocket::disconnected, this, &QObject::deleteLater); } diff --git a/loginclient.cpp b/loginclient.cpp index d2eb1f4..82c1148 100644 --- a/loginclient.cpp +++ b/loginclient.cpp @@ -8,8 +8,6 @@ LoginClient::LoginClient(std::unique_ptr &&socket, Server &server) : QObject{&server}, m_socket{std::move(socket)}, m_server{server}, m_dataStream{m_socket.get()} { - m_socket->setParent(this); - connect(m_socket.get(), &QIODevice::readyRead, this, &LoginClient::readyRead); connect(m_socket.get(), &QAbstractSocket::disconnected, this, &QObject::deleteLater); } @@ -67,7 +65,7 @@ void LoginClient::readPacket(packets::login::serverbound::PacketType type, const packet.serialize(m_dataStream); } m_dataStream.setDevice({}); - new PlayClient{std::move(m_socket), m_server}; + new PlayClient{name, std::move(m_socket), m_server}; deleteLater(); break; } diff --git a/packets.cpp b/packets.cpp index 67823d5..3787872 100644 --- a/packets.cpp +++ b/packets.cpp @@ -61,6 +61,11 @@ void packets::login::clientbound::LoginSuccess::serialize(McDataStream &stream) stream.writeRawData(buffer.constData(), buffer.length()); } +packets::play::serverbound::ChatMessage::ChatMessage(McDataStream &stream) +{ + message = stream.readVar(); +} + packets::play::serverbound::ClientSettings::ClientSettings(McDataStream &stream) { locale = stream.readVar(); diff --git a/packets.h b/packets.h index add19ed..6c951f3 100644 --- a/packets.h +++ b/packets.h @@ -116,9 +116,15 @@ namespace packets { namespace serverbound { Q_NAMESPACE - enum class PacketType { ClientSettings = 0x04, InteractEntity = 0x0E, PluginMessage = 0x0A }; + enum class PacketType { ChatMessage = 0x02, ClientSettings = 0x04, InteractEntity = 0x0E, PluginMessage = 0x0A }; Q_ENUM_NS(PacketType) + struct ChatMessage { + ChatMessage(McDataStream &stream); + + QString message; + }; + struct ClientSettings { ClientSettings(McDataStream &stream); diff --git a/playclient.cpp b/playclient.cpp index 837c9c5..0614102 100644 --- a/playclient.cpp +++ b/playclient.cpp @@ -8,12 +8,12 @@ #include "chathelper.h" #include "chunkhelper.h" -PlayClient::PlayClient(std::unique_ptr &&socket, Server &server) : - QObject{&server}, m_socket{std::move(socket)}, m_server{server}, m_dataStream{m_socket.get()} +PlayClient::PlayClient(const QString &name, std::unique_ptr &&socket, Server &server) : + QObject{&server}, m_name{name}, m_socket{std::move(socket)}, m_server{server}, m_dataStream{m_socket.get()} { m_server.add(*this); - m_socket->setParent(this); + emit distributeChatMessage(ChatPart{.extra={ChatPart{.text=m_name,.color="red"},ChatPart{.text=tr(" joined the game!")}}}.toString()); connect(m_socket.get(), &QIODevice::readyRead, this, &PlayClient::readyRead); connect(m_socket.get(), &QAbstractSocket::disconnected, this, &QObject::deleteLater); @@ -52,22 +52,22 @@ PlayClient::PlayClient(std::unique_ptr &&socket, Server &server) : packet.fieldOfViewModifier = 60.; packet.serialize(m_dataStream); } - { - packets::play::clientbound::ChunkData packet; - packet.fullChunk = true; - packet.primaryBitMask = 0b11000; // send only 2 chunks - packet.data += createChunkSection(); - packet.data += createChunkSection(); - packet.data += createBiomes(); // we are in Overworld +// { +// packets::play::clientbound::ChunkData packet; +// packet.fullChunk = true; +// packet.primaryBitMask = 0b11000; // send only 2 chunks +// packet.data += createChunkSection(); +// packet.data += createChunkSection(); +// packet.data += createBiomes(); // we are in Overworld - for (int y = -3; y <= 3; y++) - for (int x = -3; x <= 3; x++) - { - packet.chunkX = x; - packet.chunkY = y; - packet.serialize(m_dataStream); - } - } +// for (int y = -3; y <= 3; y++) +// for (int x = -3; x <= 3; x++) +// { +// packet.chunkX = x; +// packet.chunkY = y; +// packet.serialize(m_dataStream); +// } +// } // { // packets::play::clientbound::SpawnMob packet; // packet.entityId = 2; @@ -88,10 +88,14 @@ PlayClient::PlayClient(std::unique_ptr &&socket, Server &server) : PlayClient::~PlayClient() { + qDebug() << "called" << m_socket->readAll(); + + emit distributeChatMessage(ChatPart{.extra={ChatPart{.text=m_name,.color="red"},ChatPart{.text=tr(" left the game!")}}}.toString()); + m_server.remove(*this); } -void PlayClient::keepAlive() +void PlayClient::tick() { const auto now = QDateTime::currentDateTime(); if (!m_lastKeepAliveSent.isValid() || m_lastKeepAliveSent.secsTo(now) >= 1) @@ -101,24 +105,13 @@ void PlayClient::keepAlive() packet.serialize(m_dataStream); m_lastKeepAliveSent = now; } -} -void PlayClient::sendChatMessage() -{ - const auto now = QDateTime::currentDateTime(); if (!m_lastChatMessage.isValid() || m_lastChatMessage.secsTo(now) >= 2) { - packets::play::clientbound::ChatMessage packet; - packet.jsonData = normalText(tr("Time chaged to %0").arg(QTime::currentTime().toString())); - packet.position = packets::play::clientbound::ChatMessage::Chat; - packet.serialize(m_dataStream); + sendChatMessage(ChatPart{.text=tr("Time chaged to %0").arg(QTime::currentTime().toString())}.toString()); m_lastChatMessage = now; } -} -void PlayClient::randomizeStats() -{ - const auto now = QDateTime::currentDateTime(); if (!m_lastStats.isValid() || m_lastStats.msecsTo(now) >= 500) { { @@ -138,15 +131,11 @@ void PlayClient::randomizeStats() m_lastStats = now; } -} -void PlayClient::trialDisconnect() -{ - const auto now = QDateTime::currentDateTime(); if (m_connectedSince.secsTo(now) >= 30) { packets::play::clientbound::Disconnect packet; - packet.reason = boldText("Your trial has ended."); + packet.reason = ChatPart{.text="Your trial has ended.",.bold=true}.toString(); packet.serialize(m_dataStream); m_socket->flush(); m_dataStream.setDevice({}); @@ -155,6 +144,14 @@ void PlayClient::trialDisconnect() } } +void PlayClient::sendChatMessage(const QString &message) +{ + packets::play::clientbound::ChatMessage packet; + packet.jsonData = message; + packet.position = packets::play::clientbound::ChatMessage::Chat; + packet.serialize(m_dataStream); +} + void PlayClient::readyRead() { while(m_socket && m_socket->bytesAvailable()) @@ -189,7 +186,20 @@ void PlayClient::readPacket(packets::play::serverbound::PacketType type, const Q switch(type) { using namespace packets::play; - case serverbound::PacketType::ClientSettings: + using namespace serverbound; + case PacketType::ChatMessage: + { + qDebug() << type; + { + serverbound::ChatMessage packet{dataStream}; + qDebug() << "message" << packet.message; + const auto formatted = ChatPart{.extra={ChatPart{.text=m_name,.color="red"},ChatPart{.text=": " + packet.message}}}.toString(); + emit distributeChatMessage(formatted); + sendChatMessage(formatted); + } + break; + } + case PacketType::ClientSettings: { qDebug() << type; { diff --git a/playclient.h b/playclient.h index 63d3916..82126b4 100644 --- a/playclient.h +++ b/playclient.h @@ -15,13 +15,16 @@ class PlayClient : public QObject Q_OBJECT public: - explicit PlayClient(std::unique_ptr &&socket, Server &server); + explicit PlayClient(const QString &name, std::unique_ptr &&socket, Server &server); ~PlayClient() override; - void keepAlive(); - void sendChatMessage(); - void randomizeStats(); - void trialDisconnect(); + void tick(); + +signals: + void distributeChatMessage(const QString &message); + +public slots: + void sendChatMessage(const QString &message); private slots: void readyRead(); @@ -29,6 +32,7 @@ private slots: private: void readPacket(packets::play::serverbound::PacketType type, const QByteArray &buffer); + const QString m_name; std::unique_ptr m_socket; Server &m_server; diff --git a/server.cpp b/server.cpp index eeaad6d..d3172c1 100644 --- a/server.cpp +++ b/server.cpp @@ -23,6 +23,12 @@ Server::Server(QObject *parent) : void Server::add(PlayClient &playClient) { + for (auto otherClient : m_playClients) + { + connect(&playClient, &PlayClient::distributeChatMessage, otherClient, &PlayClient::sendChatMessage); + connect(otherClient, &PlayClient::distributeChatMessage, &playClient, &PlayClient::sendChatMessage); + } + m_playClients.insert(&playClient); } @@ -34,12 +40,7 @@ void Server::remove(PlayClient &playClient) void Server::timeout() { for (auto client : m_playClients) - { - client->keepAlive(); - client->sendChatMessage(); - client->randomizeStats(); - client->trialDisconnect(); - } + client->tick(); } void Server::newConnection() diff --git a/statusclient.cpp b/statusclient.cpp index 771669c..7bea0e5 100644 --- a/statusclient.cpp +++ b/statusclient.cpp @@ -11,8 +11,6 @@ StatusClient::StatusClient(std::unique_ptr &&socket, Server &server) : QObject{&server}, m_socket{std::move(socket)}, m_server{server}, m_dataStream{m_socket.get()} { - m_socket->setParent(this); - connect(m_socket.get(), &QIODevice::readyRead, this, &StatusClient::readyRead); connect(m_socket.get(), &QAbstractSocket::disconnected, this, &QObject::deleteLater); } @@ -73,7 +71,7 @@ void StatusClient::readPacket(packets::status::serverbound::PacketType type, con QJsonObject{{"name", "feedc0de"}, {"id", "6ebf7396-b6da-40b1-b7d9-9b7961450d5a"}} } } } }, - { "description", normalTextObj("Minecraft server implemented in C++") }, + { "description", ChatPart{.extra={ChatPart{.text="Minecraft", .color="red"}, ChatPart{.text=" server implemented in C++"}}}.toObject() }, { "favicon", "" } }}.toJson(); packet.serialize(m_dataStream);