From 6ed4909088077e2b28c5736069ab3b50c7d66f63 Mon Sep 17 00:00:00 2001 From: in1tiate Date: Wed, 14 Apr 2021 16:32:24 -0500 Subject: [PATCH 01/16] add ability to force immediate text processing --- bin/config_sample/areas.ini | 4 +++- include/aoclient.h | 15 +++++++++++++-- include/area_data.h | 5 +++++ src/area_data.cpp | 1 + src/commands.cpp | 10 +++++++++- src/packets.cpp | 10 ++++++---- 6 files changed, 37 insertions(+), 8 deletions(-) diff --git a/bin/config_sample/areas.ini b/bin/config_sample/areas.ini index 7dc04e7..066fcc8 100644 --- a/bin/config_sample/areas.ini +++ b/bin/config_sample/areas.ini @@ -4,10 +4,12 @@ protected_area=true iniswap_allowed=false evidence_mod=cm blankposting_allowed=true +force_immediate=true [1:Courtroom 1] background=gs4 protected_area=false iniswap_allowed=true evidence_mod=ffa -blankposting_allowed=true \ No newline at end of file +blankposting_allowed=true +force_immediate=false diff --git a/include/aoclient.h b/include/aoclient.h index d64c774..998acc1 100644 --- a/include/aoclient.h +++ b/include/aoclient.h @@ -1142,7 +1142,7 @@ class AOClient : public QObject { * * @iscommand */ - void cmdAllow_Blankposting(int argc, QStringList argv); + void cmdAllowBlankposting(int argc, QStringList argv); ///@} @@ -1528,6 +1528,15 @@ class AOClient : public QObject { */ void cmdBanInfo(int argc, QStringList argv); + /** + * @brief Toggles immediate text processing in the current area. + * + * @details No arguments. + * + * @iscommand + */ + void cmdForceImmediate(int argc, QStringList argv); + ///@} /** @@ -1751,7 +1760,8 @@ class AOClient : public QObject { {"8ball", {ACLFlags.value("NONE"), 1, &AOClient::cmd8Ball}}, {"lm", {ACLFlags.value("MODCHAT"), 1, &AOClient::cmdLM}}, {"judgelog", {ACLFlags.value("CM"), 0, &AOClient::cmdJudgeLog}}, - {"allow_blankposting", {ACLFlags.value("MODCHAT"), 0, &AOClient::cmdAllow_Blankposting}}, + {"allowblankposting", {ACLFlags.value("MODCHAT"), 0, &AOClient::cmdAllowBlankposting}}, + {"allow_blankposting", {ACLFlags.value("MODCHAT"), 0, &AOClient::cmdAllowBlankposting}}, {"gimp", {ACLFlags.value("MUTE"), 1, &AOClient::cmdGimp}}, {"ungimp", {ACLFlags.value("MUTE"), 1, &AOClient::cmdUnGimp}}, {"baninfo", {ACLFlags.value("BAN"), 1, &AOClient::cmdBanInfo}}, @@ -1765,6 +1775,7 @@ class AOClient : public QObject { {"undisemvowel", {ACLFlags.value("MUTE"), 1, &AOClient::cmdUnDisemvowel}}, {"shake", {ACLFlags.value("MUTE"), 1, &AOClient::cmdShake}}, {"unshake", {ACLFlags.value("MUTE"), 1, &AOClient::cmdUnShake}}, + {"forceimmediate", {ACLFlags.value("CM"), 1, &AOClient::cmdForceImmediate}}, }; /** diff --git a/include/area_data.h b/include/area_data.h index 5cab8ef..23df38f 100644 --- a/include/area_data.h +++ b/include/area_data.h @@ -344,6 +344,11 @@ class AreaData : public QObject { * @brief The value of logger in config.ini. */ QString log_type; + + /** + * @brief Whether or not to force immediate text processing in this area + */ + bool force_immediate; }; #endif // AREA_DATA_H diff --git a/src/area_data.cpp b/src/area_data.cpp index c40271c..0653597 100644 --- a/src/area_data.cpp +++ b/src/area_data.cpp @@ -39,6 +39,7 @@ AreaData::AreaData(QString p_name, int p_index) : bg_locked = areas_ini.value("bg_locked", "false").toBool(); QString configured_evi_mod = areas_ini.value("evidence_mod", "FFA").toString().toLower(); blankposting_allowed = areas_ini.value("blankposting_allowed","true").toBool(); + force_immediate = areas_ini.value("force_immediate", "false").toBool(); areas_ini.endGroup(); QSettings config_ini("config/config.ini", QSettings::IniFormat); config_ini.beginGroup("Options"); diff --git a/src/commands.cpp b/src/commands.cpp index e6dd316..d7473ea 100644 --- a/src/commands.cpp +++ b/src/commands.cpp @@ -1276,7 +1276,7 @@ void AOClient::cmdJudgeLog(int argc, QStringList argv) } } -void AOClient::cmdAllow_Blankposting(int argc, QStringList argv) +void AOClient::cmdAllowBlankposting(int argc, QStringList argv) { QString sender_name = ooc_name; AreaData* area = server->areas[current_area]; @@ -1521,6 +1521,14 @@ void AOClient::cmdUnShake(int argc, QStringList argv) target->is_shaken = false; } +void AOClient::cmdForceImmediate(int argc, QStringList argv) +{ + AreaData* area = server->areas[current_area]; + area->force_immediate = !area->force_immediate; + QString state = area->force_immediate ? "on." : "off."; + sendServerMessage("Forced immediate text processing in this area is now " + state); +} + QStringList AOClient::buildAreaList(int area_idx) { QStringList entries; diff --git a/src/packets.cpp b/src/packets.cpp index f271216..51b32c3 100644 --- a/src/packets.cpp +++ b/src/packets.cpp @@ -594,11 +594,13 @@ AOPacket AOClient::validateIcPacket(AOPacket packet) } args.append(other_flip); - // noninterrupting preanim - int ni_pa = incoming_args[18].toInt(); - if (ni_pa != 1 && ni_pa != 0) + // immediate text processing + int immediate = incoming_args[18].toInt(); + if (area->force_immediate) + immediate = 1; + if (immediate != 1 && immediate != 0) return invalid; - args.append(QString::number(ni_pa)); + args.append(QString::number(immediate)); } // 2.8 packet extensions From 89c96929249db736a1a77026a59b70238a4255ad Mon Sep 17 00:00:00 2001 From: in1tiate Date: Wed, 14 Apr 2021 16:34:25 -0500 Subject: [PATCH 02/16] add old noint_pres alias --- include/aoclient.h | 1 + 1 file changed, 1 insertion(+) diff --git a/include/aoclient.h b/include/aoclient.h index 998acc1..e438a28 100644 --- a/include/aoclient.h +++ b/include/aoclient.h @@ -1776,6 +1776,7 @@ class AOClient : public QObject { {"shake", {ACLFlags.value("MUTE"), 1, &AOClient::cmdShake}}, {"unshake", {ACLFlags.value("MUTE"), 1, &AOClient::cmdUnShake}}, {"forceimmediate", {ACLFlags.value("CM"), 1, &AOClient::cmdForceImmediate}}, + {"force_noint_pres", {ACLFlags.value("CM"), 1, &AOClient::cmdForceImmediate}}, }; /** From 461c4f9d6ebc311e18852a1f8cbad7eafecf9c8e Mon Sep 17 00:00:00 2001 From: in1tiate Date: Wed, 14 Apr 2021 16:38:38 -0500 Subject: [PATCH 03/16] add iniswap toggle command --- include/aoclient.h | 11 +++++++++++ src/commands.cpp | 8 ++++++++ 2 files changed, 19 insertions(+) diff --git a/include/aoclient.h b/include/aoclient.h index e438a28..a33deba 100644 --- a/include/aoclient.h +++ b/include/aoclient.h @@ -1537,6 +1537,15 @@ class AOClient : public QObject { */ void cmdForceImmediate(int argc, QStringList argv); + /** + * @brief Toggles whether iniswaps are allowed in the current area. + * + * @details No arguments. + * + * @iscommand + */ + void cmdAllowIniswap(int argc, QStringList argv); + ///@} /** @@ -1777,6 +1786,8 @@ class AOClient : public QObject { {"unshake", {ACLFlags.value("MUTE"), 1, &AOClient::cmdUnShake}}, {"forceimmediate", {ACLFlags.value("CM"), 1, &AOClient::cmdForceImmediate}}, {"force_noint_pres", {ACLFlags.value("CM"), 1, &AOClient::cmdForceImmediate}}, + {"allowiniswap", {ACLFlags.value("CM"), 1, &AOClient::cmdAllowIniswap}}, + {"allow_iniswap", {ACLFlags.value("CM"), 1, &AOClient::cmdAllowIniswap}}, }; /** diff --git a/src/commands.cpp b/src/commands.cpp index d7473ea..bf3e5ff 100644 --- a/src/commands.cpp +++ b/src/commands.cpp @@ -1529,6 +1529,14 @@ void AOClient::cmdForceImmediate(int argc, QStringList argv) sendServerMessage("Forced immediate text processing in this area is now " + state); } +void AOClient::cmdAllowIniswap(int argc, QStringList argv) +{ + AreaData* area = server->areas[current_area]; + area->iniswap_allowed = !area->iniswap_allowed; + QString state = area->iniswap_allowed ? "allowed." : "disallowed."; + sendServerMessage("Iniswapping in this area is now " + state); +} + QStringList AOClient::buildAreaList(int area_idx) { QStringList entries; From 78c51cad419fd81e6171d92af00f12fb423d98e4 Mon Sep 17 00:00:00 2001 From: MangosArentLiterature <58055358+MangosArentLiterature@users.noreply.github.com> Date: Thu, 15 Apr 2021 00:59:22 -0500 Subject: [PATCH 04/16] Fix cmdAddStatement - Fix cmdAddStatement not being in the command table - Fix add statement not correctly setting added statements to green text --- include/aoclient.h | 1 + src/testimony_recorder.cpp | 1 + 2 files changed, 2 insertions(+) diff --git a/include/aoclient.h b/include/aoclient.h index d64c774..6626972 100644 --- a/include/aoclient.h +++ b/include/aoclient.h @@ -1760,6 +1760,7 @@ class AOClient : public QObject { {"pause", {ACLFlags.value("CM"), 0, &AOClient::cmdPauseTestimony}}, {"delete", {ACLFlags.value("CM"), 0, &AOClient::cmdDeleteStatement}}, {"update", {ACLFlags.value("CM"), 0, &AOClient::cmdUpdateStatement}}, + {"add", {ACLFlags.value("CM"), 0, &AOClient::cmdAddStatement}}, {"reload", {ACLFlags.value("SUPER"), 0, &AOClient::cmdReload}}, {"disemvowel", {ACLFlags.value("MUTE"), 1, &AOClient::cmdDisemvowel}}, {"undisemvowel", {ACLFlags.value("MUTE"), 1, &AOClient::cmdUnDisemvowel}}, diff --git a/src/testimony_recorder.cpp b/src/testimony_recorder.cpp index 3c6897d..01178b5 100644 --- a/src/testimony_recorder.cpp +++ b/src/testimony_recorder.cpp @@ -39,6 +39,7 @@ void AOClient::addStatement(QStringList packet) } } else if (area->test_rec == AreaData::TestimonyRecording::ADD) { + packet[14] = "1"; area->testimony.insert(c_statement,packet); area->test_rec = AreaData::TestimonyRecording::PLAYBACK; } From f0412e0e05cbe5508ea5a93b1470ef14c9462884 Mon Sep 17 00:00:00 2001 From: in1tiate Date: Thu, 15 Apr 2021 06:34:31 -0500 Subject: [PATCH 05/16] add AFK system --- include/aoclient.h | 40 +++++++++++++++++++++++++++++++++++----- include/server.h | 5 +++++ src/aoclient.cpp | 15 ++++++++++++++- src/commands.cpp | 9 +++++++++ src/server.cpp | 10 +++++++--- 5 files changed, 70 insertions(+), 9 deletions(-) diff --git a/include/aoclient.h b/include/aoclient.h index 29eab62..598ead6 100644 --- a/include/aoclient.h +++ b/include/aoclient.h @@ -53,7 +53,10 @@ class AOClient : public QObject { AOClient(Server* p_server, QTcpSocket* p_socket, QObject* parent = nullptr, int user_id = 0) : QObject(parent), id(user_id), remote_ip(p_socket->peerAddress()), password(""), joined(false), current_area(0), current_char(""), socket(p_socket), server(p_server), - is_partial(false), last_wtce_time(0) {}; + is_partial(false), last_wtce_time(0) { + afk_timer = new QTimer; + connect(afk_timer, SIGNAL(timeout()), this, SLOT(onAfkTimeout())); + }; /** * @brief Destructor for the AOClient instance. @@ -233,6 +236,18 @@ class AOClient : public QObject { */ bool is_gimped = false; + /** + * @brief If true, the client will be marked as AFK in /getarea. Automatically applied when a configurable + * amount of time has passed since the last interaction, or manually applied by /afk. + */ + bool is_afk = false; + + /** + * @brief Timer for tracking user interaction. Automatically restarted whenever a user interacts (i.e. sends any packet besides CH) + */ + QTimer* afk_timer; + + public slots: /** * @brief A slot for when the client disconnects from the server. @@ -261,6 +276,11 @@ class AOClient : public QObject { */ void sendPacket(QString header); + /** + * @brief A slot for when the client's AFK timer runs out. + */ + void onAfkTimeout(); + private: /** * @brief The TCP socket used to communicate with the client. @@ -1546,6 +1566,15 @@ class AOClient : public QObject { */ void cmdAllowIniswap(int argc, QStringList argv); + /** + * @brief Toggles whether this client is considered AFK. + * + * @details No arguments. + * + * @iscommand + */ + void cmdAfk(int argc, QStringList argv); + ///@} /** @@ -1785,10 +1814,11 @@ class AOClient : public QObject { {"undisemvowel", {ACLFlags.value("MUTE"), 1, &AOClient::cmdUnDisemvowel}}, {"shake", {ACLFlags.value("MUTE"), 1, &AOClient::cmdShake}}, {"unshake", {ACLFlags.value("MUTE"), 1, &AOClient::cmdUnShake}}, - {"forceimmediate", {ACLFlags.value("CM"), 1, &AOClient::cmdForceImmediate}}, - {"force_noint_pres", {ACLFlags.value("CM"), 1, &AOClient::cmdForceImmediate}}, - {"allowiniswap", {ACLFlags.value("CM"), 1, &AOClient::cmdAllowIniswap}}, - {"allow_iniswap", {ACLFlags.value("CM"), 1, &AOClient::cmdAllowIniswap}}, + {"forceimmediate", {ACLFlags.value("CM"), 0, &AOClient::cmdForceImmediate}}, + {"force_noint_pres", {ACLFlags.value("CM"), 0, &AOClient::cmdForceImmediate}}, + {"allowiniswap", {ACLFlags.value("CM"), 0, &AOClient::cmdAllowIniswap}}, + {"allow_iniswap", {ACLFlags.value("CM"), 0, &AOClient::cmdAllowIniswap}}, + {"afk", {ACLFlags.value("NONE"), 0, &AOClient::cmdAfk}}, }; /** diff --git a/include/server.h b/include/server.h index 91fb5ef..0017621 100644 --- a/include/server.h +++ b/include/server.h @@ -232,6 +232,11 @@ class Server : public QObject { */ int max_dice; + /** + * @brief The amount of time in seconds to wait before marking a user AFK. + */ + int afk_timeout; + /** * @brief The server-wide global timer. */ diff --git a/src/aoclient.cpp b/src/aoclient.cpp index 762d99f..3136dd5 100644 --- a/src/aoclient.cpp +++ b/src/aoclient.cpp @@ -77,6 +77,13 @@ void AOClient::handlePacket(AOPacket packet) return; } + if (packet.header != "CH") { + if (is_afk) + sendServerMessage("You are no longer AFK."); + is_afk = false; + afk_timer->start(server->afk_timeout * 1000); + } + if (packet.contents.length() < info.minArgs) { #ifdef NET_DEBUG qDebug() << "Invalid packet args length. Minimum is" << info.minArgs << "but only" << packet.contents.length() << "were given."; @@ -313,7 +320,13 @@ bool AOClient::checkAuth(unsigned long long acl_mask) QString AOClient::getIpid() { return ipid; } -Server* AOClient::getServer() { return server; }; +Server* AOClient::getServer() { return server; } + +void AOClient::onAfkTimeout() +{ + is_afk = true; + sendServerMessage("You are now AFK."); +} AOClient::~AOClient() { socket->deleteLater(); diff --git a/src/commands.cpp b/src/commands.cpp index bf3e5ff..eb45cde 100644 --- a/src/commands.cpp +++ b/src/commands.cpp @@ -1537,6 +1537,13 @@ void AOClient::cmdAllowIniswap(int argc, QStringList argv) sendServerMessage("Iniswapping in this area is now " + state); } +void AOClient::cmdAfk(int argc, QStringList argv) +{ + is_afk = !is_afk; + QString state = is_afk ? "now" : "no longer"; + sendServerMessage("You are " + state + " AFK."); +} + QStringList AOClient::buildAreaList(int area_idx) { QStringList entries; @@ -1564,6 +1571,8 @@ QStringList AOClient::buildAreaList(int area_idx) char_entry.insert(0, "[CM] "); if (authenticated) char_entry += " (" + client->getIpid() + "): " + client->ooc_name; + if (client->is_afk) + char_entry += " [AFK]"; entries.append(char_entry); } } diff --git a/src/server.cpp b/src/server.cpp index 48d2787..10dbd66 100644 --- a/src/server.cpp +++ b/src/server.cpp @@ -232,9 +232,13 @@ void Server::loadServerConfig() auth_type = config.value("auth","simple").toString(); modpass = config.value("modpass","").toString(); bool zalgo_tolerance_conversion_success; - zalgo_tolerance = config.value("zalgo_tolerance", "3").toInt(&zalgo_tolerance_conversion_success); - if (!zalgo_tolerance_conversion_success) - zalgo_tolerance = 3; + zalgo_tolerance = config.value("zalgo_tolerance", "3").toInt(&zalgo_tolerance_conversion_success); + if (!zalgo_tolerance_conversion_success) + zalgo_tolerance = 3; + bool afk_timeout_conversion_success; + afk_timeout = config.value("afk_timeout", "300").toInt(&afk_timeout_conversion_success); + if (!afk_timeout_conversion_success) + afk_timeout = 300; config.endGroup(); //Load dice values From 42ca1d277912305f98debea0efdcc0f0ef375238 Mon Sep 17 00:00:00 2001 From: in1tiate Date: Thu, 15 Apr 2021 06:40:07 -0500 Subject: [PATCH 06/16] fix AFK message being sent repeatedly --- src/aoclient.cpp | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/aoclient.cpp b/src/aoclient.cpp index 3136dd5..b425683 100644 --- a/src/aoclient.cpp +++ b/src/aoclient.cpp @@ -324,8 +324,9 @@ Server* AOClient::getServer() { return server; } void AOClient::onAfkTimeout() { + if (!is_afk) + sendServerMessage("You are now AFK."); is_afk = true; - sendServerMessage("You are now AFK."); } AOClient::~AOClient() { From 96f2204b79d062e035dd094c2df1038a762175a6 Mon Sep 17 00:00:00 2001 From: in1tiate Date: Thu, 15 Apr 2021 06:42:14 -0500 Subject: [PATCH 07/16] more elegantly fix repeated AFKing --- src/aoclient.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/aoclient.cpp b/src/aoclient.cpp index b425683..55f1a58 100644 --- a/src/aoclient.cpp +++ b/src/aoclient.cpp @@ -324,9 +324,9 @@ Server* AOClient::getServer() { return server; } void AOClient::onAfkTimeout() { - if (!is_afk) - sendServerMessage("You are now AFK."); + sendServerMessage("You are now AFK."); is_afk = true; + afk_timer->stop(); } AOClient::~AOClient() { From a32b33d957d7d305d94d705bc2867678c37849fd Mon Sep 17 00:00:00 2001 From: in1tiate Date: Thu, 15 Apr 2021 06:44:14 -0500 Subject: [PATCH 08/16] even more elegant repeat afk fix --- include/aoclient.h | 1 + src/aoclient.cpp | 1 - 2 files changed, 1 insertion(+), 1 deletion(-) diff --git a/include/aoclient.h b/include/aoclient.h index 598ead6..be39612 100644 --- a/include/aoclient.h +++ b/include/aoclient.h @@ -55,6 +55,7 @@ class AOClient : public QObject { joined(false), current_area(0), current_char(""), socket(p_socket), server(p_server), is_partial(false), last_wtce_time(0) { afk_timer = new QTimer; + afk_timer->setSingleShot(true); connect(afk_timer, SIGNAL(timeout()), this, SLOT(onAfkTimeout())); }; diff --git a/src/aoclient.cpp b/src/aoclient.cpp index 55f1a58..7e37fef 100644 --- a/src/aoclient.cpp +++ b/src/aoclient.cpp @@ -326,7 +326,6 @@ void AOClient::onAfkTimeout() { sendServerMessage("You are now AFK."); is_afk = true; - afk_timer->stop(); } AOClient::~AOClient() { From 9c6fa22ee5f3277965339b2fe93a54def9e1aef3 Mon Sep 17 00:00:00 2001 From: in1tiate Date: Thu, 15 Apr 2021 06:48:07 -0500 Subject: [PATCH 09/16] fix being told youre afk twice if you use /afk before the timer runs out --- src/aoclient.cpp | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/aoclient.cpp b/src/aoclient.cpp index 7e37fef..b425683 100644 --- a/src/aoclient.cpp +++ b/src/aoclient.cpp @@ -324,7 +324,8 @@ Server* AOClient::getServer() { return server; } void AOClient::onAfkTimeout() { - sendServerMessage("You are now AFK."); + if (!is_afk) + sendServerMessage("You are now AFK."); is_afk = true; } From 1a7c23f404efb10ceaa72cf38f3b9c1ab1309c01 Mon Sep 17 00:00:00 2001 From: in1tiate Date: Thu, 15 Apr 2021 06:49:09 -0500 Subject: [PATCH 10/16] simplify /afk to just set is_afk to true previously it would toggle is_afk, but because sending the command sends a CT packet, you could never toggle AFK off with /afk because it would have been toggled off due to activity immediately prior to using it --- src/commands.cpp | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/src/commands.cpp b/src/commands.cpp index eb45cde..135c06a 100644 --- a/src/commands.cpp +++ b/src/commands.cpp @@ -1539,9 +1539,8 @@ void AOClient::cmdAllowIniswap(int argc, QStringList argv) void AOClient::cmdAfk(int argc, QStringList argv) { - is_afk = !is_afk; - QString state = is_afk ? "now" : "no longer"; - sendServerMessage("You are " + state + " AFK."); + is_afk = true; + sendServerMessage("You are now AFK."); } QStringList AOClient::buildAreaList(int area_idx) From 12bba40a991420585419332170eb7d9614194ca7 Mon Sep 17 00:00:00 2001 From: in1tiate Date: Thu, 15 Apr 2021 10:25:25 -0500 Subject: [PATCH 11/16] implement case announcements --- include/aoclient.h | 29 ++++++++++++++++ src/aoclient.cpp | 1 + src/packets.cpp | 83 ++++++++++++++++++++++++++++++++++++++++++++-- 3 files changed, 111 insertions(+), 2 deletions(-) diff --git a/include/aoclient.h b/include/aoclient.h index 29eab62..a99c8e6 100644 --- a/include/aoclient.h +++ b/include/aoclient.h @@ -218,6 +218,27 @@ class AOClient : public QObject { {"SUPER", ~0ULL }, }; + + /** + * @brief A structure for storing the client's casing alert preferences. + */ + struct CasingPreferences { + QString caselist = ""; //!< The list of cases this user is willing to host (assuming they are also willing to CM) (unused) + bool cm = false; //!< If the user is willing to host cases (unused) + bool defense = false; //!< If the user is willing to defend a case / play as a defense attorney (or a co-defense attorney) + bool prosecution = false; //!< If the user is willing to prosecute a case / play as a prosecutor (or a co-prosecutor) + bool judge = false; //!< If the user is willing to judge a case + bool jury = false; //!< If the user is willing to be a member of the jury in a case + bool stenographer = false; //!< If the user is willing to be the stenographer of a case + }; + + /** + * @brief The client's casing alert preferences. + * + * @see The struct itself for more details. + */ + CasingPreferences casing_preferences; + /** * @brief If true, the client's in-character messages will have their word order randomised. */ @@ -472,6 +493,12 @@ class AOClient : public QObject { /// Implements [editing evidence](https://github.com/AttorneyOnline/docs/blob/master/docs/development/network.md#edit). void pktEditEvidence(AreaData* area, int argc, QStringList argv, AOPacket packet); + /// Implements [updating casing preferences](https://github.com/AttorneyOnline/docs/blob/master/docs/development/network.md#case-preferences-update). + void pktSetCase(AreaData* area, int argc, QStringList argv, AOPacket packet); + + /// Implements [announcing a case](https://github.com/AttorneyOnline/docs/blob/master/docs/development/network.md#case-alert). + void pktAnnounceCase(AreaData* area, int argc, QStringList argv, AOPacket packet); + ///@} /** @@ -622,6 +649,8 @@ class AOClient : public QObject { {"PE", {ACLFlags.value("NONE"), 3, &AOClient::pktAddEvidence }}, {"DE", {ACLFlags.value("NONE"), 1, &AOClient::pktRemoveEvidence }}, {"EE", {ACLFlags.value("NONE"), 4, &AOClient::pktEditEvidence }}, + {"SETCASE", {ACLFlags.value("NONE"), 7, &AOClient::pktSetCase }}, + {"CASEA", {ACLFlags.value("NONE"), 6, &AOClient::pktAnnounceCase }}, }; /** diff --git a/src/aoclient.cpp b/src/aoclient.cpp index 762d99f..4baacec 100644 --- a/src/aoclient.cpp +++ b/src/aoclient.cpp @@ -311,6 +311,7 @@ bool AOClient::checkAuth(unsigned long long acl_mask) return true; } + QString AOClient::getIpid() { return ipid; } Server* AOClient::getServer() { return server; }; diff --git a/src/packets.cpp b/src/packets.cpp index 51b32c3..221fa56 100644 --- a/src/packets.cpp +++ b/src/packets.cpp @@ -344,6 +344,87 @@ void AOClient::pktEditEvidence(AreaData* area, int argc, QStringList argv, AOPac sendEvidenceList(area); } +void AOClient::pktSetCase(AreaData* area, int argc, QStringList argv, AOPacket packet) +{ + casing_preferences.caselist = argv[0]; + QList prefs_list; + for (int i = 1; i <=6; i++) { + bool is_int = false; + bool pref = argv[i].toInt(&is_int); + if (!is_int) + return; + prefs_list.append(pref); + } + casing_preferences.cm = prefs_list[0]; + casing_preferences.defense = prefs_list[1]; + casing_preferences.prosecution = prefs_list[2]; + casing_preferences.judge = prefs_list[3]; + casing_preferences.jury = prefs_list[4]; + casing_preferences.stenographer = prefs_list[5]; + + qDebug() << casing_preferences.cm << casing_preferences.defense << casing_preferences.prosecution << casing_preferences.judge << casing_preferences.jury << casing_preferences.stenographer; +} + +void AOClient::pktAnnounceCase(AreaData* area, int argc, QStringList argv, AOPacket packet) +{ + // the following is an example of the type of code AO2 makes me write + // you may wish to do something more pleasant rather than read this garbage + // taking a bath in battery acid, for instance + QString case_title = argv[0]; + QStringList needed_roles; + QList needs_list; + for (int i = 1; i <=5; i++) { + bool is_int = false; + bool need = argv[i].toInt(&is_int); + if (!is_int) + return; + needs_list.append(need); + } + // this is stupid stupid stupid i hate this + if (needs_list[0]) { + needed_roles.append("defense attorney"); + } + if (needs_list[1]) { + needed_roles.append("prosecutor"); + } + if (needs_list[2]) { + needed_roles.append("judge"); + } + if (needs_list[3]) { + needed_roles.append("jurors"); + } + if (needs_list[4]) { + needed_roles.append("stenographer"); + } + if (needed_roles.isEmpty()) { + return; + } + + QString message = "=== Case Announcement ===\r\n" + ooc_name + " needs " + needed_roles.join(", ") + " for " + case_title + "!"; + + QList clients_to_alert; + // this is morton the indented if statement + // please do not feed morton + for (AOClient* client : server->clients) { + if (((client->casing_preferences.defense && needed_roles.contains("defense attorney")) || + (client->casing_preferences.prosecution && needed_roles.contains("prosecutor")) || + (client->casing_preferences.judge && needed_roles.contains("judge")) || + (client->casing_preferences.jury && needed_roles.contains("jurors")) || + (client->casing_preferences.stenographer && needed_roles.contains("stenographer"))) + && !clients_to_alert.contains(client)) + clients_to_alert.append(client); + } + // morton is a little ugly but we love him anyway + + for (AOClient* client : clients_to_alert) { + client->sendPacket(AOPacket("CASEA", {message, argv[1], argv[2], argv[3], argv[4], argv[5], "1"})); + // you may be thinking, "hey wait a minute the network protocol documentation doesn't mention that last argument!" + // if you are in fact thinking that, you are correct! it is not in the documentation! + // however for some inscrutable reason Attorney Online 2 will outright reject a CASEA packet that does not have + // at least 7 arguments despite only using the first 6. Cera, i kneel. you have truly broken me. + } +} + void AOClient::sendEvidenceList(AreaData* area) { for (AOClient* client : server->clients) { @@ -719,6 +800,4 @@ void AOClient::updateJudgeLog(AreaData* area, AOClient* client, QString action) area->judgelog.append(logmessage); } else area->judgelog.append(logmessage); - - } From c9700f236f9de518c1ac609ee0ef2ffcf0df36c3 Mon Sep 17 00:00:00 2001 From: in1tiate Date: Thu, 15 Apr 2021 10:26:46 -0500 Subject: [PATCH 12/16] more cleanly handle empty input, snip debug call --- src/packets.cpp | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) diff --git a/src/packets.cpp b/src/packets.cpp index 221fa56..15823cb 100644 --- a/src/packets.cpp +++ b/src/packets.cpp @@ -361,8 +361,6 @@ void AOClient::pktSetCase(AreaData* area, int argc, QStringList argv, AOPacket p casing_preferences.judge = prefs_list[3]; casing_preferences.jury = prefs_list[4]; casing_preferences.stenographer = prefs_list[5]; - - qDebug() << casing_preferences.cm << casing_preferences.defense << casing_preferences.prosecution << casing_preferences.judge << casing_preferences.jury << casing_preferences.stenographer; } void AOClient::pktAnnounceCase(AreaData* area, int argc, QStringList argv, AOPacket packet) @@ -400,7 +398,7 @@ void AOClient::pktAnnounceCase(AreaData* area, int argc, QStringList argv, AOPac return; } - QString message = "=== Case Announcement ===\r\n" + ooc_name + " needs " + needed_roles.join(", ") + " for " + case_title + "!"; + QString message = "=== Case Announcement ===\r\n" + ooc_name == "" ? current_char : ooc_name + " needs " + needed_roles.join(", ") + " for " + case_title == "" ? "a case" : case_title + "!"; QList clients_to_alert; // this is morton the indented if statement From 06b70a35d520a42c0e4996488e2539c657188eef Mon Sep 17 00:00:00 2001 From: MangosArentLiterature <58055358+MangosArentLiterature@users.noreply.github.com> Date: Thu, 15 Apr 2021 11:53:20 -0500 Subject: [PATCH 13/16] Make maximum_statements actually get loaded in Also has the effect of making it changeable with /reload, hooray! --- src/server.cpp | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/src/server.cpp b/src/server.cpp index 48d2787..f02c918 100644 --- a/src/server.cpp +++ b/src/server.cpp @@ -53,9 +53,7 @@ void Server::start() loadServerConfig(); loadCommandConfig(); - - maximum_statements = config.value("maximum_statements", 50).toInt(); - + proxy = new WSProxy(port, ws_port, this); if(ws_port != -1) proxy->start(); @@ -235,6 +233,7 @@ void Server::loadServerConfig() zalgo_tolerance = config.value("zalgo_tolerance", "3").toInt(&zalgo_tolerance_conversion_success); if (!zalgo_tolerance_conversion_success) zalgo_tolerance = 3; + maximum_statements = config.value("maximum_statements", 10).toInt(); config.endGroup(); //Load dice values From 122e993a8bda2b02a631b4f95abe8ea684662d2b Mon Sep 17 00:00:00 2001 From: in1tiate Date: Thu, 15 Apr 2021 13:49:27 -0500 Subject: [PATCH 14/16] condense some overly long code (ty marisa) --- src/packets.cpp | 22 +++++----------------- 1 file changed, 5 insertions(+), 17 deletions(-) diff --git a/src/packets.cpp b/src/packets.cpp index 15823cb..c2f77e0 100644 --- a/src/packets.cpp +++ b/src/packets.cpp @@ -378,25 +378,13 @@ void AOClient::pktAnnounceCase(AreaData* area, int argc, QStringList argv, AOPac return; needs_list.append(need); } - // this is stupid stupid stupid i hate this - if (needs_list[0]) { - needed_roles.append("defense attorney"); + QStringList roles = {"defense attorney", "prosecutor", "judge", "jurors", "stenographer"}; + for (int i = 0; i < 5; i++) { + if (needs_list[i]) + needed_roles.append(roles[i]); } - if (needs_list[1]) { - needed_roles.append("prosecutor"); - } - if (needs_list[2]) { - needed_roles.append("judge"); - } - if (needs_list[3]) { - needed_roles.append("jurors"); - } - if (needs_list[4]) { - needed_roles.append("stenographer"); - } - if (needed_roles.isEmpty()) { + if (needed_roles.isEmpty()) return; - } QString message = "=== Case Announcement ===\r\n" + ooc_name == "" ? current_char : ooc_name + " needs " + needed_roles.join(", ") + " for " + case_title == "" ? "a case" : case_title + "!"; From b1dfeec8f5a92c89bc77c6ba0b8e5de01ab76254 Mon Sep 17 00:00:00 2001 From: in1tiate Date: Thu, 15 Apr 2021 13:58:41 -0500 Subject: [PATCH 15/16] euthanize morton, code cleanup --- include/aoclient.h | 19 ++----------------- src/packets.cpp | 25 +++++-------------------- 2 files changed, 7 insertions(+), 37 deletions(-) diff --git a/include/aoclient.h b/include/aoclient.h index a99c8e6..79bcbf1 100644 --- a/include/aoclient.h +++ b/include/aoclient.h @@ -220,24 +220,9 @@ class AOClient : public QObject { /** - * @brief A structure for storing the client's casing alert preferences. + * @brief A list of 5 casing preferences (def, pro, judge, jury, steno) */ - struct CasingPreferences { - QString caselist = ""; //!< The list of cases this user is willing to host (assuming they are also willing to CM) (unused) - bool cm = false; //!< If the user is willing to host cases (unused) - bool defense = false; //!< If the user is willing to defend a case / play as a defense attorney (or a co-defense attorney) - bool prosecution = false; //!< If the user is willing to prosecute a case / play as a prosecutor (or a co-prosecutor) - bool judge = false; //!< If the user is willing to judge a case - bool jury = false; //!< If the user is willing to be a member of the jury in a case - bool stenographer = false; //!< If the user is willing to be the stenographer of a case - }; - - /** - * @brief The client's casing alert preferences. - * - * @see The struct itself for more details. - */ - CasingPreferences casing_preferences; + QList casing_preferences = {false, false, false, false, false}; /** * @brief If true, the client's in-character messages will have their word order randomised. diff --git a/src/packets.cpp b/src/packets.cpp index c2f77e0..9790289 100644 --- a/src/packets.cpp +++ b/src/packets.cpp @@ -346,28 +346,19 @@ void AOClient::pktEditEvidence(AreaData* area, int argc, QStringList argv, AOPac void AOClient::pktSetCase(AreaData* area, int argc, QStringList argv, AOPacket packet) { - casing_preferences.caselist = argv[0]; QList prefs_list; - for (int i = 1; i <=6; i++) { + for (int i = 2; i <=6; i++) { bool is_int = false; bool pref = argv[i].toInt(&is_int); if (!is_int) return; prefs_list.append(pref); } - casing_preferences.cm = prefs_list[0]; - casing_preferences.defense = prefs_list[1]; - casing_preferences.prosecution = prefs_list[2]; - casing_preferences.judge = prefs_list[3]; - casing_preferences.jury = prefs_list[4]; - casing_preferences.stenographer = prefs_list[5]; + casing_preferences = prefs_list; } void AOClient::pktAnnounceCase(AreaData* area, int argc, QStringList argv, AOPacket packet) { - // the following is an example of the type of code AO2 makes me write - // you may wish to do something more pleasant rather than read this garbage - // taking a bath in battery acid, for instance QString case_title = argv[0]; QStringList needed_roles; QList needs_list; @@ -389,18 +380,12 @@ void AOClient::pktAnnounceCase(AreaData* area, int argc, QStringList argv, AOPac QString message = "=== Case Announcement ===\r\n" + ooc_name == "" ? current_char : ooc_name + " needs " + needed_roles.join(", ") + " for " + case_title == "" ? "a case" : case_title + "!"; QList clients_to_alert; - // this is morton the indented if statement - // please do not feed morton + // here lies morton, RIP for (AOClient* client : server->clients) { - if (((client->casing_preferences.defense && needed_roles.contains("defense attorney")) || - (client->casing_preferences.prosecution && needed_roles.contains("prosecutor")) || - (client->casing_preferences.judge && needed_roles.contains("judge")) || - (client->casing_preferences.jury && needed_roles.contains("jurors")) || - (client->casing_preferences.stenographer && needed_roles.contains("stenographer"))) - && !clients_to_alert.contains(client)) + QSet matches = client->casing_preferences.toSet().intersect(needs_list.toSet()); + if (matches.isEmpty() && !clients_to_alert.contains(client)) clients_to_alert.append(client); } - // morton is a little ugly but we love him anyway for (AOClient* client : clients_to_alert) { client->sendPacket(AOPacket("CASEA", {message, argv[1], argv[2], argv[3], argv[4], argv[5], "1"})); From 38d73d92289303c5af79955cb4e97be85cb42b78 Mon Sep 17 00:00:00 2001 From: in1tiate Date: Thu, 15 Apr 2021 14:10:53 -0500 Subject: [PATCH 16/16] cleanup --- src/packets.cpp | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/src/packets.cpp b/src/packets.cpp index 9790289..6ddce8a 100644 --- a/src/packets.cpp +++ b/src/packets.cpp @@ -377,13 +377,14 @@ void AOClient::pktAnnounceCase(AreaData* area, int argc, QStringList argv, AOPac if (needed_roles.isEmpty()) return; - QString message = "=== Case Announcement ===\r\n" + ooc_name == "" ? current_char : ooc_name + " needs " + needed_roles.join(", ") + " for " + case_title == "" ? "a case" : case_title + "!"; + QString message = "=== Case Announcement ===\r\n" + (ooc_name == "" ? current_char : ooc_name) + " needs " + needed_roles.join(", ") + " for " + (case_title == "" ? "a case" : case_title) + "!"; QList clients_to_alert; // here lies morton, RIP + QSet needs_set = needs_list.toSet(); for (AOClient* client : server->clients) { - QSet matches = client->casing_preferences.toSet().intersect(needs_list.toSet()); - if (matches.isEmpty() && !clients_to_alert.contains(client)) + QSet matches = client->casing_preferences.toSet().intersect(needs_set); + if (!matches.isEmpty() && !clients_to_alert.contains(client)) clients_to_alert.append(client); }