Edit discord_impl.cpp

This commit is contained in:
collecting
2025-10-02 07:10:58 +00:00
parent 26f93ad8ad
commit af86d09b45

View File

@@ -2,44 +2,54 @@
// SPDX-FileCopyrightText: 2025 citron Emulator Project // SPDX-FileCopyrightText: 2025 citron Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later // SPDX-License-Identifier: GPL-2.0-or-later
#include <chrono>
#include <string>
#include <QEventLoop>
#include <QNetworkAccessManager>
#include <QNetworkReply>
#include <discord_rpc.h> #include <discord_rpc.h>
#include <chrono>
#include <string>
#include <thread>
#include <QEventLoop>
#include <QNetworkAccessManager>
#include <QNetworkReply>
#include <fmt/format.h> #include <fmt/format.h>
#include "common/common_types.h" #include "common/common_types.h"
#include "common/string_util.h" #include "common/string_util.h"
#include "core/core.h" #include "core/core.h"
#include "core/loader/loader.h" #include "core/loader/loader.h"
#include "citron/discord_impl.h" #include "citron/discord_impl.h"
#include "citron/uisettings.h" #include "citron/uisettings.h"
static void OnDiscordReady(const DiscordUser* request) {
fmt::print("\n[DISCORD CALLBACK] SUCCESS: Connected to Discord as {}#{}\n\n", request->username, request->discriminator);
}
static void OnDiscordDisconnected(int errcode, const char* message) {
fmt::print("\n[DISCORD CALLBACK] ERROR: Disconnected from Discord. Code: {}, Message: {}\n\n", errcode, message);
}
static void OnDiscordError(int errcode, const char* message) {
fmt::print("\n[DISCORD CALLBACK] ERROR: An error occurred. Code: {}, Message: {}\n\n", errcode, message);
}
namespace DiscordRPC { namespace DiscordRPC {
DiscordImpl::DiscordImpl(Core::System & system_): system { DiscordImpl::DiscordImpl(Core::System& system_) : system{system_} {
system_
} {
DiscordEventHandlers handlers{}; DiscordEventHandlers handlers{};
// The number is the client ID for citron, it's used for images and the handlers.ready = OnDiscordReady;
// application name handlers.disconnected = OnDiscordDisconnected;
handlers.errored = OnDiscordError;
Discord_Initialize("1361252452329848892", &handlers, 1, nullptr); Discord_Initialize("1361252452329848892", &handlers, 1, nullptr);
discord_thread_running = true;
discord_thread = std::thread(&DiscordImpl::ThreadRun, this);
} }
DiscordImpl::~DiscordImpl() { DiscordImpl::~DiscordImpl() {
if (discord_thread_running) {
discord_thread_running = false;
if (discord_thread.joinable()) {
discord_thread.join();
}
}
Discord_ClearPresence(); Discord_ClearPresence();
Discord_Shutdown(); Discord_Shutdown();
} }
@@ -51,28 +61,24 @@ Discord_ClearPresence();
void DiscordImpl::UpdateGameStatus(bool use_default) { void DiscordImpl::UpdateGameStatus(bool use_default) {
const std::string default_text = "Citron Is A Homebrew Emulator For The Nintendo Switch"; const std::string default_text = "Citron Is A Homebrew Emulator For The Nintendo Switch";
const std::string default_image = "citron_logo"; const std::string default_image = "citron_logo";
const std::string tinfoil_base_url = "https://tinfoil.media/ti/";
s64 start_time = std::chrono::duration_cast<std::chrono::seconds>( s64 start_time = std::chrono::duration_cast<std::chrono::seconds>(
std::chrono::system_clock::now().time_since_epoch()) std::chrono::system_clock::now().time_since_epoch())
.count(); .count();
DiscordRichPresence presence{}; DiscordRichPresence presence{};
// Store the URL string to prevent it from being destroyed if (!use_default && !game_title_id.empty()) {
if (!game_title_id.empty()) { game_url = fmt::format("{}{}/256/256", "https://tinfoil.media/ti/", game_title_id);
game_url = fmt::format("{}{}/256/256", tinfoil_base_url, game_title_id);
// Make sure the string stays alive for the duration of the presence
cached_url = game_url; cached_url = game_url;
presence.largeImageKey = cached_url.c_str(); presence.largeImageKey = cached_url.c_str();
} else { } else {
presence.largeImageKey = cached_url.c_str(); presence.largeImageKey = default_image.c_str();
} }
presence.largeImageText = game_title.c_str(); presence.largeImageText = game_title.c_str();
presence.smallImageKey = default_image.c_str(); presence.smallImageKey = default_image.c_str();
presence.smallImageText = default_text.c_str(); presence.smallImageText = default_text.c_str();
// Remove title ID from display, only show game title presence.details = game_title.c_str();
presence.state = game_title.c_str(); presence.state = "Currently in game";
presence.details = "Currently in game";
presence.startTimestamp = start_time; presence.startTimestamp = start_time;
Discord_UpdatePresence(&presence); Discord_UpdatePresence(&presence);
} }
@@ -86,32 +92,22 @@ system.GetAppLoader().ReadTitle(game_title);
system.GetAppLoader().ReadProgramId(program_id); system.GetAppLoader().ReadProgramId(program_id);
game_title_id = fmt::format("{:016X}", program_id); game_title_id = fmt::format("{:016X}", program_id);
fmt::print("Title ID: {}\n", game_title_id);
QNetworkAccessManager manager; QNetworkAccessManager manager;
QNetworkRequest request; QNetworkRequest request;
request.setUrl(QUrl(QString::fromStdString( request.setUrl(QUrl(QString::fromStdString(
fmt::format("https://tinfoil.media/ti/{}/256/256", game_title_id)))); fmt::format("https://tinfoil.media/ti/{}/256/256", game_title_id))));
request.setTransferTimeout(10000); request.setTransferTimeout(5000); // 5 second timeout
QNetworkReply* reply = manager.head(request); QNetworkReply* reply = manager.head(request);
QEventLoop request_event_loop; QEventLoop request_event_loop;
QObject::connect(reply, &QNetworkReply::finished, &request_event_loop, &QEventLoop::quit); QObject::connect(reply, &QNetworkReply::finished, &request_event_loop, &QEventLoop::quit);
request_event_loop.exec(); request_event_loop.exec();
if (reply -> error()) { UpdateGameStatus(reply->error() != QNetworkReply::NoError);
fmt::print("Failed to fetch game image: {} ({})\n", reply -> errorString().toStdString(),
program_id);
}
UpdateGameStatus(reply -> error());
reply->deleteLater(); reply->deleteLater();
return; } else {
}
s64 start_time = std::chrono::duration_cast<std::chrono::seconds>( s64 start_time = std::chrono::duration_cast<std::chrono::seconds>(
std::chrono::system_clock::now().time_since_epoch()) std::chrono::system_clock::now().time_since_epoch())
.count(); .count();
DiscordRichPresence presence{}; DiscordRichPresence presence{};
presence.largeImageKey = default_image.c_str(); presence.largeImageKey = default_image.c_str();
presence.largeImageText = default_text.c_str(); presence.largeImageText = default_text.c_str();
@@ -119,4 +115,14 @@ presence.details = "Currently not in game";
presence.startTimestamp = start_time; presence.startTimestamp = start_time;
Discord_UpdatePresence(&presence); Discord_UpdatePresence(&presence);
} }
}
void DiscordImpl::ThreadRun() {
while (discord_thread_running) {
Update(); // This is the line that was likely missing.
Discord_RunCallbacks();
std::this_thread::sleep_for(std::chrono::seconds(15));
}
}
} // namespace DiscordRPC } // namespace DiscordRPC