diff --git a/bin/storage/testimony/sample.txt b/bin/storage/testimony/sample.txt new file mode 100644 index 0000000..125e173 --- /dev/null +++ b/bin/storage/testimony/sample.txt @@ -0,0 +1,4 @@ +1#-#Hobo_Phoenix#hobo-normal#~~\n-- Sample Recording --#wit#1#0#22#1#0#0#0#0#3##-1#0#0#00#0#0#0#0#0#-^(b)hobo-normal^(a)hobo-normal^#-^(b)hobo-normal^(a)hobo-normal^#-^(b)hobo-normal^(a)hobo-normal^#0#|| +1#-#Hobo_Phoenix#hobo-normal#This is a sample recording to test the loading and saving of testimonies.#wit#1#0#22#1#0#0#0#0#1##-1#0#0#00#0#0#0#0#0#-^(b)hobo-normal^(a)hobo-normal^#-^(b)hobo-normal^(a)hobo-normal^#-^(b)hobo-normal^(a)hobo-normal^#0#|| +1#-#Hobo_Phoenix#hobo-bigeyed#If you are reading this your server supports testimony loading correctly.#wit#1#0#22#1#0#0#0#0#1##-1#0#0#00#0#0#0#0#0#-^(b)hobo-bigeyed^(a)hobo-bigeyed^#-^(b)hobo-bigeyed^(a)hobo-bigeyed^#-^(b)hobo-bigeyed^(a)hobo-bigeyed^#0#|| +1#-#Hobo_Phoenix#hobo-bigeyed#There is nothing else to be read here.#wit#1#0#22#1#0#0#0#0#1##-1#0#0#00#0#0#0#0#0#-^(b)hobo-bigeyed^(a)hobo-bigeyed^#-^(b)hobo-bigeyed^(a)hobo-bigeyed^#-^(b)hobo-bigeyed^(a)hobo-bigeyed^#0#|| diff --git a/include/aoclient.h b/include/aoclient.h index 80b3c09..e198ed5 100644 --- a/include/aoclient.h +++ b/include/aoclient.h @@ -220,7 +220,9 @@ class AOClient : public QObject { {"MODCHAT", 1ULL << 9 }, {"MUTE", 1ULL << 10}, {"UNCM", 1ULL << 11}, - {"SUPER", ~0ULL }, + {"SAVETEST", 1ULL << 12}, + {"FORCE_CHARSELECT",1ULL << 13}, + {"SUPER", ~0ULL } }; @@ -272,6 +274,10 @@ class AOClient : public QObject { QStringList charcurse_list; + /** + * @brief Temporary client permission if client is allowed to save a testimony to server storage. + */ + bool testimony_saving = false; public slots: /** @@ -1161,6 +1167,16 @@ class AOClient : public QObject { */ void cmdAllowIniswap(int argc, QStringList argv); + /** + * @brief Grants a client the temporary permission to save a testimony. + * + * @details ClientID as the target of permission + * + * @iscommand + * + */ + void cmdPermitSaving(int argc, QStringList argv); + ///@} /** @@ -1315,8 +1331,6 @@ class AOClient : public QObject { * * @details No arguments. * - * Can silently "fail" if the character picked is already being used by another client. - * * @iscommand */ void cmdRandomChar(int argc, QStringList argv); @@ -1502,6 +1516,7 @@ class AOClient : public QObject { * @iscommand */ void cmdUnCharCurse(int argc, QStringList argv); + void cmdCharSelect(int argc, QStringList argv); ///@} @@ -1598,6 +1613,34 @@ class AOClient : public QObject { */ void cmdAddStatement(int argc, QStringList argv); + + /** + * @brief Sends a list of the testimony to OOC of the requesting client + * + * @details Retrieves all stored IC-Messages of the area and dumps them into OOC with some formatting. + * + */ + void cmdTestimony(int argc, QStringList argv); + + /** + * @brief Saves a testimony recording to the servers storage. + * + * @details Saves a titled text file which contains the edited packets into a text file. + * The filename will always be lowercase. + * + */ + void cmdSaveTestimony(int argc, QStringList argv); + + /** + * @brief Loads testimony for the testimony replay. Argument is the testimony name. + * + * @details Loads a titled text file which contains the edited packets to be loaded into the QVector. + * Validates the size of the testimony to ensure the entire testimony can be replayed. + * Testimony name will always be converted to lowercase. + * + */ + void cmdLoadTestimony(int argc, QStringList argv); + ///@} /** @@ -1890,6 +1933,7 @@ class AOClient : public QObject { {"ungimp", {ACLFlags.value("MUTE"), 1, &AOClient::cmdUnGimp}}, {"baninfo", {ACLFlags.value("BAN"), 1, &AOClient::cmdBanInfo}}, {"testify", {ACLFlags.value("CM"), 0, &AOClient::cmdTestify}}, + {"testimony", {ACLFlags.value("NONE"), 0, &AOClient::cmdTestimony}}, {"examine", {ACLFlags.value("CM"), 0, &AOClient::cmdExamine}}, {"pause", {ACLFlags.value("CM"), 0, &AOClient::cmdPauseTestimony}}, {"delete", {ACLFlags.value("CM"), 0, &AOClient::cmdDeleteStatement}}, @@ -1905,6 +1949,9 @@ class AOClient : public QObject { {"allowiniswap", {ACLFlags.value("CM"), 0, &AOClient::cmdAllowIniswap}}, {"allow_iniswap", {ACLFlags.value("CM"), 0, &AOClient::cmdAllowIniswap}}, {"afk", {ACLFlags.value("NONE"), 0, &AOClient::cmdAfk}}, + {"savetestimony", {ACLFlags.value("NONE"), 1, &AOClient::cmdSaveTestimony}}, + {"loadtestimony", {ACLFlags.value("CM"), 1, &AOClient::cmdLoadTestimony}}, + {"permitsaving", {ACLFlags.value("MODCHAT"), 1, &AOClient::cmdPermitSaving}}, {"mutepm", {ACLFlags.value("NONE"), 0, &AOClient::cmdMutePM}}, {"toggleadverts", {ACLFlags.value("NONE"), 0, &AOClient::cmdToggleAdverts}}, {"oocmute", {ACLFlags.value("MUTE"), 1, &AOClient::cmdOocMute}}, @@ -1921,6 +1968,7 @@ class AOClient : public QObject { {"unblock_dj", {ACLFlags.value("MUTE"), 1, &AOClient::cmdUnBlockDj}}, {"charcurse", {ACLFlags.value("MUTE"), 1, &AOClient::cmdCharCurse}}, {"uncharcurse", {ACLFlags.value("MUTE"), 1, &AOClient::cmdUnCharCurse}}, + {"charselect", {ACLFlags.value("NONE"), 0, &AOClient::cmdCharSelect}} }; /** diff --git a/src/aoclient.cpp b/src/aoclient.cpp index d6394e7..df0e26d 100644 --- a/src/aoclient.cpp +++ b/src/aoclient.cpp @@ -151,6 +151,10 @@ bool AOClient::changeCharacter(int char_id) if (is_charcursed && !charcurse_list.contains(QString::number(char_id))) { return false; } + + if (current_char != "") { + area->characters_taken.removeAll(server->getCharID(current_char)); + } if (char_id >= 0) { QString char_selected = server->characters[char_id]; @@ -158,10 +162,6 @@ bool AOClient::changeCharacter(int char_id) if (taken || char_selected == "") return false; - if (current_char != "") { - area->characters_taken.removeAll(server->getCharID(current_char)); - } - area->characters_taken.append(char_id); current_char = char_selected; } diff --git a/src/commands/casing.cpp b/src/commands/casing.cpp index db4ce40..3a43ac5 100644 --- a/src/commands/casing.cpp +++ b/src/commands/casing.cpp @@ -130,6 +130,24 @@ void AOClient::cmdExamine(int argc, QStringList argv) sendServerMessage("Unable to start replay without prior examination."); } +void AOClient::cmdTestimony(int argc, QStringList argv) +{ + AreaData* area = server->areas[current_area]; + if (area->testimony.size() -1 < 1) { + sendServerMessage("Unable to display empty testimony."); + return; + } + + QString ooc_message; + for (int i = 1; i <= area->testimony.size() -1; i++) + { + QStringList packet = area->testimony.at(i); + QString ic_message = packet[4]; + ooc_message.append( "[" + QString::number(i) + "]" + ic_message + "\n"); + } + sendServerMessage(ooc_message); +} + void AOClient::cmdDeleteStatement(int argc, QStringList argv) { AreaData* area = server->areas[current_area]; @@ -153,6 +171,7 @@ void AOClient::cmdPauseTestimony(int argc, QStringList argv) { AreaData* area = server->areas[current_area]; area->test_rec = AreaData::TestimonyRecording::STOPPED; + server->broadcast(AOPacket("RT",{"testimony1"}), current_area); sendServerMessage("Testimony has been stopped."); } @@ -165,3 +184,86 @@ void AOClient::cmdAddStatement(int argc, QStringList argv) else sendServerMessage("Unable to add anymore statements. Please remove any unused ones."); } + +void AOClient::cmdSaveTestimony(int argc, QStringList argv) +{ + bool permission_found = false; + if (checkAuth(ACLFlags.value("SAVETEST"))) + permission_found = true; + + if (testimony_saving == true) + permission_found = true; + + if (permission_found) { + AreaData* area = server->areas[current_area]; + if (area->testimony.size() -1 <= 0) { + sendServerMessage("Can't save an empty testimony."); + return; + } + + QDir dir_testimony("storage/testimony"); + if (!dir_testimony.exists()) { + dir_testimony.mkpath("."); + } + + QString testimony_name = argv[0].trimmed().toLower().replace("..",""); // :) + QFile file("storage/testimony/" + testimony_name + ".txt"); + if (file.exists()) { + sendServerMessage("Unable to save testimony. Testimony name already exists."); + return; + } + + QTextStream out(&file); + if(file.open(QIODevice::WriteOnly | QIODevice::Append | QIODevice::Text)) { + for (int i = 0; i <= area->testimony.size() -1; i++) + { + out << area->testimony.at(i).join("#") << "\n"; + } + sendServerMessage("Testimony saved. To load it use /loadtestimony " + testimony_name); + testimony_saving = false; + } + } + else { + sendServerMessage("You don't have permission to save a testimony. Please contact a moderator for permission."); + return; + } +} + +void AOClient::cmdLoadTestimony(int argc, QStringList argv) +{ + AreaData* area = server->areas[current_area]; + QDir dir_testimony("storage/testimony"); + if (!dir_testimony.exists()) { + sendServerMessage("Unable to load testimonies. Testimony storage not found."); + return; + } + + QString testimony_name = argv[0].trimmed().toLower().replace("..",""); // :) + QFile file("storage/testimony/" + testimony_name + ".txt"); + if (!file.exists()) { + sendServerMessage("Unable to load testimony. Testimony name not found."); + return; + } + if (!file.open(QIODevice::ReadOnly | QIODevice::Text)) { + sendServerMessage("Unable to load testimony. Permission denied."); + return; + } + + clearTestimony(); + int testimony_lines = 0; + QTextStream in(&file); + while (!in.atEnd()) { + if (testimony_lines <= server->maximum_statements) { + QString line = in.readLine(); + QStringList packet = line.split("#"); + area->testimony.append(packet); + testimony_lines = testimony_lines + 1; + } + else { + sendServerMessage("Testimony too large to be loaded."); + clearTestimony(); + return; + } + } + sendServerMessage("Testimony loaded successfully. Use /examine to start playback."); +} diff --git a/src/commands/messaging.cpp b/src/commands/messaging.cpp index e2cafa1..71c39d9 100644 --- a/src/commands/messaging.cpp +++ b/src/commands/messaging.cpp @@ -86,18 +86,33 @@ void AOClient::cmdNeed(int argc, QStringList argv) void AOClient::cmdSwitch(int argc, QStringList argv) { - int char_id = server->getCharID(argv.join(" ")); - if (char_id == -1) { + int selected_char_id = server->getCharID(argv.join(" ")); + if (selected_char_id == -1) { sendServerMessage("That does not look like a valid character."); return; } - changeCharacter(char_id); + if (changeCharacter(selected_char_id)) { + char_id = selected_char_id; + } + else { + sendServerMessage("The character you picked is either taken or invalid."); + } } void AOClient::cmdRandomChar(int argc, QStringList argv) { - int char_id = genRand(0, server->characters.size() - 1); - changeCharacter(char_id); + AreaData* area = server->areas[current_area]; + int selected_char_id; + bool taken = true; + while (taken) { + selected_char_id = genRand(0, server->characters.size() - 1); + if (!area->characters_taken.contains(selected_char_id)) { + taken = false; + } + } + if (changeCharacter(selected_char_id)) { + char_id = selected_char_id; + } } void AOClient::cmdToggleGlobal(int argc, QStringList argv) @@ -369,3 +384,26 @@ void AOClient::cmdUnCharCurse(int argc, QStringList argv) sendServerMessage("Uncharcursed plater."); target->sendServerMessage("You were uncharcursed."); } + +void AOClient::cmdCharSelect(int argc, QStringList argv) +{ + if (argc == 0) { + changeCharacter(-1); + sendPacket("DONE"); + } + else { + if (!checkAuth(ACLFlags.value("FORCE_CHARSELECT"))) { + sendServerMessage("You do not have permission to force another player to character select!"); + return; + } + + bool ok = false; + int target_id = argv[0].toInt(&ok); + if (!ok) + return; + + AOClient* target = server->getClientByID(target_id); + target->changeCharacter(-1); + target->sendPacket("DONE"); + } +} diff --git a/src/commands/moderation.cpp b/src/commands/moderation.cpp index 2fa8efe..6e3b99c 100644 --- a/src/commands/moderation.cpp +++ b/src/commands/moderation.cpp @@ -402,3 +402,13 @@ void AOClient::cmdAllowIniswap(int argc, QStringList argv) QString state = area->iniswap_allowed ? "allowed." : "disallowed."; sendServerMessage("Iniswapping in this area is now " + state); } + +void AOClient::cmdPermitSaving(int argc, QStringList argv) +{ + AOClient* client = server->getClientByID(argv[0].toInt()); + if (client == nullptr) { + sendServerMessage("Invalid ID."); + return; + } + client->testimony_saving = true; +}