From f0412e0e05cbe5508ea5a93b1470ef14c9462884 Mon Sep 17 00:00:00 2001 From: in1tiate Date: Thu, 15 Apr 2021 06:34:31 -0500 Subject: [PATCH 1/6] 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 2/6] 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 3/6] 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 4/6] 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 5/6] 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 6/6] 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)