#include "qthostinterface.h" #include "common/assert.h" #include "common/audio_stream.h" #include "common/byte_stream.h" #include "common/file_system.h" #include "common/log.h" #include "common/string_util.h" #include "core/cheats.h" #include "core/controller.h" #include "core/gpu.h" #include "core/memory_card.h" #include "core/system.h" #include "frontend-common/fullscreen_ui.h" #include "frontend-common/game_list.h" #include "frontend-common/imgui_fullscreen.h" #include "frontend-common/imgui_styles.h" #include "frontend-common/ini_settings_interface.h" #include "frontend-common/opengl_host_display.h" #include "frontend-common/sdl_audio_stream.h" #include "frontend-common/sdl_controller_interface.h" #include "frontend-common/vulkan_host_display.h" #include "imgui.h" #include "mainwindow.h" #include "qtdisplaywidget.h" #include "qtprogresscallback.h" #include "qtutils.h" #include #include #include #include #include #include #include #include #include #include #include #include #include Log_SetChannel(QtHostInterface); #ifdef _WIN32 #include "common/windows_headers.h" #include "frontend-common/d3d11_host_display.h" #include "frontend-common/d3d12_host_display.h" #include #include #endif #ifdef WITH_CHEEVOS #include "frontend-common/cheevos.h" #endif QtHostInterface::QtHostInterface(QObject* parent) : QObject(parent), CommonHostInterface() { qRegisterMetaType>(); qRegisterMetaType(); qRegisterMetaType(); } QtHostInterface::~QtHostInterface() { Assert(!m_display); } const char* QtHostInterface::GetFrontendName() const { return "DuckStation Qt Frontend"; } std::vector> QtHostInterface::getAvailableLanguageList() { return {{QStringLiteral("English"), QStringLiteral("en")}, {QStringLiteral("Deutsch"), QStringLiteral("de")}, {QStringLiteral("Español"), QStringLiteral("es")}, {QStringLiteral("Français"), QStringLiteral("fr")}, {QStringLiteral("עברית"), QStringLiteral("he")}, {QStringLiteral("日本語"), QStringLiteral("ja")}, {QStringLiteral("Italiano"), QStringLiteral("it")}, {QStringLiteral("Nederlands"), QStringLiteral("nl")}, {QStringLiteral("Polski"), QStringLiteral("pl")}, {QStringLiteral("Português (Pt)"), QStringLiteral("pt-pt")}, {QStringLiteral("Português (Br)"), QStringLiteral("pt-br")}, {QStringLiteral("Русский"), QStringLiteral("ru")}, {QStringLiteral("Türkçe"), QStringLiteral("tr")}, {QStringLiteral("简体中文"), QStringLiteral("zh-cn")}}; } bool QtHostInterface::Initialize() { createThread(); if (!m_worker_thread->waitForInit()) return false; installTranslator(); return true; } void QtHostInterface::Shutdown() { stopThread(); } bool QtHostInterface::initializeOnThread() { SetUserDirectory(); m_settings_interface = std::make_unique(GetSettingsFileName()); if (!CommonHostInterface::Initialize()) return false; // imgui setup setImGuiFont(); setImGuiKeyMap(); // make sure the controllers have been detected if (m_controller_interface) m_controller_interface->PollEvents(); // bind buttons/axises createBackgroundControllerPollTimer(); startBackgroundControllerPollTimer(); updateInputMap(); return true; } void QtHostInterface::shutdownOnThread() { destroyBackgroundControllerPollTimer(); CommonHostInterface::Shutdown(); } void QtHostInterface::installTranslator() { std::string language = GetStringSettingValue("Main", "Language", ""); if (language.empty()) language = "en"; const QString path = QStringLiteral("%1/translations/duckstation-qt_%3.qm").arg(qApp->applicationDirPath()).arg(language.c_str()); if (!QFile::exists(path)) { QMessageBox::warning( nullptr, QStringLiteral("Translation Error"), QStringLiteral("Failed to find translation file for language '%1':\n%2").arg(language.c_str()).arg(path)); return; } auto translator = std::make_unique(qApp); if (!translator->load(path)) { QMessageBox::warning( nullptr, QStringLiteral("Translation Error"), QStringLiteral("Failed to load translation file for language '%1':\n%2").arg(language.c_str()).arg(path)); return; } Log_InfoPrintf("Loaded translation file for language '%s'", language.c_str()); qApp->installTranslator(translator.release()); } void QtHostInterface::ReportError(const char* message) { HostInterface::ReportError(message); const bool was_fullscreen = m_is_fullscreen; if (was_fullscreen) SetFullscreen(false); emit errorReported(QString::fromUtf8(message)); if (was_fullscreen) SetFullscreen(true); } void QtHostInterface::ReportMessage(const char* message) { HostInterface::ReportMessage(message); emit messageReported(QString::fromUtf8(message)); } void QtHostInterface::ReportDebuggerMessage(const char* message) { HostInterface::ReportDebuggerMessage(message); emit debuggerMessageReported(QString::fromUtf8(message)); } bool QtHostInterface::ConfirmMessage(const char* message) { const bool was_fullscreen = m_is_fullscreen; if (was_fullscreen) SetFullscreen(false); const bool result = messageConfirmed(QString::fromUtf8(message)); if (was_fullscreen) SetFullscreen(true); return result; } void QtHostInterface::SetBoolSettingValue(const char* section, const char* key, bool value) { std::lock_guard guard(m_settings_mutex); m_settings_interface->SetBoolValue(section, key, value); queueSettingsSave(); } void QtHostInterface::SetIntSettingValue(const char* section, const char* key, int value) { std::lock_guard guard(m_settings_mutex); m_settings_interface->SetIntValue(section, key, value); queueSettingsSave(); } void QtHostInterface::SetFloatSettingValue(const char* section, const char* key, float value) { std::lock_guard guard(m_settings_mutex); m_settings_interface->SetFloatValue(section, key, value); queueSettingsSave(); } void QtHostInterface::SetStringSettingValue(const char* section, const char* key, const char* value) { std::lock_guard guard(m_settings_mutex); m_settings_interface->SetStringValue(section, key, value); queueSettingsSave(); } void QtHostInterface::SetStringListSettingValue(const char* section, const char* key, const std::vector& values) { std::lock_guard guard(m_settings_mutex); m_settings_interface->SetStringList(section, key, values); queueSettingsSave(); } bool QtHostInterface::AddValueToStringList(const char* section, const char* key, const char* value) { std::lock_guard guard(m_settings_mutex); if (!m_settings_interface->AddToStringList(section, key, value)) return false; queueSettingsSave(); return true; } bool QtHostInterface::RemoveValueFromStringList(const char* section, const char* key, const char* value) { std::lock_guard guard(m_settings_mutex); if (!m_settings_interface->RemoveFromStringList(section, key, value)) return false; queueSettingsSave(); return true; } void QtHostInterface::RemoveSettingValue(const char* section, const char* key) { std::lock_guard guard(m_settings_mutex); m_settings_interface->DeleteValue(section, key); queueSettingsSave(); } void QtHostInterface::queueSettingsSave() { if (m_settings_save_timer) return; m_settings_save_timer = std::make_unique(); connect(m_settings_save_timer.get(), &QTimer::timeout, this, &QtHostInterface::doSaveSettings); m_settings_save_timer->setSingleShot(true); m_settings_save_timer->start(SETTINGS_SAVE_DELAY); m_settings_save_timer->moveToThread(m_worker_thread); } void QtHostInterface::doSaveSettings() { std::lock_guard guard(m_settings_mutex); m_settings_interface->Save(); m_settings_save_timer.reset(); } void QtHostInterface::setDefaultSettings() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "setDefaultSettings", Qt::QueuedConnection); return; } SetDefaultSettings(); } void QtHostInterface::SetDefaultSettings() { CommonHostInterface::SetDefaultSettings(); checkRenderToMainState(); queueSettingsSave(); emit settingsResetToDefault(); } void QtHostInterface::applySettings(bool display_osd_messages /* = false */) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "applySettings", Qt::QueuedConnection, Q_ARG(bool, display_osd_messages)); return; } ApplySettings(display_osd_messages); } void QtHostInterface::ApplySettings(bool display_osd_messages) { CommonHostInterface::ApplySettings(display_osd_messages); checkRenderToMainState(); } void QtHostInterface::checkRenderToMainState() { // detect when render-to-main flag changes if (!System::IsShutdown()) { const bool render_to_main = m_settings_interface->GetBoolValue("Main", "RenderToMainWindow", true); if (m_display && !m_is_fullscreen && render_to_main != m_is_rendering_to_main) { m_is_rendering_to_main = render_to_main; updateDisplayState(); } else if (!m_fullscreen_ui_enabled) { renderDisplay(); } } } void QtHostInterface::refreshGameList(bool invalidate_cache /* = false */, bool invalidate_database /* = false */) { Assert(!isOnWorkerThread()); std::lock_guard lock(m_settings_mutex); m_game_list->SetSearchDirectoriesFromSettings(*m_settings_interface.get()); QtProgressCallback progress(m_main_window, invalidate_cache ? 0.0f : 1.0f); m_game_list->Refresh(invalidate_cache, invalidate_database, &progress); emit gameListRefreshed(); } void QtHostInterface::setMainWindow(MainWindow* window) { DebugAssert((!m_main_window && window) || (m_main_window && !window)); m_main_window = window; } void QtHostInterface::bootSystem(std::shared_ptr params) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "bootSystem", Qt::QueuedConnection, Q_ARG(std::shared_ptr, std::move(params))); return; } emit emulationStarting(); if (!BootSystem(std::move(params))) return; // force a frame to be drawn to repaint the window renderDisplay(); } void QtHostInterface::resumeSystemFromState(const QString& filename, bool boot_on_failure) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "resumeSystemFromState", Qt::QueuedConnection, Q_ARG(const QString&, filename), Q_ARG(bool, boot_on_failure)); return; } emit emulationStarting(); if (filename.isEmpty()) ResumeSystemFromMostRecentState(); else ResumeSystemFromState(filename.toStdString().c_str(), boot_on_failure); } void QtHostInterface::resumeSystemFromMostRecentState() { std::string state_filename = GetMostRecentResumeSaveStatePath(); if (state_filename.empty()) { emit errorReported(tr("No resume save state found.")); return; } loadState(QString::fromStdString(state_filename)); } void QtHostInterface::onDisplayWindowKeyEvent(int key, int mods, bool pressed) { DebugAssert(isOnWorkerThread()); ImGuiIO& io = ImGui::GetIO(); const u32 masked_key = static_cast(key) & IMGUI_KEY_MASK; if (masked_key < countof(ImGuiIO::KeysDown)) io.KeysDown[masked_key] = pressed; if (m_fullscreen_ui_enabled && FullscreenUI::IsBindingInput()) { QString key_string(QtUtils::KeyEventToString(key, static_cast(mods))); if (!key_string.isEmpty()) { if (FullscreenUI::HandleKeyboardBinding(key_string.toUtf8().constData(), pressed)) return; } } if (io.WantCaptureKeyboard) return; const int key_id = QtUtils::KeyEventToInt(key, static_cast(mods)); HandleHostKeyEvent(key_id & ~Qt::KeyboardModifierMask, key_id & Qt::KeyboardModifierMask, pressed); } void QtHostInterface::onDisplayWindowMouseMoveEvent(int x, int y) { // display might be null here if the event happened after shutdown DebugAssert(isOnWorkerThread()); if (!m_display) return; ImGuiIO& io = ImGui::GetIO(); io.MousePos[0] = static_cast(x); io.MousePos[1] = static_cast(y); m_display->SetMousePosition(x, y); } void QtHostInterface::onDisplayWindowMouseButtonEvent(int button, bool pressed) { DebugAssert(isOnWorkerThread()); if (ImGui::GetCurrentContext() && (button > 0 && button <= static_cast(countof(ImGuiIO::MouseDown)))) { ImGuiIO& io = ImGui::GetIO(); io.MouseDown[button - 1] = pressed; if (io.WantCaptureMouse) { // don't consume input events if it's hitting the UI instead return; } } HandleHostMouseEvent(button, pressed); } void QtHostInterface::onDisplayWindowMouseWheelEvent(const QPoint& delta_angle) { DebugAssert(isOnWorkerThread()); if (ImGui::GetCurrentContext()) { ImGuiIO& io = ImGui::GetIO(); if (delta_angle.x() > 0) io.MouseWheelH += 1.0f; else if (delta_angle.x() < 0) io.MouseWheelH -= 1.0f; if (delta_angle.y() > 0) io.MouseWheel += 1.0f; else if (delta_angle.y() < 0) io.MouseWheel -= 1.0f; if (io.WantCaptureMouse) { // don't consume input events if it's hitting the UI instead return; } } // HandleHostMouseWheelEvent(delta_angle.x(), delta_angle.y()); } void QtHostInterface::onDisplayWindowResized(int width, int height) { // this can be null if it was destroyed and the main thread is late catching up if (!m_display) return; Log_DevPrintf("Display window resized to %dx%d", width, height); m_display->ResizeRenderWindow(width, height); OnHostDisplayResized(); // re-render the display, since otherwise it will be out of date and stretched if paused if (!System::IsShutdown()) { if (m_is_exclusive_fullscreen && !m_display->IsFullscreen()) { // we lost exclusive fullscreen, switch to borderless AddOSDMessage(TranslateStdString("OSDMessage", "Lost exclusive fullscreen."), 10.0f); m_is_exclusive_fullscreen = false; m_is_fullscreen = false; m_lost_exclusive_fullscreen = true; } // force redraw if we're paused if (!m_fullscreen_ui_enabled) renderDisplay(); } } void QtHostInterface::onDisplayWindowFocused() { if (!m_display || !m_lost_exclusive_fullscreen) return; // try to restore exclusive fullscreen m_lost_exclusive_fullscreen = false; m_is_exclusive_fullscreen = true; m_is_fullscreen = true; updateDisplayState(); } void QtHostInterface::redrawDisplayWindow() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "redrawDisplayWindow", Qt::QueuedConnection); return; } if (!m_display || System::IsShutdown()) return; renderDisplay(); } void QtHostInterface::toggleFullscreen() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "toggleFullscreen", Qt::QueuedConnection); return; } SetFullscreen(!m_is_fullscreen); } bool QtHostInterface::AcquireHostDisplay() { Assert(!m_display); m_is_rendering_to_main = m_settings_interface->GetBoolValue("Main", "RenderToMainWindow", true); QtDisplayWidget* display_widget = createDisplayRequested(m_worker_thread, m_is_fullscreen, m_is_rendering_to_main); if (!display_widget || !m_display->HasRenderDevice()) { emit destroyDisplayRequested(); m_display.reset(); return false; } if (!m_display->MakeRenderContextCurrent() || !m_display->InitializeRenderDevice(GetShaderCacheBasePath(), g_settings.gpu_use_debug_device, g_settings.gpu_threaded_presentation) || !CreateHostDisplayResources()) { ReleaseHostDisplayResources(); m_display->DestroyRenderDevice(); emit destroyDisplayRequested(); m_display.reset(); return false; } m_is_exclusive_fullscreen = m_display->IsFullscreen(); return true; } HostDisplay* QtHostInterface::createHostDisplay() { switch (g_settings.gpu_renderer) { case GPURenderer::HardwareVulkan: m_display = std::make_unique(); break; case GPURenderer::HardwareOpenGL: #ifndef _WIN32 default: #endif m_display = std::make_unique(); break; #ifdef _WIN32 case GPURenderer::HardwareD3D12: m_display = std::make_unique(); break; case GPURenderer::HardwareD3D11: default: m_display = std::make_unique(); break; #endif } return m_display.get(); } void QtHostInterface::connectDisplaySignals(QtDisplayWidget* widget) { widget->disconnect(this); connect(widget, &QtDisplayWidget::windowFocusEvent, this, &QtHostInterface::onDisplayWindowFocused); connect(widget, &QtDisplayWidget::windowResizedEvent, this, &QtHostInterface::onDisplayWindowResized); connect(widget, &QtDisplayWidget::windowRestoredEvent, this, &QtHostInterface::redrawDisplayWindow); connect(widget, &QtDisplayWidget::windowClosedEvent, this, &QtHostInterface::powerOffSystem, Qt::BlockingQueuedConnection); connect(widget, &QtDisplayWidget::windowKeyEvent, this, &QtHostInterface::onDisplayWindowKeyEvent); connect(widget, &QtDisplayWidget::windowMouseMoveEvent, this, &QtHostInterface::onDisplayWindowMouseMoveEvent); connect(widget, &QtDisplayWidget::windowMouseButtonEvent, this, &QtHostInterface::onDisplayWindowMouseButtonEvent); connect(widget, &QtDisplayWidget::windowMouseWheelEvent, this, &QtHostInterface::onDisplayWindowMouseWheelEvent); } void QtHostInterface::updateDisplayState() { if (!m_display) return; // this expects the context to get moved back to us afterwards m_display->DoneRenderContextCurrent(); QtDisplayWidget* display_widget = updateDisplayRequested(m_worker_thread, m_is_fullscreen, m_is_rendering_to_main && !m_is_fullscreen); if (!display_widget || !m_display->MakeRenderContextCurrent()) Panic("Failed to make device context current after updating"); m_is_exclusive_fullscreen = m_display->IsFullscreen(); OnHostDisplayResized(); if (!System::IsShutdown()) { UpdateSoftwareCursor(); if (!m_fullscreen_ui_enabled) redrawDisplayWindow(); } UpdateSpeedLimiterState(); } void QtHostInterface::ReleaseHostDisplay() { Assert(m_display); ReleaseHostDisplayResources(); m_display->DestroyRenderDevice(); emit destroyDisplayRequested(); m_display.reset(); m_is_fullscreen = false; } bool QtHostInterface::IsFullscreen() const { return m_is_fullscreen; } bool QtHostInterface::SetFullscreen(bool enabled) { if (m_is_fullscreen == enabled) return true; m_is_fullscreen = enabled; updateDisplayState(); return true; } bool QtHostInterface::RequestRenderWindowSize(s32 new_window_width, s32 new_window_height) { if (new_window_width <= 0 || new_window_height <= 0 || m_is_fullscreen || m_is_exclusive_fullscreen) return false; emit displaySizeRequested(new_window_width, new_window_height); return true; } void* QtHostInterface::GetTopLevelWindowHandle() const { return reinterpret_cast(m_main_window->winId()); } void QtHostInterface::RequestExit() { emit exitRequested(); } std::optional QtHostInterface::GetHostKeyCode(const std::string_view key_code) const { const std::optional code = QtUtils::ParseKeyString(QString::fromUtf8(key_code.data(), static_cast(key_code.length()))); if (!code) return std::nullopt; return static_cast(*code); } void QtHostInterface::OnSystemCreated() { CommonHostInterface::OnSystemCreated(); wakeThread(); stopBackgroundControllerPollTimer(); emit emulationStarted(); emit emulationPaused(false); } void QtHostInterface::OnSystemPaused(bool paused) { CommonHostInterface::OnSystemPaused(paused); emit emulationPaused(paused); if (!paused) { wakeThread(); stopBackgroundControllerPollTimer(); emit focusDisplayWidgetRequested(); } else { startBackgroundControllerPollTimer(); } } void QtHostInterface::OnSystemDestroyed() { CommonHostInterface::OnSystemDestroyed(); ClearOSDMessages(); startBackgroundControllerPollTimer(); emit emulationStopped(); } void QtHostInterface::OnSystemPerformanceCountersUpdated() { HostInterface::OnSystemPerformanceCountersUpdated(); GPURenderer renderer = GPURenderer::Count; u32 render_width = 0; u32 render_height = 0; bool render_interlaced = false; if (g_gpu) { renderer = g_gpu->GetRendererType(); std::tie(render_width, render_height) = g_gpu->GetEffectiveDisplayResolution(); render_interlaced = g_gpu->IsInterlacedDisplayEnabled(); } emit systemPerformanceCountersUpdated(System::GetEmulationSpeed(), System::GetFPS(), System::GetVPS(), System::GetAverageFrameTime(), System::GetWorstFrameTime(), renderer, render_width, render_height, render_interlaced); } void QtHostInterface::OnRunningGameChanged(const std::string& path, CDImage* image, const std::string& game_code, const std::string& game_title) { CommonHostInterface::OnRunningGameChanged(path, image, game_code, game_title); if (!System::IsShutdown()) { emit runningGameChanged(QString::fromStdString(System::GetRunningPath()), QString::fromStdString(System::GetRunningCode()), QString::fromStdString(System::GetRunningTitle())); } else { emit runningGameChanged(QString(), QString(), QString()); } } void QtHostInterface::SetDefaultSettings(SettingsInterface& si) { CommonHostInterface::SetDefaultSettings(si); si.SetStringValue("Hotkeys", "PowerOff", "Keyboard/Escape"); si.SetStringValue("Hotkeys", "LoadSelectedSaveState", "Keyboard/F1"); si.SetStringValue("Hotkeys", "SaveSelectedSaveState", "Keyboard/F2"); si.SetStringValue("Hotkeys", "SelectPreviousSaveStateSlot", "Keyboard/F3"); si.SetStringValue("Hotkeys", "SelectNextSaveStateSlot", "Keyboard/F4"); si.SetBoolValue("Main", "RenderToMainWindow", true); } void QtHostInterface::SetMouseMode(bool relative, bool hide_cursor) { emit mouseModeRequested(relative, hide_cursor); } void QtHostInterface::updateInputMap() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "updateInputMap", Qt::QueuedConnection); return; } std::lock_guard lock(m_settings_mutex); CommonHostInterface::UpdateInputMap(*m_settings_interface.get()); } void QtHostInterface::applyInputProfile(const QString& profile_path) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "applyInputProfile", Qt::QueuedConnection, Q_ARG(const QString&, profile_path)); return; } ApplyInputProfile(profile_path.toUtf8().data()); emit inputProfileLoaded(); } void QtHostInterface::saveInputProfile(const QString& profile_name) { std::lock_guard lock(m_settings_mutex); SaveInputProfile(profile_name.toUtf8().data()); } QString QtHostInterface::getUserDirectoryRelativePath(const QString& arg) const { QString result = QString::fromStdString(m_user_directory); result += FS_OSPATH_SEPARATOR_CHARACTER; result += arg; return result; } QString QtHostInterface::getProgramDirectoryRelativePath(const QString& arg) const { QString result = QString::fromStdString(m_program_directory); result += FS_OSPATH_SEPARATOR_CHARACTER; result += arg; return result; } QString QtHostInterface::getProgramDirectory() const { return QString::fromStdString(m_program_directory); } void QtHostInterface::powerOffSystem() { if (!isOnWorkerThread()) { System::CancelPendingStartup(); QMetaObject::invokeMethod(this, "powerOffSystem", Qt::QueuedConnection); return; } PowerOffSystem(ShouldSaveResumeState()); } void QtHostInterface::powerOffSystemWithoutSaving() { if (!isOnWorkerThread()) { System::CancelPendingStartup(); QMetaObject::invokeMethod(this, "powerOffSystemWithoutSaving", Qt::QueuedConnection); return; } PowerOffSystem(false); } void QtHostInterface::synchronousPowerOffSystem() { if (!isOnWorkerThread()) { System::CancelPendingStartup(); QMetaObject::invokeMethod(this, "powerOffSystem", Qt::BlockingQueuedConnection); } else { powerOffSystem(); } } void QtHostInterface::resetSystem() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "resetSystem", Qt::QueuedConnection); return; } if (System::IsShutdown()) { Log_ErrorPrintf("resetSystem() called without system"); return; } HostInterface::ResetSystem(); } void QtHostInterface::pauseSystem(bool paused, bool wait_until_paused /* = false */) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "pauseSystem", wait_until_paused ? Qt::BlockingQueuedConnection : Qt::QueuedConnection, Q_ARG(bool, paused), Q_ARG(bool, wait_until_paused)); return; } CommonHostInterface::PauseSystem(paused); } void QtHostInterface::changeDisc(const QString& new_disc_filename) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "changeDisc", Qt::QueuedConnection, Q_ARG(const QString&, new_disc_filename)); return; } if (System::IsShutdown()) return; if (!new_disc_filename.isEmpty()) System::InsertMedia(new_disc_filename.toStdString().c_str()); else System::RemoveMedia(); } void QtHostInterface::changeDiscFromPlaylist(quint32 index) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "changeDiscFromPlaylist", Qt::QueuedConnection, Q_ARG(quint32, index)); return; } if (System::IsShutdown()) return; if (!System::SwitchMediaSubImage(index)) ReportFormattedError("Failed to switch to subimage %u", index); } static QString FormatTimestampForSaveStateMenu(u64 timestamp) { const QDateTime qtime(QDateTime::fromSecsSinceEpoch(static_cast(timestamp))); return qtime.toString(QLocale::system().dateTimeFormat(QLocale::ShortFormat)); } void QtHostInterface::populateLoadStateMenu(const char* game_code, QMenu* menu) { auto add_slot = [this, game_code, menu](const QString& title, const QString& empty_title, bool global, s32 slot) { std::optional ssi = GetSaveStateInfo(global ? nullptr : game_code, slot); const QString menu_title = ssi.has_value() ? title.arg(slot).arg(FormatTimestampForSaveStateMenu(ssi->timestamp)) : empty_title.arg(slot); QAction* load_action = menu->addAction(menu_title); load_action->setEnabled(ssi.has_value()); if (ssi.has_value()) { const QString path(QString::fromStdString(ssi->path)); connect(load_action, &QAction::triggered, [this, path]() { loadState(path); }); } }; menu->clear(); connect(menu->addAction(tr("Load From File...")), &QAction::triggered, [this]() { const QString path( QFileDialog::getOpenFileName(m_main_window, tr("Select Save State File"), QString(), tr("Save States (*.sav)"))); if (path.isEmpty()) return; loadState(path); }); QAction* load_from_state = menu->addAction(tr("Undo Load State")); load_from_state->setEnabled(CanUndoLoadState()); connect(load_from_state, &QAction::triggered, this, &QtHostInterface::undoLoadState); menu->addSeparator(); if (game_code && std::strlen(game_code) > 0) { for (u32 slot = 1; slot <= PER_GAME_SAVE_STATE_SLOTS; slot++) add_slot(tr("Game Save %1 (%2)"), tr("Game Save %1 (Empty)"), false, static_cast(slot)); menu->addSeparator(); } for (u32 slot = 1; slot <= GLOBAL_SAVE_STATE_SLOTS; slot++) add_slot(tr("Global Save %1 (%2)"), tr("Global Save %1 (Empty)"), true, static_cast(slot)); } void QtHostInterface::populateSaveStateMenu(const char* game_code, QMenu* menu) { auto add_slot = [this, game_code, menu](const QString& title, const QString& empty_title, bool global, s32 slot) { std::optional ssi = GetSaveStateInfo(global ? nullptr : game_code, slot); const QString menu_title = ssi.has_value() ? title.arg(slot).arg(FormatTimestampForSaveStateMenu(ssi->timestamp)) : empty_title.arg(slot); QAction* save_action = menu->addAction(menu_title); connect(save_action, &QAction::triggered, [this, global, slot]() { saveState(global, slot); }); }; menu->clear(); connect(menu->addAction(tr("Save To File...")), &QAction::triggered, [this]() { if (!System::IsValid()) return; const QString path( QFileDialog::getSaveFileName(m_main_window, tr("Select Save State File"), QString(), tr("Save States (*.sav)"))); if (path.isEmpty()) return; SaveState(path.toUtf8().constData()); }); menu->addSeparator(); if (game_code && std::strlen(game_code) > 0) { for (u32 slot = 1; slot <= PER_GAME_SAVE_STATE_SLOTS; slot++) add_slot(tr("Game Save %1 (%2)"), tr("Game Save %1 (Empty)"), false, static_cast(slot)); menu->addSeparator(); } for (u32 slot = 1; slot <= GLOBAL_SAVE_STATE_SLOTS; slot++) add_slot(tr("Global Save %1 (%2)"), tr("Global Save %1 (Empty)"), true, static_cast(slot)); } void QtHostInterface::populateGameListContextMenu(const GameListEntry* entry, QWidget* parent_window, QMenu* menu) { QAction* resume_action = menu->addAction(tr("Resume")); resume_action->setEnabled(false); QMenu* load_state_menu = menu->addMenu(tr("Load State")); load_state_menu->setEnabled(false); if (!entry->code.empty()) { const std::vector available_states(GetAvailableSaveStates(entry->code.c_str())); const QString timestamp_format = QLocale::system().dateTimeFormat(QLocale::ShortFormat); const bool challenge_mode = IsCheevosChallengeModeActive(); for (const SaveStateInfo& ssi : available_states) { if (ssi.global) continue; const s32 slot = ssi.slot; const QDateTime timestamp(QDateTime::fromSecsSinceEpoch(static_cast(ssi.timestamp))); const QString timestamp_str(timestamp.toString(timestamp_format)); const QString path(QString::fromStdString(ssi.path)); QAction* action; if (slot < 0) { resume_action->setText(tr("Resume (%1)").arg(timestamp_str)); resume_action->setEnabled(!challenge_mode); action = resume_action; } else { load_state_menu->setEnabled(true); action = load_state_menu->addAction(tr("Game Save %1 (%2)").arg(slot).arg(timestamp_str)); } action->setDisabled(challenge_mode); connect(action, &QAction::triggered, [this, path]() { loadState(path); }); } } QAction* open_memory_cards_action = menu->addAction(tr("Edit Memory Cards...")); connect(open_memory_cards_action, &QAction::triggered, [this, entry]() { QString paths[2]; for (u32 i = 0; i < 2; i++) { MemoryCardType type = g_settings.memory_card_types[i]; if (entry->code.empty() && type == MemoryCardType::PerGame) type = MemoryCardType::Shared; switch (type) { case MemoryCardType::None: continue; case MemoryCardType::Shared: if (g_settings.memory_card_paths[i].empty()) { paths[i] = QString::fromStdString(GetSharedMemoryCardPath(i)); } else { QFileInfo path(QString::fromStdString(g_settings.memory_card_paths[i])); path.makeAbsolute(); paths[i] = QDir::toNativeSeparators(path.canonicalFilePath()); } break; case MemoryCardType::PerGame: paths[i] = QString::fromStdString(GetGameMemoryCardPath(entry->code.c_str(), i)); break; case MemoryCardType::PerGameTitle: paths[i] = QString::fromStdString( GetGameMemoryCardPath(MemoryCard::SanitizeGameTitleForFileName(entry->title).c_str(), i)); break; case MemoryCardType::PerGameFileTitle: { const std::string display_name(FileSystem::GetDisplayNameFromPath(entry->path)); paths[i] = QString::fromStdString(GetGameMemoryCardPath( MemoryCard::SanitizeGameTitleForFileName(FileSystem::GetFileTitleFromPath(display_name)).c_str(), i)); } break; default: break; } } m_main_window->openMemoryCardEditor(paths[0], paths[1]); }); const bool has_any_states = resume_action->isEnabled() || load_state_menu->isEnabled(); QAction* delete_save_states_action = menu->addAction(tr("Delete Save States...")); delete_save_states_action->setEnabled(has_any_states); if (has_any_states) { connect(delete_save_states_action, &QAction::triggered, [this, parent_window, entry] { if (QMessageBox::warning( parent_window, tr("Confirm Save State Deletion"), tr("Are you sure you want to delete all save states for %1?\n\nThe saves will not be recoverable.") .arg(QString::fromStdString(entry->code)), QMessageBox::Yes, QMessageBox::No) != QMessageBox::Yes) { return; } DeleteSaveStates(entry->code.c_str(), true); }); } } void QtHostInterface::populateChangeDiscSubImageMenu(QMenu* menu, QActionGroup* action_group) { if (!System::IsValid() || !System::HasMediaSubImages()) return; const u32 count = System::GetMediaSubImageCount(); const u32 current = System::GetMediaSubImageIndex(); for (u32 i = 0; i < count; i++) { QAction* action = action_group->addAction(QString::fromStdString(System::GetMediaSubImageTitle(i))); action->setCheckable(true); action->setChecked(i == current); connect(action, &QAction::triggered, [this, i]() { changeDiscFromPlaylist(i); }); menu->addAction(action); } } void QtHostInterface::populateCheatsMenu(QMenu* menu) { Assert(!isOnWorkerThread()); if (!System::IsValid()) return; const bool has_cheat_list = System::HasCheatList(); QMenu* enabled_menu = menu->addMenu(tr("&Enabled Cheats")); enabled_menu->setEnabled(false); QMenu* apply_menu = menu->addMenu(tr("&Apply Cheats")); apply_menu->setEnabled(false); if (has_cheat_list) { CheatList* cl = System::GetCheatList(); for (const std::string& group : cl->GetCodeGroups()) { QMenu* enabled_submenu = nullptr; QMenu* apply_submenu = nullptr; for (u32 i = 0; i < cl->GetCodeCount(); i++) { CheatCode& cc = cl->GetCode(i); if (cc.group != group) continue; QString desc(QString::fromStdString(cc.description)); if (cc.IsManuallyActivated()) { if (!apply_submenu) { apply_menu->setEnabled(true); apply_submenu = apply_menu->addMenu(QString::fromStdString(group)); } QAction* action = apply_submenu->addAction(desc); connect(action, &QAction::triggered, [this, i]() { applyCheat(i); }); } else { if (!enabled_submenu) { enabled_menu->setEnabled(true); enabled_submenu = enabled_menu->addMenu(QString::fromStdString(group)); } QAction* action = enabled_submenu->addAction(desc); action->setCheckable(true); action->setChecked(cc.enabled); connect(action, &QAction::toggled, [this, i](bool enabled) { setCheatEnabled(i, enabled); }); } } } } } void QtHostInterface::loadCheatList(const QString& filename) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "loadCheatList", Qt::QueuedConnection, Q_ARG(const QString&, filename)); return; } LoadCheatList(filename.toUtf8().constData()); } void QtHostInterface::setCheatEnabled(quint32 index, bool enabled) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "setCheatEnabled", Qt::QueuedConnection, Q_ARG(quint32, index), Q_ARG(bool, enabled)); return; } SetCheatCodeState(index, enabled, g_settings.auto_load_cheats); } void QtHostInterface::applyCheat(quint32 index) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "applyCheat", Qt::QueuedConnection, Q_ARG(quint32, index)); return; } ApplyCheatCode(index); } void QtHostInterface::reloadPostProcessingShaders() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "reloadPostProcessingShaders", Qt::QueuedConnection); return; } ReloadPostProcessingShaders(); } void QtHostInterface::requestRenderWindowScale(qreal scale) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "requestRenderWindowScale", Qt::QueuedConnection, Q_ARG(qreal, scale)); return; } RequestRenderWindowScale(scale); } void QtHostInterface::executeOnEmulationThread(std::function callback, bool wait) { if (isOnWorkerThread()) { callback(); if (wait) m_worker_thread_sync_execute_done.Signal(); return; } QMetaObject::invokeMethod(this, "executeOnEmulationThread", Qt::QueuedConnection, Q_ARG(std::function, std::move(callback)), Q_ARG(bool, wait)); if (wait) { // don't deadlock while (!m_worker_thread_sync_execute_done.TryWait(10)) qApp->processEvents(QEventLoop::ExcludeSocketNotifiers); m_worker_thread_sync_execute_done.Reset(); } } void QtHostInterface::RunLater(std::function func) { QMetaObject::invokeMethod(this, "executeOnEmulationThread", Qt::QueuedConnection, Q_ARG(std::function, std::move(func)), Q_ARG(bool, false)); } void QtHostInterface::loadState(const QString& filename) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "loadState", Qt::QueuedConnection, Q_ARG(const QString&, filename)); return; } if (System::IsShutdown()) emit emulationStarting(); LoadState(filename.toStdString().c_str()); } void QtHostInterface::loadState(bool global, qint32 slot) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "loadState", Qt::QueuedConnection, Q_ARG(bool, global), Q_ARG(qint32, slot)); return; } LoadState(global, slot); } void QtHostInterface::saveState(const QString& filename, bool block_until_done /* = false */) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "saveState", block_until_done ? Qt::BlockingQueuedConnection : Qt::QueuedConnection, Q_ARG(const QString&, filename), Q_ARG(bool, block_until_done)); return; } if (!System::IsShutdown()) SaveState(filename.toUtf8().data()); } void QtHostInterface::saveState(bool global, qint32 slot, bool block_until_done /* = false */) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "saveState", block_until_done ? Qt::BlockingQueuedConnection : Qt::QueuedConnection, Q_ARG(bool, global), Q_ARG(qint32, slot), Q_ARG(bool, block_until_done)); return; } if (!System::IsShutdown()) SaveState(global, slot); } void QtHostInterface::undoLoadState() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "undoLoadState", Qt::QueuedConnection); return; } UndoLoadState(); } void QtHostInterface::setAudioOutputVolume(int volume, int fast_forward_volume) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "setAudioOutputVolume", Qt::QueuedConnection, Q_ARG(int, volume), Q_ARG(int, fast_forward_volume)); return; } g_settings.audio_output_volume = volume; g_settings.audio_fast_forward_volume = fast_forward_volume; if (m_audio_stream) m_audio_stream->SetOutputVolume(GetAudioOutputVolume()); } void QtHostInterface::setAudioOutputMuted(bool muted) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "setAudioOutputMuted", Qt::QueuedConnection, Q_ARG(bool, muted)); return; } g_settings.audio_output_muted = muted; if (m_audio_stream) m_audio_stream->SetOutputVolume(GetAudioOutputVolume()); } void QtHostInterface::startDumpingAudio() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "startDumpingAudio", Qt::QueuedConnection); return; } StartDumpingAudio(); } void QtHostInterface::stopDumpingAudio() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "stopDumpingAudio", Qt::QueuedConnection); return; } StopDumpingAudio(); } void QtHostInterface::singleStepCPU() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "singleStepCPU", Qt::BlockingQueuedConnection); return; } if (!System::IsValid()) return; System::SingleStepCPU(); renderDisplay(); } void QtHostInterface::dumpRAM(const QString& filename) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "dumpRAM", Qt::QueuedConnection, Q_ARG(const QString&, filename)); return; } const std::string filename_str = filename.toStdString(); if (System::DumpRAM(filename_str.c_str())) ReportFormattedMessage("RAM dumped to '%s'", filename_str.c_str()); else ReportFormattedMessage("Failed to dump RAM to '%s'", filename_str.c_str()); } void QtHostInterface::dumpVRAM(const QString& filename) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "dumpVRAM", Qt::QueuedConnection, Q_ARG(const QString&, filename)); return; } const std::string filename_str = filename.toStdString(); if (System::DumpVRAM(filename_str.c_str())) ReportFormattedMessage("VRAM dumped to '%s'", filename_str.c_str()); else ReportFormattedMessage("Failed to dump VRAM to '%s'", filename_str.c_str()); } void QtHostInterface::dumpSPURAM(const QString& filename) { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "dumpSPURAM", Qt::QueuedConnection, Q_ARG(const QString&, filename)); return; } const std::string filename_str = filename.toStdString(); if (System::DumpSPURAM(filename_str.c_str())) ReportFormattedMessage("SPU RAM dumped to '%s'", filename_str.c_str()); else ReportFormattedMessage("Failed to dump SPU RAM to '%s'", filename_str.c_str()); } void QtHostInterface::saveScreenshot() { if (!isOnWorkerThread()) { QMetaObject::invokeMethod(this, "saveScreenshot", Qt::QueuedConnection); return; } SaveScreenshot(nullptr, true, true); } void QtHostInterface::OnAchievementsRefreshed() { #ifdef WITH_CHEEVOS QString game_info; if (Cheevos::HasActiveGame()) { game_info = tr("Game ID: %1\n" "Game Title: %2\n" "Game Developer: %3\n" "Game Publisher: %4\n" "Achievements: %5 (%6)\n\n") .arg(Cheevos::GetGameID()) .arg(QString::fromStdString(Cheevos::GetGameTitle())) .arg(QString::fromStdString(Cheevos::GetGameDeveloper())) .arg(QString::fromStdString(Cheevos::GetGamePublisher())) .arg(Cheevos::GetAchievementCount()) .arg(tr("%n points", "", Cheevos::GetMaximumPointsForGame())); const std::string& rich_presence_string = Cheevos::GetRichPresenceString(); if (!rich_presence_string.empty()) game_info.append(QString::fromStdString(rich_presence_string)); else game_info.append(tr("Rich presence inactive or unsupported.")); } else { game_info = tr("Game not loaded or no RetroAchievements available."); } emit achievementsLoaded(Cheevos::GetGameID(), game_info, Cheevos::GetAchievementCount(), Cheevos::GetMaximumPointsForGame()); #endif } void QtHostInterface::OnDisplayInvalidated() { renderDisplay(); } void QtHostInterface::doBackgroundControllerPoll() { PollAndUpdate(); } void QtHostInterface::createBackgroundControllerPollTimer() { DebugAssert(!m_background_controller_polling_timer); m_background_controller_polling_timer = new QTimer(this); m_background_controller_polling_timer->setSingleShot(false); m_background_controller_polling_timer->setTimerType(Qt::CoarseTimer); connect(m_background_controller_polling_timer, &QTimer::timeout, this, &QtHostInterface::doBackgroundControllerPoll); } void QtHostInterface::destroyBackgroundControllerPollTimer() { delete m_background_controller_polling_timer; m_background_controller_polling_timer = nullptr; } void QtHostInterface::startBackgroundControllerPollTimer() { if (m_background_controller_polling_timer->isActive() || !m_controller_interface) return; m_background_controller_polling_timer->start(BACKGROUND_CONTROLLER_POLLING_INTERVAL); } void QtHostInterface::stopBackgroundControllerPollTimer() { if (!m_background_controller_polling_timer->isActive() || !m_controller_interface) return; m_background_controller_polling_timer->stop(); } void QtHostInterface::createThread() { m_original_thread = QThread::currentThread(); m_worker_thread = new Thread(this); m_worker_thread->start(); moveToThread(m_worker_thread); } void QtHostInterface::stopThread() { Assert(!isOnWorkerThread()); QMetaObject::invokeMethod(this, "doStopThread", Qt::QueuedConnection); m_worker_thread->wait(); } void QtHostInterface::doStopThread() { m_shutdown_flag.store(true); m_worker_thread_event_loop->quit(); } void QtHostInterface::threadEntryPoint() { m_worker_thread_event_loop = new QEventLoop(); // set up controller interface and immediate poll to pick up the controller attached events m_worker_thread->setInitResult(initializeOnThread()); // TODO: Event which flags the thread as ready while (!m_shutdown_flag.load()) { if (System::IsRunning()) { if (m_display_all_frames) System::RunFrame(); else System::RunFrames(); UpdateControllerMetaState(); if (m_frame_step_request) { m_frame_step_request = false; PauseSystem(true); } renderDisplay(); System::UpdatePerformanceCounters(); if (m_throttler_enabled) System::Throttle(); } else { // we want to keep rendering the UI when paused and fullscreen UI is enabled if (!m_fullscreen_ui_enabled || !System::IsValid()) { // wait until we have a system before running m_worker_thread_event_loop->exec(); continue; } renderDisplay(); } m_worker_thread_event_loop->processEvents(QEventLoop::AllEvents); PollAndUpdate(); } shutdownOnThread(); delete m_worker_thread_event_loop; m_worker_thread_event_loop = nullptr; if (m_settings_save_timer) { m_settings_save_timer.reset(); doSaveSettings(); } // move back to UI thread moveToThread(m_original_thread); } void QtHostInterface::renderDisplay() { ImGui::NewFrame(); DrawImGuiWindows(); m_display->Render(); ImGui::EndFrame(); } void QtHostInterface::wakeThread() { if (isOnWorkerThread()) m_worker_thread_event_loop->quit(); else QMetaObject::invokeMethod(m_worker_thread_event_loop, "quit", Qt::QueuedConnection); } static std::string GetFontPath(const char* name) { #ifdef _WIN32 PWSTR folder_path; if (FAILED(SHGetKnownFolderPath(FOLDERID_Fonts, 0, nullptr, &folder_path))) return StringUtil::StdStringFromFormat("C:\\Windows\\Fonts\\%s", name); std::string font_path(StringUtil::WideStringToUTF8String(folder_path)); CoTaskMemFree(folder_path); font_path += "\\"; font_path += name; return font_path; #else return name; #endif } void QtHostInterface::setImGuiFont() { std::string language(GetStringSettingValue("Main", "Language", "")); std::string path; const ImWchar* range = nullptr; #ifdef _WIN32 if (language == "ja") { path = GetFontPath("msgothic.ttc"); range = ImGui::GetIO().Fonts->GetGlyphRangesJapanese(); } else if (language == "zh-cn") { path = GetFontPath("msyh.ttc"); range = ImGui::GetIO().Fonts->GetGlyphRangesChineseSimplifiedCommon(); } #endif if (!path.empty()) ImGuiFullscreen::SetFontFilename(std::move(path)); if (range) ImGuiFullscreen::SetFontGlyphRanges(range); } void QtHostInterface::setImGuiKeyMap() { ImGuiIO& io = ImGui::GetIO(); io.KeyMap[ImGuiKey_Tab] = Qt::Key_Tab & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_LeftArrow] = Qt::Key_Left & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_RightArrow] = Qt::Key_Right & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_UpArrow] = Qt::Key_Up & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_DownArrow] = Qt::Key_Down & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_PageUp] = Qt::Key_PageUp & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_PageDown] = Qt::Key_PageDown & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_Home] = Qt::Key_Home & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_End] = Qt::Key_End & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_Insert] = Qt::Key_Insert & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_Delete] = Qt::Key_Delete & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_Backspace] = Qt::Key_Backspace & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_Space] = Qt::Key_Space & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_Enter] = Qt::Key_Enter & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_Escape] = Qt::Key_Escape & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_A] = Qt::Key_A & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_C] = Qt::Key_C & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_V] = Qt::Key_V & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_X] = Qt::Key_X & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_Y] = Qt::Key_Y & IMGUI_KEY_MASK; io.KeyMap[ImGuiKey_Z] = Qt::Key_Z & IMGUI_KEY_MASK; } TinyString QtHostInterface::TranslateString(const char* context, const char* str, const char* disambiguation /*= nullptr*/, int n /*= -1*/) const { const QByteArray bytes(qApp->translate(context, str, disambiguation, n).toUtf8()); return TinyString(bytes.constData(), bytes.size()); } std::string QtHostInterface::TranslateStdString(const char* context, const char* str, const char* disambiguation /*= nullptr*/, int n /*= -1*/) const { return qApp->translate(context, str, disambiguation, n).toStdString(); } QtHostInterface::Thread::Thread(QtHostInterface* parent) : QThread(parent), m_parent(parent) {} QtHostInterface::Thread::~Thread() = default; void QtHostInterface::Thread::run() { m_parent->threadEntryPoint(); } void QtHostInterface::Thread::setInitResult(bool result) { m_init_result.store(result); m_init_event.Signal(); } bool QtHostInterface::Thread::waitForInit() { while (!m_init_event.TryWait(100)) QCoreApplication::processEvents(QEventLoop::ExcludeUserInputEvents); return m_init_result.load(); }