From 5115b97721e5b37c0df9149b26bc31a5af4404e4 Mon Sep 17 00:00:00 2001 From: Salanto <62221668+Salanto@users.noreply.github.com> Date: Sun, 12 Sep 2021 13:46:18 +0200 Subject: [PATCH 1/8] Implement JSON musiclist file loading --- bin/config_sample/music.json | 203 ++++++++++++++++++++++++++++++++++ bin/config_sample/music.txt | 49 -------- core/include/config_manager.h | 25 ++++- core/src/config_manager.cpp | 55 +++++++-- 4 files changed, 271 insertions(+), 61 deletions(-) create mode 100644 bin/config_sample/music.json delete mode 100644 bin/config_sample/music.txt diff --git a/bin/config_sample/music.json b/bin/config_sample/music.json new file mode 100644 index 0000000..14d3975 --- /dev/null +++ b/bin/config_sample/music.json @@ -0,0 +1,203 @@ +[ + { + "category":"==Music==", + "songs":[ + { + "name":"Announce The Truth (AA).opus", + "length":-1 + }, + { + "name":"Announce The Truth (AJ).opus", + "length":-1 + }, + { + "name":"Announce The Truth (JFA).opus", + "length":-1 + }, + { + "name":"Announce The Truth (Miles).opus", + "length":-1 + }, + { + "name":"Announce The Truth (T&T).opus", + "length":-1 + }, + { + "name":"Confrontation ~ Presto 2009.opus", + "length":-1 + }, + { + "name":"Crises of Fate.opus", + "length":-1 + }, + { + "name":"Forgotten Legend.opus", + "length":-1 + }, + { + "name":"Godot - The Fragrance of Dark Coffee.opus", + "length":-1 + }, + { + "name":"Great Revival ~ Franziska von Karma.opus", + "length":-1 + }, + { + "name":"Great Revival ~ Miles Edgeworth.opus", + "length":-1 + }, + { + "name":"Hotline of Fate.opus", + "length":-1 + }, + { + "name":"Interesting People.opus", + "length":-1 + }, + { + "name":"Logic and Trick.opus", + "length":-1 + }, + { + "name":"Luke Atmey ~ I Just Want Love.opus", + "length":-1 + }, + { + "name":"Noisy People.opus", + "length":-1 + }, + { + "name":"OBJECTION (AA).opus", + "length":-1 + }, + { + "name":"Objection (AJ).opus", + "length":-1 + }, + { + "name":"OBJECTION (JFA).opus", + "length":-1 + }, + { + "name":"Objection (Miles).opus", + "length":-1 + }, + { + "name":"OBJECTION (T&T).opus", + "length":-1 + }, + { + "name":"Others ~ Guilty love.opus", + "length":-1 + }, + { + "name":"Prelude (AA).opus", + "length":-1 + }, + { + "name":"Prelude (AJ).opus", + "length":-1 + }, + { + "name":"Prologue (AA).opus", + "length":-1 + }, + { + "name":"Pursuit (AA) - variation.opus", + "length":-1 + }, + { + "name":"Pursuit (AA).opus", + "length":-1 + }, + { + "name":"Pursuit (AJ).opus", + "length":-1 + }, + { + "name":"Pursuit (DS).opus", + "length":-1 + }, + { + "name":"Pursuit (JFA) - variation.opus", + "length":-1 + }, + { + "name":"Pursuit (JFA).opus", + "length":-1 + }, + { + "name":"Pursuit (Miles).opus", + "length":-1 + }, + { + "name":"Pursuit (T&T) - variation.opus", + "length":-1 + }, + { + "name":"Pursuit (T&T).opus", + "length":-1 + }, + { + "name":"Pursuit ~ I Want to Find the Truth(Orchestra).opus", + "length":-1 + }, + { + "name":"Questioning AA (Allegro).opus", + "length":-1 + }, + { + "name":"Questioning AA (Moderato).opus", + "length":-1 + }, + { + "name":"Questioning AJ (Allegro).opus", + "length":-1 + }, + { + "name":"Questioning AJ (Moderato).opus", + "length":-1 + }, + { + "name":"Questioning JFA (Allegro).opus", + "length":-1 + }, + { + "name":"Questioning JFA (Moderato).opus", + "length":-1 + }, + { + "name":"Questioning T&T (Allegro).opus", + "length":-1 + }, + { + "name":"Questioning T&T (Moderato).opus", + "length":-1 + }, + { + "name":"Speak up Pup.opus", + "length":-1 + }, + { + "name":"Suspense (AA).opus", + "length":-1 + }, + { + "name":"The Great Truth Burglar.opus", + "length":-1 + }, + { + "name":"Trial (AA).opus", + "length":-1 + }, + { + "name":"Trial (AJ).opus", + "length":-1 + }, + { + "name":"Trial (Miles).opus", + "length":-1 + } + ] + } + ] \ No newline at end of file diff --git a/bin/config_sample/music.txt b/bin/config_sample/music.txt deleted file mode 100644 index 585b439..0000000 --- a/bin/config_sample/music.txt +++ /dev/null @@ -1,49 +0,0 @@ -Announce The Truth (AA).opus -Announce The Truth (AJ).opus -Announce The Truth (JFA).opus -Announce The Truth (Miles).opus -Announce The Truth (T&T).opus -Confrontation ~ Presto 2009.opus -Crises of Fate.opus -Forgotten Legend.opus -Godot - The Fragrance of Dark Coffee.opus -Great Revival ~ Franziska von Karma.opus -Great Revival ~ Miles Edgeworth.opus -Hotline of Fate.opus -Interesting People.opus -Logic and Trick.opus -Luke Atmey ~ I Just Want Love.opus -Noisy People.opus -OBJECTION (AA).opus -Objection (AJ).opus -OBJECTION (JFA).opus -Objection (Miles).opus -OBJECTION (T&T).opus -Others ~ Guilty love.opus -Prelude (AA).opus -Prelude (AJ).opus -Prologue (AA).opus -Pursuit (AA) - variation.opus -Pursuit (AA).opus -Pursuit (AJ).opus -Pursuit (DS).opus -Pursuit (JFA) - variation.opus -Pursuit (JFA).opus -Pursuit (Miles).opus -Pursuit (T&T) - variation.opus -Pursuit (T&T).opus -Pursuit ~ I Want to Find the Truth (Orchestra).opus -Questioning AA (Allegro).opus -Questioning AA (Moderato).opus -Questioning AJ (Allegro).opus -Questioning AJ (Moderato).opus -Questioning JFA (Allegro).opus -Questioning JFA (Moderato).opus -Questioning T&T (Allegro).opus -Questioning T&T (Moderato).opus -Speak up Pup.opus -Suspense (AA).opus -The Great Truth Burglar.opus -Trial (AA).opus -Trial (AJ).opus -Trial (Miles).opus diff --git a/core/include/config_manager.h b/core/include/config_manager.h index 64e0993..e6db121 100644 --- a/core/include/config_manager.h +++ b/core/include/config_manager.h @@ -31,6 +31,11 @@ #include #include +//JSON loading requirements +#include +#include +#include + /** * @brief The config file handler class. */ @@ -73,10 +78,18 @@ class ConfigManager { static QStringList musiclist(); /** - * @brief Returns the content of - * @return + * @brief Returns the duration of a song in the songlist. + * @param The name of the song where duration is requested + * @return The duration of the song */ - static QSettings *areaData(); + static int songInformation(const QString& f_songName); + + /** + * @brief Returns the content of + * + * @return See short description. + */ + static QSettings* areaData(); /** * @brief Returns a sanitized QStringList of the areas. @@ -91,6 +104,7 @@ class ConfigManager { * @return See short description. */ static QStringList rawAreaNames(); + /** * @brief Returns true if the server should advertise to the master server. * @@ -481,6 +495,11 @@ private: */ static QElapsedTimer* m_uptimeTimer; + /** + * @brief Contains the musiclist with time durations. + */ + static QHash* m_musicList; + /** * @brief Returns a stringlist with the contents of a .txt file from config/text/. * diff --git a/core/src/config_manager.cpp b/core/src/config_manager.cpp index 0a2b37e..c8f1680 100644 --- a/core/src/config_manager.cpp +++ b/core/src/config_manager.cpp @@ -24,6 +24,7 @@ QSettings* ConfigManager::m_discord = new QSettings("config/discord.ini", QSetti QSettings* ConfigManager::m_areas = new QSettings("config/areas.ini", QSettings::IniFormat); ConfigManager::CommandSettings* ConfigManager::m_commands = new CommandSettings(); QElapsedTimer* ConfigManager::m_uptimeTimer = new QElapsedTimer; +QHash* ConfigManager::m_musicList = new QHash; bool ConfigManager::verifyServerConfig() { @@ -127,16 +128,52 @@ QStringList ConfigManager::backgrounds() QStringList ConfigManager::musiclist() { - QStringList l_music_list; - QFile l_file("config/music.txt"); - l_file.open(QIODevice::ReadOnly | QIODevice::Text); - while (!l_file.atEnd()) { - l_music_list.append(l_file.readLine().trimmed()); + QFile l_music_json("config/music.json"); + l_music_json.open(QIODevice::ReadOnly | QIODevice::Text); + + QJsonParseError l_error; + QJsonDocument l_music_list_json = QJsonDocument::fromJson(l_music_json.readAll(), &l_error); + if (!(l_error.error == QJsonParseError::NoError)) { //Non-Terminating error. + qWarning() << "Unable to load musiclist. The following error was encounted : " + l_error.errorString(); + return QStringList {}; //Server can still run without music. } - l_file.close(); - if(l_music_list[0].contains(".")) // Add a default category if none exists - l_music_list.insert(0, "==Music=="); - return l_music_list; + + // Akashi expects the musiclist to be contained in a JSON array, even if its only a single category. + QJsonArray l_Json_root_array = l_music_list_json.array(); + QJsonObject l_child_obj; + QJsonArray l_child_array; + for (int i = 0; i <= l_Json_root_array.size() -1; i++){ //Iterate trough entire JSON file to assemble musiclist + l_child_obj = l_Json_root_array.at(i).toObject(); + + //Technically not a requirement, but neat for organisation. + QString l_category_name = l_child_obj["category"].toString(); + if (!l_category_name.isEmpty()) { + m_musicList->insert(l_category_name,0); + } + else { + qWarning() << "Category name not set. This may cause the musiclist to be displayed incorrectly."; + } + + l_child_array = l_child_obj["songs"].toArray(); + for (int i = 0; i <= l_child_array.size() -1; i++){ // Inner for loop because a category can contain multiple songs. + QJsonObject l_song_obj = l_child_array.at(i).toObject(); + QString l_song_name = l_song_obj["name"].toString(); + int l_song_duration = l_song_obj["length"].toVariant().toFloat(); + m_musicList->insert(l_song_name,l_song_duration); + } + } + l_music_json.close(); + + QStringList l_musiclist = m_musicList->keys(); + if(!l_musiclist[0].contains("==")) // Add a default category if none exists + l_musiclist.insert(0,"==Music=="); + + return l_musiclist; +} + +int ConfigManager::songInformation(const QString &f_songName) +{ + return m_musicList->value(f_songName); } QSettings* ConfigManager::areaData() From f9dcaa2aa51b4e1fe149924ba806ec94adaf28c9 Mon Sep 17 00:00:00 2001 From: Salanto <62221668+Salanto@users.noreply.github.com> Date: Sun, 12 Sep 2021 22:32:10 +0200 Subject: [PATCH 2/8] Implement a basic and functional jukebox implementation --- core/include/aoclient.h | 8 ++++++ core/include/area_data.h | 56 +++++++++++++++++++++++++++++++++++++ core/src/area_data.cpp | 49 ++++++++++++++++++++++++++++++++ core/src/commands/music.cpp | 17 ++++++++++- core/src/config_manager.cpp | 4 ++- core/src/packets.cpp | 9 ++++++ core/src/server.cpp | 5 +++- 7 files changed, 145 insertions(+), 3 deletions(-) diff --git a/core/include/aoclient.h b/core/include/aoclient.h index baacc4a..da735eb 100644 --- a/core/include/aoclient.h +++ b/core/include/aoclient.h @@ -1809,6 +1809,13 @@ class AOClient : public QObject { */ void cmdToggleMusic(int argc, QStringList argv); + /** + * @brief cmdToggleJukebox Toggles jukebox status in the current area. + * + * @details No arguments. + */ + void cmdToggleJukebox(int argc, QStringList argv); + ///@} /** @@ -2121,6 +2128,7 @@ class AOClient : public QObject { {"ignore_bglist", {ACLFlags.value("IGNORE_BGLIST"),0, &AOClient::cmdIgnoreBgList}}, {"notice", {ACLFlags.value("SEND_NOTICE"), 1, &AOClient::cmdNotice}}, {"noticeg", {ACLFlags.value("SEND_NOTICE"), 1, &AOClient::cmdNoticeGlobal}}, + {"togglejukebox", {ACLFlags.value("None"), 0, &AOClient::cmdToggleJukebox}}, }; /** diff --git a/core/include/area_data.h b/core/include/area_data.h index 2bdad36..a5bddac 100644 --- a/core/include/area_data.h +++ b/core/include/area_data.h @@ -27,6 +27,7 @@ #include #include #include +#include class Logger; @@ -295,6 +296,15 @@ class AreaData : public QObject { */ LockStatus lockStatus() const; + /** + * @brief Returns the jukebox status of the area. + * + * @return See short description. + * + * @see #m_jukebox + */ + bool isjukeboxEnabled() const; + /** * @brief Locks the area, setting it to LOCKED. */ @@ -778,6 +788,30 @@ class AreaData : public QObject { */ void toggleIgnoreBgList(); + /** + * @brief Toggles wether the jukebox is enabled or not. + */ + void toggleJukebox(); + + /** + * @brief Adds a song to the Jukeboxs queue. + */ + bool addJukeboxSong(QString f_song); + + public slots: + + /** + * @brief Plays a random song from the jukebox. Plays the same if only one is left. + */ + void switchJukeboxSong(); + + signals: + + /** + * @brief Changes the song in the current area when the jukebox timer expires. + */ + void playJukeboxSong(AOPacket f_packet, int f_area_index); + private: /** * @brief The list of timers available in the area. @@ -966,6 +1000,28 @@ private: * @brief Whether or not to ignore the server defined background list. If true, any background can be set in an area. */ bool m_ignoreBgList; + + // Jukebox specific members + /** + * @brief Stores the songs added to the jukebox to be played. + * + * @details This contains the names of each song, noteworthy is that none of the songs are able to be entered twice. + * + */ + QVector m_jukebox_queue; + + /** + * @brief Triggers the playing of the next song once the last song has fully played. + * + * @details While this may be considered bad design, I do not care. + * It triggers a direct broadcast of the MC packet in the area. + */ + QTimer* m_jukebox_timer; + + /** + * @brief Wether or not the jukebox is enabled in this area. + */ + bool m_jukebox; }; #endif // AREA_DATA_H diff --git a/core/src/area_data.cpp b/core/src/area_data.cpp index 7f5eddc..2e0df34 100644 --- a/core/src/area_data.cpp +++ b/core/src/area_data.cpp @@ -48,6 +48,7 @@ AreaData::AreaData(QString p_name, int p_index) : m_toggleMusic = areas_ini->value("toggle_music", "true").toBool(); m_shownameAllowed = areas_ini->value("shownames_allowed", "true").toBool(); m_ignoreBgList = areas_ini->value("ignore_bglist", "false").toBool(); + m_jukebox = areas_ini->value("jukebox_enabled", "false").toBool(); areas_ini->endGroup(); QTimer* timer1 = new QTimer(); m_timers.append(timer1); @@ -57,6 +58,9 @@ AreaData::AreaData(QString p_name, int p_index) : m_timers.append(timer3); QTimer* timer4 = new QTimer(); m_timers.append(timer4); + m_jukebox_timer = new QTimer(); + connect(m_jukebox_timer, &QTimer::timeout, + this, &AreaData::switchJukeboxSong); } const QMap AreaData::map_statuses = { @@ -131,6 +135,11 @@ AreaData::LockStatus AreaData::lockStatus() const return m_locked; } +bool AreaData::isjukeboxEnabled() const +{ + return m_jukebox; +} + void AreaData::lock() { m_locked = LockStatus::LOCKED; @@ -504,3 +513,43 @@ void AreaData::toggleIgnoreBgList() { m_ignoreBgList = !m_ignoreBgList; } + +void AreaData::toggleJukebox() +{ + m_jukebox = !m_jukebox; +} + +bool AreaData::addJukeboxSong(QString f_song) +{ + if(!m_jukebox_queue.contains(f_song)) { + if (m_jukebox_queue.size() == 0) { + emit playJukeboxSong(AOPacket("MC",{f_song,QString::number(-1)}), index()); + m_jukebox_timer->start(ConfigManager::songInformation(f_song) * 1000); + } + m_jukebox_queue.append(f_song); + return true; + } + else { + return false; + } +} + +void AreaData::switchJukeboxSong() +{ + QString l_song_name; + if(m_jukebox_queue.size() == 1) { + l_song_name = m_jukebox_queue[0]; + emit playJukeboxSong(AOPacket("MC",{l_song_name,QString::number(-1)}), m_index); + m_jukebox_timer->start(ConfigManager::songInformation(l_song_name) * 1000); + } + else { + int l_random_index = QRandomGenerator::system()->bounded(m_jukebox_queue.size() -1); + l_song_name = m_jukebox_queue[l_random_index]; + emit playJukeboxSong(AOPacket("MC",{l_song_name,QString::number(-1)}), m_index); + m_jukebox_timer->start(ConfigManager::songInformation(m_jukebox_queue[l_random_index]) * 1000); + m_jukebox_queue.remove(l_random_index); + m_jukebox_queue.squeeze(); + } + currentMusic() = l_song_name; + musicPlayerBy() = "Jukebox"; +} diff --git a/core/src/commands/music.cpp b/core/src/commands/music.cpp index 35fec63..5a5b58e 100644 --- a/core/src/commands/music.cpp +++ b/core/src/commands/music.cpp @@ -16,7 +16,6 @@ // along with this program. If not, see . // ////////////////////////////////////////////////////////////////////////////////////// #include "include/aoclient.h" - // This file is for commands under the music category in aoclient.h // Be sure to register the command in the header before adding it here! @@ -112,3 +111,19 @@ void AOClient::cmdToggleMusic(int argc, QStringList argv) QString l_state = l_area->isMusicAllowed() ? "allowed." : "disallowed."; sendServerMessage("Music in this area is now " + l_state); } + +void AOClient::cmdToggleJukebox(int argc, QStringList argv) +{ + Q_UNUSED(argc); + Q_UNUSED(argv); + + if (checkAuth(ACLFlags.value("CM")) | m_authenticated) { + AreaData* l_area = server->m_areas.value(m_current_area); + l_area->toggleJukebox(); + QString l_state = l_area->isjukeboxEnabled() ? "enabled." : "disabled."; + sendServerMessageArea("The jukebox in this area has been " + l_state); + } + else { + sendServerMessage("You do not have permission to change the jukebox status."); + } +} diff --git a/core/src/config_manager.cpp b/core/src/config_manager.cpp index c8f1680..fbfadf8 100644 --- a/core/src/config_manager.cpp +++ b/core/src/config_manager.cpp @@ -142,6 +142,7 @@ QStringList ConfigManager::musiclist() QJsonArray l_Json_root_array = l_music_list_json.array(); QJsonObject l_child_obj; QJsonArray l_child_array; + QStringList l_musiclist; for (int i = 0; i <= l_Json_root_array.size() -1; i++){ //Iterate trough entire JSON file to assemble musiclist l_child_obj = l_Json_root_array.at(i).toObject(); @@ -149,6 +150,7 @@ QStringList ConfigManager::musiclist() QString l_category_name = l_child_obj["category"].toString(); if (!l_category_name.isEmpty()) { m_musicList->insert(l_category_name,0); + l_musiclist.append(l_category_name); } else { qWarning() << "Category name not set. This may cause the musiclist to be displayed incorrectly."; @@ -160,11 +162,11 @@ QStringList ConfigManager::musiclist() QString l_song_name = l_song_obj["name"].toString(); int l_song_duration = l_song_obj["length"].toVariant().toFloat(); m_musicList->insert(l_song_name,l_song_duration); + l_musiclist.append(l_song_name); } } l_music_json.close(); - QStringList l_musiclist = m_musicList->keys(); if(!l_musiclist[0].contains("==")) // Add a default category if none exists l_musiclist.insert(0,"==Music=="); diff --git a/core/src/packets.cpp b/core/src/packets.cpp index 5360f4f..846a073 100644 --- a/core/src/packets.cpp +++ b/core/src/packets.cpp @@ -319,6 +319,15 @@ void AOClient::pktChangeMusic(AreaData* area, int argc, QStringList argv, AOPack l_final_song = "~stop.mp3"; else l_final_song = l_argument; + + if (area->isjukeboxEnabled()) { + if (area->addJukeboxSong(l_final_song)) + sendServerMessage("Your song has been added to the Jukebox queue."); + else + sendServerMessage("Your song could not be added to the jukebox queue. It already exists"); + return; + } + AOPacket l_music_change("MC", {l_final_song, argv[1], m_showname, "1", "0", l_effects}); area->currentMusic() = l_final_song; area->musicPlayerBy() = m_showname; diff --git a/core/src/server.cpp b/core/src/server.cpp index e9f0ab4..8ff9212 100644 --- a/core/src/server.cpp +++ b/core/src/server.cpp @@ -91,7 +91,10 @@ void Server::start() QStringList raw_area_names = ConfigManager::rawAreaNames(); for (int i = 0; i < raw_area_names.length(); i++) { QString area_name = raw_area_names[i]; - m_areas.insert(i, new AreaData(area_name, i)); + AreaData* l_area = new AreaData(area_name, i); + m_areas.insert(i, l_area); + connect(l_area, &AreaData::playJukeboxSong, + this, QOverload::of(&Server::broadcast)); } //Rate-Limiter for IC-Chat From 8bdfe1e04525b0494b479ba63bd6b652b5e88274 Mon Sep 17 00:00:00 2001 From: Salanto <62221668+Salanto@users.noreply.github.com> Date: Sun, 12 Sep 2021 23:20:09 +0200 Subject: [PATCH 3/8] Ensure that no stale songs are in the jukebox queue when its disabled Housekeeping and fix /currentmusic not returning a song --- core/include/area_data.h | 18 ++++++++++++++++++ core/src/area_data.cpp | 20 ++++++++++++++++++-- core/src/commands/music.cpp | 2 +- core/src/packets.cpp | 9 +++++---- 4 files changed, 42 insertions(+), 7 deletions(-) diff --git a/core/include/area_data.h b/core/include/area_data.h index a5bddac..8d6692e 100644 --- a/core/include/area_data.h +++ b/core/include/area_data.h @@ -577,6 +577,15 @@ class AreaData : public QObject { */ QString currentMusic() const; + /** + * @brief Sets the music currently being played in the area. + * + * @param Name of the song being played. + * + * @see #m_currentMusic + */ + void setCurrentMusic(QString f_current_song); + /** * @brief Returns the showname of the client who played the music in the area. * @@ -586,6 +595,15 @@ class AreaData : public QObject { */ QString musicPlayerBy() const; + /** + * @brief Sets the showname of the client who played the music in the area. + * + * @param Showname of the client. + * + * @see #m_musicPlayedBy + */ + void setMusicPlayedBy(const QString& f_music_player); + /** * @brief Changes the music being played in the area. * diff --git a/core/src/area_data.cpp b/core/src/area_data.cpp index 2e0df34..93f8a36 100644 --- a/core/src/area_data.cpp +++ b/core/src/area_data.cpp @@ -429,6 +429,11 @@ QString AreaData::musicPlayerBy() const return m_musicPlayedBy; } +void AreaData::setMusicPlayedBy(const QString& f_music_player) +{ + m_musicPlayedBy = f_music_player; +} + void AreaData::changeMusic(const QString &f_source_r, const QString &f_newSong_r) { m_currentMusic = f_newSong_r; @@ -440,6 +445,11 @@ QString AreaData::currentMusic() const return m_currentMusic; } +void AreaData::setCurrentMusic(QString f_current_song) +{ + m_currentMusic = f_current_song; +} + int AreaData::proHP() const { return m_proHP; @@ -517,6 +527,10 @@ void AreaData::toggleIgnoreBgList() void AreaData::toggleJukebox() { m_jukebox = !m_jukebox; + if (!m_jukebox) { + m_jukebox_queue.clear(); + m_jukebox_timer->stop(); + } } bool AreaData::addJukeboxSong(QString f_song) @@ -525,6 +539,8 @@ bool AreaData::addJukeboxSong(QString f_song) if (m_jukebox_queue.size() == 0) { emit playJukeboxSong(AOPacket("MC",{f_song,QString::number(-1)}), index()); m_jukebox_timer->start(ConfigManager::songInformation(f_song) * 1000); + setCurrentMusic(f_song); + setMusicPlayedBy("Jukebox"); } m_jukebox_queue.append(f_song); return true; @@ -550,6 +566,6 @@ void AreaData::switchJukeboxSong() m_jukebox_queue.remove(l_random_index); m_jukebox_queue.squeeze(); } - currentMusic() = l_song_name; - musicPlayerBy() = "Jukebox"; + setCurrentMusic(l_song_name); + setMusicPlayedBy("Jukebox"); } diff --git a/core/src/commands/music.cpp b/core/src/commands/music.cpp index 5a5b58e..88052ad 100644 --- a/core/src/commands/music.cpp +++ b/core/src/commands/music.cpp @@ -41,7 +41,7 @@ void AOClient::cmdCurrentMusic(int argc, QStringList argv) Q_UNUSED(argv); AreaData* l_area = server->m_areas[m_current_area]; - if (l_area->currentMusic() != "" && l_area->currentMusic() != "~stop.mp3") // dummy track for stopping music + if (!l_area->currentMusic().isEmpty() && !l_area->currentMusic().contains("~stop.mp3")) // dummy track for stopping music sendServerMessage("The current song is " + l_area->currentMusic() + " played by " + l_area->musicPlayerBy()); else sendServerMessage("There is no music playing."); diff --git a/core/src/packets.cpp b/core/src/packets.cpp index 846a073..9f3670c 100644 --- a/core/src/packets.cpp +++ b/core/src/packets.cpp @@ -320,17 +320,18 @@ void AOClient::pktChangeMusic(AreaData* area, int argc, QStringList argv, AOPack else l_final_song = l_argument; + //Jukebox intercepts the direct playing of messages. if (area->isjukeboxEnabled()) { if (area->addJukeboxSong(l_final_song)) - sendServerMessage("Your song has been added to the Jukebox queue."); + sendServerMessage("Song added to jukebox."); else - sendServerMessage("Your song could not be added to the jukebox queue. It already exists"); + sendServerMessage("Unable to add. Song already in jukebox."); return; } AOPacket l_music_change("MC", {l_final_song, argv[1], m_showname, "1", "0", l_effects}); - area->currentMusic() = l_final_song; - area->musicPlayerBy() = m_showname; + area->setCurrentMusic(l_final_song); + area->setMusicPlayedBy(m_showname); server->broadcast(l_music_change, m_current_area); return; } From 00da956ec8e9d628f3c9014387b891f6b7033102 Mon Sep 17 00:00:00 2001 From: Salanto <62221668+Salanto@users.noreply.github.com> Date: Sun, 12 Sep 2021 23:49:20 +0200 Subject: [PATCH 4/8] Add ACL check so moderators can toggle jukebox --- core/include/aoclient.h | 1 + core/src/commands/music.cpp | 2 +- 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/core/include/aoclient.h b/core/include/aoclient.h index da735eb..d35a914 100644 --- a/core/include/aoclient.h +++ b/core/include/aoclient.h @@ -241,6 +241,7 @@ class AOClient : public QObject { {"BYPASS_LOCKS", 1ULL << 14}, {"IGNORE_BGLIST", 1ULL << 15}, {"SEND_NOTICE", 1ULL << 16}, + {"JUKEBOX", 1ULL << 17}, {"SUPER", ~0ULL } }; diff --git a/core/src/commands/music.cpp b/core/src/commands/music.cpp index 88052ad..9504b50 100644 --- a/core/src/commands/music.cpp +++ b/core/src/commands/music.cpp @@ -117,7 +117,7 @@ void AOClient::cmdToggleJukebox(int argc, QStringList argv) Q_UNUSED(argc); Q_UNUSED(argv); - if (checkAuth(ACLFlags.value("CM")) | m_authenticated) { + if (checkAuth(ACLFlags.value("CM")) | checkAuth(ACLFlags.value("Jukebox"))) { AreaData* l_area = server->m_areas.value(m_current_area); l_area->toggleJukebox(); QString l_state = l_area->isjukeboxEnabled() ? "enabled." : "disabled."; From 3f212aeb35eadc9b019598b2f37682224caee45b Mon Sep 17 00:00:00 2001 From: Salanto <62221668+Salanto@users.noreply.github.com> Date: Mon, 13 Sep 2021 15:29:25 +0200 Subject: [PATCH 5/8] Add reload capability for the musiclist --- core/src/commands/moderation.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/core/src/commands/moderation.cpp b/core/src/commands/moderation.cpp index fa466dc..35019ba 100644 --- a/core/src/commands/moderation.cpp +++ b/core/src/commands/moderation.cpp @@ -462,6 +462,7 @@ void AOClient::cmdReload(int argc, QStringList argv) emit server->reloadRequest(ConfigManager::serverName(), ConfigManager::serverDescription()); server->updateHTTPAdvertiserConfig(); server->handleDiscordIntegration(); + server->m_music_list = ConfigManager::musiclist(); sendServerMessage("Reloaded configurations"); } From df237bbe3a9c7bdfed0188d5cf9ffd0e1e8cce91 Mon Sep 17 00:00:00 2001 From: Salanto <62221668+Salanto@users.noreply.github.com> Date: Mon, 13 Sep 2021 18:13:48 +0200 Subject: [PATCH 6/8] Improve error handling and messaging to user --- core/include/area_data.h | 2 +- core/src/area_data.cpp | 20 ++++++++++++-------- core/src/packets.cpp | 7 +++---- 3 files changed, 16 insertions(+), 13 deletions(-) diff --git a/core/include/area_data.h b/core/include/area_data.h index 8d6692e..b270f24 100644 --- a/core/include/area_data.h +++ b/core/include/area_data.h @@ -814,7 +814,7 @@ class AreaData : public QObject { /** * @brief Adds a song to the Jukeboxs queue. */ - bool addJukeboxSong(QString f_song); + QString addJukeboxSong(QString f_song); public slots: diff --git a/core/src/area_data.cpp b/core/src/area_data.cpp index 93f8a36..f11d05d 100644 --- a/core/src/area_data.cpp +++ b/core/src/area_data.cpp @@ -533,20 +533,24 @@ void AreaData::toggleJukebox() } } -bool AreaData::addJukeboxSong(QString f_song) +QString AreaData::addJukeboxSong(QString f_song) { if(!m_jukebox_queue.contains(f_song)) { if (m_jukebox_queue.size() == 0) { - emit playJukeboxSong(AOPacket("MC",{f_song,QString::number(-1)}), index()); - m_jukebox_timer->start(ConfigManager::songInformation(f_song) * 1000); - setCurrentMusic(f_song); - setMusicPlayedBy("Jukebox"); + int l_song_duration = ConfigManager::songInformation(f_song); + if (l_song_duration >= 0) { + emit playJukeboxSong(AOPacket("MC",{f_song,QString::number(-1)}), index()); + m_jukebox_timer->start(l_song_duration * 1000); + setCurrentMusic(f_song); + setMusicPlayedBy("Jukebox"); + m_jukebox_queue.append(f_song); + return "Song added to Jukebox."; + } } - m_jukebox_queue.append(f_song); - return true; + return "Unable to add song. Duration shorther than 1."; } else { - return false; + return "Unable to add song. Song already in Jukebox."; } } diff --git a/core/src/packets.cpp b/core/src/packets.cpp index 9f3670c..52c37c3 100644 --- a/core/src/packets.cpp +++ b/core/src/packets.cpp @@ -322,13 +322,12 @@ void AOClient::pktChangeMusic(AreaData* area, int argc, QStringList argv, AOPack //Jukebox intercepts the direct playing of messages. if (area->isjukeboxEnabled()) { - if (area->addJukeboxSong(l_final_song)) - sendServerMessage("Song added to jukebox."); - else - sendServerMessage("Unable to add. Song already in jukebox."); + QString l_jukebox_reply = area->addJukeboxSong(l_final_song); + sendServerMessage(l_jukebox_reply); return; } + AOPacket l_music_change("MC", {l_final_song, argv[1], m_showname, "1", "0", l_effects}); area->setCurrentMusic(l_final_song); area->setMusicPlayedBy(m_showname); From fc8a74e273f30f500d4a151332e546ed31b4c6a0 Mon Sep 17 00:00:00 2001 From: Salanto <62221668+Salanto@users.noreply.github.com> Date: Mon, 13 Sep 2021 18:25:23 +0200 Subject: [PATCH 7/8] Add default durations to the musiclist. I have no idea how accurate they are. --- bin/config_sample/music.json | 98 ++++++++++++++++++------------------ 1 file changed, 49 insertions(+), 49 deletions(-) diff --git a/bin/config_sample/music.json b/bin/config_sample/music.json index 14d3975..abd0d67 100644 --- a/bin/config_sample/music.json +++ b/bin/config_sample/music.json @@ -4,199 +4,199 @@ "songs":[ { "name":"Announce The Truth (AA).opus", - "length":-1 + "length":79.5 }, { "name":"Announce The Truth (AJ).opus", - "length":-1 + "length":59.5 }, { "name":"Announce The Truth (JFA).opus", - "length":-1 + "length":98.5 }, { "name":"Announce The Truth (Miles).opus", - "length":-1 + "length":153.5 }, { "name":"Announce The Truth (T&T).opus", - "length":-1 + "length":126.5 }, { "name":"Confrontation ~ Presto 2009.opus", - "length":-1 + "length":187.5 }, { "name":"Crises of Fate.opus", - "length":-1 + "length":143.5 }, { "name":"Forgotten Legend.opus", - "length":-1 + "length":141.5 }, { "name":"Godot - The Fragrance of Dark Coffee.opus", - "length":-1 + "length":148.5 }, { "name":"Great Revival ~ Franziska von Karma.opus", - "length":-1 + "length":86.5 }, { "name":"Great Revival ~ Miles Edgeworth.opus", - "length":-1 + "length":89.5 }, { "name":"Hotline of Fate.opus", - "length":-1 + "length":51.5 }, { "name":"Interesting People.opus", - "length":-1 + "length":142.5 }, { "name":"Logic and Trick.opus", - "length":-1 + "length":152.5 }, { "name":"Luke Atmey ~ I Just Want Love.opus", - "length":-1 + "length":103.5 }, { "name":"Noisy People.opus", - "length":-1 + "length":91.5 }, { "name":"OBJECTION (AA).opus", - "length":-1 + "length":73.5 }, { "name":"Objection (AJ).opus", - "length":-1 + "length":96.5 }, { "name":"OBJECTION (JFA).opus", - "length":-1 + "length":93.5 }, { "name":"Objection (Miles).opus", - "length":-1 + "length":176 }, { "name":"OBJECTION (T&T).opus", - "length":-1 + "length":119 }, { "name":"Others ~ Guilty love.opus", - "length":-1 + "length":99 }, { "name":"Prelude (AA).opus", - "length":-1 + "length":78 }, { "name":"Prelude (AJ).opus", - "length":-1 + "length":71.5 }, { "name":"Prologue (AA).opus", - "length":-1 + "length":40 }, { "name":"Pursuit (AA) - variation.opus", - "length":-1 + "length":90.5 }, { "name":"Pursuit (AA).opus", - "length":-1 + "length":90.5 }, { "name":"Pursuit (AJ).opus", - "length":-1 + "length":109 }, { "name":"Pursuit (DS).opus", - "length":-1 + "length":226.5 }, { "name":"Pursuit (JFA) - variation.opus", - "length":-1 + "length":76 }, { "name":"Pursuit (JFA).opus", - "length":-1 + "length":120 }, { "name":"Pursuit (Miles).opus", - "length":-1 + "length":197 }, { "name":"Pursuit (T&T) - variation.opus", - "length":-1 + "length":114 }, { "name":"Pursuit (T&T).opus", - "length":-1 + "length":120 }, { "name":"Pursuit ~ I Want to Find the Truth(Orchestra).opus", - "length":-1 + "length":294.5 }, { "name":"Questioning AA (Allegro).opus", - "length":-1 + "length":127 }, { "name":"Questioning AA (Moderato).opus", - "length":-1 + "length":104.5 }, { "name":"Questioning AJ (Allegro).opus", - "length":-1 + "length":103.5 }, { "name":"Questioning AJ (Moderato).opus", - "length":-1 + "length":80.5 }, { "name":"Questioning JFA (Allegro).opus", - "length":-1 + "length":104 }, { "name":"Questioning JFA (Moderato).opus", - "length":-1 + "length":90 }, { "name":"Questioning T&T (Allegro).opus", - "length":-1 + "length":160 }, { "name":"Questioning T&T (Moderato).opus", - "length":-1 + "length":116 }, { "name":"Speak up Pup.opus", - "length":-1 + "length":167 }, { "name":"Suspense (AA).opus", - "length":-1 + "length":92.5 }, { "name":"The Great Truth Burglar.opus", - "length":-1 + "length":149 }, { "name":"Trial (AA).opus", - "length":-1 + "length":109 }, { "name":"Trial (AJ).opus", - "length":-1 + "length":126 }, { "name":"Trial (Miles).opus", - "length":-1 + "length":275.5 } ] } From 11336ca68158db66405964fb0b71845b2aa56786 Mon Sep 17 00:00:00 2001 From: Rosemary Witchaven <32779090+in1tiate@users.noreply.github.com> Date: Mon, 13 Sep 2021 11:32:42 -0500 Subject: [PATCH 8/8] string literal over QString::number --- core/src/area_data.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/core/src/area_data.cpp b/core/src/area_data.cpp index f11d05d..4f2e763 100644 --- a/core/src/area_data.cpp +++ b/core/src/area_data.cpp @@ -559,7 +559,7 @@ void AreaData::switchJukeboxSong() QString l_song_name; if(m_jukebox_queue.size() == 1) { l_song_name = m_jukebox_queue[0]; - emit playJukeboxSong(AOPacket("MC",{l_song_name,QString::number(-1)}), m_index); + emit playJukeboxSong(AOPacket("MC",{l_song_name,"-1"}), m_index); m_jukebox_timer->start(ConfigManager::songInformation(l_song_name) * 1000); } else {