// Copyright 2008 Dolphin Emulator Project // Licensed under GPLv2+ // Refer to the license.txt file included. #include "DolphinWX/Frame.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #if defined(__unix__) || defined(__unix) || defined(__APPLE__) #include #endif #ifdef _WIN32 #include #endif #include "AudioCommon/AudioCommon.h" #include "Common/CommonTypes.h" #include "Common/FileUtil.h" #include "Common/Flag.h" #include "Common/Logging/ConsoleListener.h" #include "Common/StringUtil.h" #include "Common/Thread.h" #include "Common/Version.h" #include "Core/Config/GraphicsSettings.h" #include "Core/ConfigManager.h" #include "Core/Core.h" #include "Core/HW/DVD/DVDInterface.h" #include "Core/HW/GCKeyboard.h" #include "Core/HW/GCPad.h" #include "Core/HW/Wiimote.h" #include "Core/HW/WiimoteEmu/WiimoteEmu.h" #include "Core/HotkeyManager.h" #include "Core/IOS/IOS.h" #include "Core/IOS/USB/Bluetooth/BTBase.h" #include "Core/Movie.h" #include "Core/State.h" #include "DolphinWX/Config/ConfigMain.h" #include "DolphinWX/Debugger/BreakpointDlg.h" #include "DolphinWX/Debugger/CodeWindow.h" #include "DolphinWX/Debugger/MemoryCheckDlg.h" #include "DolphinWX/GameListCtrl.h" #include "DolphinWX/Globals.h" #include "DolphinWX/LogWindow.h" #include "DolphinWX/Main.h" #include "DolphinWX/TASInputDlg.h" #include "DolphinWX/WxUtils.h" #include "InputCommon/ControllerInterface/ControllerInterface.h" #include "InputCommon/GCPadStatus.h" #include "VideoCommon/OnScreenDisplay.h" #include "VideoCommon/RenderBase.h" #include "VideoCommon/VertexShaderManager.h" #include "VideoCommon/VideoConfig.h" #if defined(HAVE_X11) && HAVE_X11 #include #include // X11Utils nastiness that's only used here namespace X11Utils { Window XWindowFromHandle(void* Handle) { return GDK_WINDOW_XID(gtk_widget_get_window(GTK_WIDGET(Handle))); } Display* XDisplayFromHandle(void* Handle) { return GDK_WINDOW_XDISPLAY(gtk_widget_get_window(GTK_WIDGET(Handle))); } } #endif CRenderFrame::CRenderFrame(wxFrame* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style) : wxFrame(parent, id, title, pos, size, style) { // Give it an icon SetIcons(WxUtils::GetDolphinIconBundle()); DragAcceptFiles(true); Bind(wxEVT_DROP_FILES, &CRenderFrame::OnDropFiles, this); } void CRenderFrame::OnDropFiles(wxDropFilesEvent& event) { if (event.GetNumberOfFiles() != 1) return; if (File::IsDirectory(WxStrToStr(event.GetFiles()[0]))) return; wxFileName file = event.GetFiles()[0]; const std::string filepath = WxStrToStr(file.GetFullPath()); if (file.GetExt() == "dtm") { if (Core::IsRunning()) return; if (!Movie::IsReadOnly()) { // let's make the read-only flag consistent at the start of a movie. Movie::SetReadOnly(true); main_frame->GetMenuBar()->FindItem(IDM_RECORD_READ_ONLY)->Check(true); } std::optional savestate_path; if (Movie::PlayInput(filepath, &savestate_path)) main_frame->BootGame("", savestate_path); } else if (!Core::IsRunning()) { main_frame->BootGame(filepath); } else if (IsValidSavestateDropped(filepath) && Core::IsRunning()) { State::LoadAs(filepath); } else { Core::RunAsCPUThread([&filepath] { DVDInterface::ChangeDisc(filepath); }); } } bool CRenderFrame::IsValidSavestateDropped(const std::string& filepath) { const int game_id_length = 6; std::ifstream file; File::OpenFStream(file, filepath, std::ios::in | std::ios::binary); if (!file) return false; std::string internal_game_id(game_id_length, ' '); file.read(&internal_game_id[0], game_id_length); return internal_game_id == SConfig::GetInstance().GetGameID(); } #ifdef _WIN32 WXLRESULT CRenderFrame::MSWWindowProc(WXUINT nMsg, WXWPARAM wParam, WXLPARAM lParam) { switch (nMsg) { case WM_SYSCOMMAND: switch (wParam) { case SC_SCREENSAVE: case SC_MONITORPOWER: if (Core::GetState() == Core::State::Running && SConfig::GetInstance().bDisableScreenSaver) break; default: return wxFrame::MSWWindowProc(nMsg, wParam, lParam); } break; case WM_USER: switch (wParam) { case WM_USER_STOP: main_frame->DoStop(); break; case WM_USER_SETCURSOR: if (SConfig::GetInstance().bHideCursor && main_frame->RendererHasFocus() && Core::GetState() == Core::State::Running) SetCursor(wxCURSOR_BLANK); else SetCursor(wxNullCursor); break; } break; case WM_CLOSE: // Let Core finish initializing before accepting any WM_CLOSE messages if (!Core::IsRunning()) break; // Use default action otherwise default: // By default let wxWidgets do what it normally does with this event return wxFrame::MSWWindowProc(nMsg, wParam, lParam); } return 0; } #endif bool CRenderFrame::ShowFullScreen(bool show, long style) { #if defined WIN32 if (show && !g_Config.bBorderlessFullscreen) { // OpenGL requires the pop-up style to activate exclusive mode. SetWindowStyle((GetWindowStyle() & ~wxDEFAULT_FRAME_STYLE) | wxPOPUP_WINDOW); } #endif bool result = wxTopLevelWindow::ShowFullScreen(show, style); #if defined WIN32 if (!show) { // Restore the default style. SetWindowStyle((GetWindowStyle() & ~wxPOPUP_WINDOW) | wxDEFAULT_FRAME_STYLE); } #endif return result; } wxDEFINE_EVENT(wxEVT_HOST_COMMAND, wxCommandEvent); wxDEFINE_EVENT(DOLPHIN_EVT_LOCAL_INI_CHANGED, wxCommandEvent); wxDEFINE_EVENT(DOLPHIN_EVT_RELOAD_THEME_BITMAPS, wxCommandEvent); wxDEFINE_EVENT(DOLPHIN_EVT_UPDATE_LOAD_WII_MENU_ITEM, wxCommandEvent); wxDEFINE_EVENT(DOLPHIN_EVT_BOOT_SOFTWARE, wxCommandEvent); wxDEFINE_EVENT(DOLPHIN_EVT_STOP_SOFTWARE, wxCommandEvent); // Event tables BEGIN_EVENT_TABLE(CFrame, CRenderFrame) // Debugger pane context menu EVT_MENU_RANGE(IDM_FLOAT_LOG_WINDOW, IDM_FLOAT_CODE_WINDOW, CFrame::OnFloatWindow) // Game list context menu EVT_MENU(IDM_LIST_PERFORM_DISC_UPDATE, CFrame::OnPerformDiscWiiUpdate) EVT_MENU(IDM_LIST_INSTALL_WAD, CFrame::OnInstallWAD) EVT_MENU(IDM_LIST_UNINSTALL_WAD, CFrame::OnUninstallWAD) // Other EVT_ACTIVATE(CFrame::OnActive) EVT_CLOSE(CFrame::OnClose) EVT_SIZE(CFrame::OnResize) EVT_MOVE(CFrame::OnMove) EVT_HOST_COMMAND(wxID_ANY, CFrame::OnHostMessage) EVT_AUI_PANE_CLOSE(CFrame::OnPaneClose) // Post events to child panels EVT_MENU_RANGE(IDM_INTERPRETER, IDM_ADDRBOX, CFrame::PostEvent) END_EVENT_TABLE() // --------------- // Creation and close, quit functions bool CFrame::InitControllers() { if (!g_controller_interface.IsInit()) { #if defined(HAVE_X11) && HAVE_X11 void* win = reinterpret_cast(X11Utils::XWindowFromHandle(GetHandle())); #else void* win = reinterpret_cast(GetHandle()); #endif g_controller_interface.Initialize(win); Pad::Initialize(); Keyboard::Initialize(); Wiimote::Initialize(Wiimote::InitializeMode::DO_NOT_WAIT_FOR_WIIMOTES); HotkeyManagerEmu::Initialize(); return true; } return false; } static Common::Flag s_shutdown_signal_received; #ifdef _WIN32 static BOOL WINAPI s_ctrl_handler(DWORD fdwCtrlType) { switch (fdwCtrlType) { case CTRL_C_EVENT: case CTRL_BREAK_EVENT: case CTRL_CLOSE_EVENT: case CTRL_LOGOFF_EVENT: case CTRL_SHUTDOWN_EVENT: SetConsoleCtrlHandler(s_ctrl_handler, FALSE); s_shutdown_signal_received.Set(); return TRUE; default: return FALSE; } } #endif #if defined(__unix__) || defined(__unix) || defined(__APPLE__) static void SignalHandler(int) { const char message[] = "A signal was received. A second signal will force Dolphin to stop.\n"; if (write(STDERR_FILENO, message, sizeof(message)) < 0) { } s_shutdown_signal_received.Set(); } #endif CFrame::CFrame(wxFrame* parent, wxWindowID id, const wxString& title, wxRect geometry, bool use_debugger, bool batch_mode, bool show_log_window, long style) : CRenderFrame(parent, id, title, wxDefaultPosition, wxSize(800, 600), style), m_use_debugger(use_debugger), m_batch_mode(batch_mode) { BindEvents(); if (show_log_window) SConfig::GetInstance().m_InterfaceLogWindow = true; // Debugger class if (m_use_debugger) { m_code_window = new CCodeWindow(this, IDM_CODE_WINDOW); LoadIniPerspectives(); m_code_window->Load(); } wxFrame::CreateToolBar(wxTB_DEFAULT_STYLE | wxTB_TEXT | wxTB_FLAT)->Realize(); // Give it a status bar SetStatusBar( CreateStatusBar(2, wxSTB_SIZEGRIP | wxSTB_ELLIPSIZE_END | wxSTB_SHOW_TIPS, ID_STATUSBAR)); if (!SConfig::GetInstance().m_InterfaceStatusbar) GetStatusBar()->Hide(); // Give it a menu bar SetMenuBar(CreateMenuBar()); // Create a menubar to service requests while the real menubar is hidden from the screen m_menubar_shadow = CreateMenuBar(); // --------------- // Main panel // This panel is the parent for rendering and it holds the gamelistctrl m_panel = new wxPanel(this, IDM_MPANEL, wxDefaultPosition, wxDefaultSize, 0); m_game_list_ctrl = new GameListCtrl(m_batch_mode, m_panel, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxLC_REPORT | wxSUNKEN_BORDER | wxLC_ALIGN_LEFT); m_game_list_ctrl->Bind(wxEVT_LIST_ITEM_ACTIVATED, &CFrame::OnGameListCtrlItemActivated, this); wxBoxSizer* sizerPanel = new wxBoxSizer(wxHORIZONTAL); sizerPanel->Add(m_game_list_ctrl, 1, wxEXPAND | wxALL); m_panel->SetSizer(sizerPanel); // --------------- // Manager m_mgr = new wxAuiManager(this, wxAUI_MGR_DEFAULT | wxAUI_MGR_LIVE_RESIZE); m_mgr->AddPane(m_panel, wxAuiPaneInfo() .Name("Pane 0") .Caption("Pane 0") .PaneBorder(false) .CaptionVisible(false) .Layer(0) .Center() .Show()); if (!m_code_window) m_mgr->AddPane(CreateEmptyNotebook(), wxAuiPaneInfo() .Name("Pane 1") .Caption(_("Logging")) .CaptionVisible(true) .Layer(0) .FloatingSize(wxSize(600, 350)) .CloseButton(true) .Hide()); m_aui_fullscreen_perspective = m_mgr->SavePerspective(); if (!SConfig::GetInstance().m_InterfaceToolbar) DoToggleToolbar(false); m_log_window = new CLogWindow(this, IDM_LOG_WINDOW); m_log_window->Hide(); m_log_window->Disable(); InitializeTASDialogs(); InitializeCoreCallbacks(); // Setup perspectives if (m_code_window) { // Load perspective DoLoadPerspective(); } else { if (SConfig::GetInstance().m_InterfaceLogWindow) ToggleLogWindow(true); if (SConfig::GetInstance().m_InterfaceLogConfigWindow) ToggleLogConfigWindow(true); } // Setup the window size. // This has to be done here instead of in Main because the Show() happens here. SetMinSize(FromDIP(wxSize(400, 300))); WxUtils::SetWindowSizeAndFitToScreen(this, geometry.GetPosition(), geometry.GetSize(), FromDIP(wxSize(800, 600))); // Start debugging maximized (Must be after the window has been positioned) if (m_use_debugger) Maximize(true); // Commit m_mgr->Update(); // The window must be shown for m_xrr_config to be created (wxGTK will not allocate X11 // resources until the window is shown for the first time). Show(); #ifdef _WIN32 SetToolTip(""); GetToolTip()->SetAutoPop(25000); #endif #if defined(HAVE_XRANDR) && HAVE_XRANDR m_xrr_config = new X11Utils::XRRConfiguration(X11Utils::XDisplayFromHandle(GetHandle()), X11Utils::XWindowFromHandle(GetHandle())); #endif // Connect event handlers m_mgr->Bind(wxEVT_AUI_RENDER, &CFrame::OnManagerResize, this); // Update controls UpdateGUI(); // check if game is running InitControllers(); m_poll_hotkey_timer.SetOwner(this); Bind(wxEVT_TIMER, &CFrame::PollHotkeys, this, m_poll_hotkey_timer.GetId()); m_poll_hotkey_timer.Start(1000 / 60, wxTIMER_CONTINUOUS); m_cursor_timer.SetOwner(this); Bind(wxEVT_TIMER, &CFrame::HandleCursorTimer, this, m_cursor_timer.GetId()); m_cursor_timer.StartOnce(MOUSE_HIDE_DELAY); // Shut down cleanly on SIGINT, SIGTERM (Unix) and on various signals on Windows m_handle_signal_timer.SetOwner(this); Bind(wxEVT_TIMER, &CFrame::HandleSignal, this, m_handle_signal_timer.GetId()); m_handle_signal_timer.Start(100, wxTIMER_CONTINUOUS); #if defined(__unix__) || defined(__unix) || defined(__APPLE__) struct sigaction sa; sa.sa_handler = SignalHandler; sigemptyset(&sa.sa_mask); sa.sa_flags = SA_RESETHAND; sigaction(SIGINT, &sa, nullptr); sigaction(SIGTERM, &sa, nullptr); #endif #ifdef _WIN32 SetConsoleCtrlHandler(s_ctrl_handler, TRUE); #endif } // Destructor CFrame::~CFrame() { Wiimote::Shutdown(); Keyboard::Shutdown(); Pad::Shutdown(); HotkeyManagerEmu::Shutdown(); g_controller_interface.Shutdown(); #if defined(HAVE_XRANDR) && HAVE_XRANDR delete m_xrr_config; #endif ClosePages(); delete m_mgr; // This object is owned by us, not wxw m_menubar_shadow->Destroy(); m_menubar_shadow = nullptr; } void CFrame::BindEvents() { BindMenuBarEvents(); Bind(DOLPHIN_EVT_RELOAD_THEME_BITMAPS, &CFrame::OnReloadThemeBitmaps, this); Bind(DOLPHIN_EVT_REFRESH_GAMELIST, &CFrame::OnRefreshGameList, this); Bind(DOLPHIN_EVT_RESCAN_GAMELIST, &CFrame::OnRescanGameList, this); Bind(DOLPHIN_EVT_BOOT_SOFTWARE, &CFrame::OnPlay, this); Bind(DOLPHIN_EVT_STOP_SOFTWARE, &CFrame::OnStop, this); } void CFrame::InitializeTASDialogs() { std::generate(m_tas_input_dialogs.begin(), m_tas_input_dialogs.end(), [this] { return new TASInputDlg{this}; }); Movie::SetGCInputManip([this](GCPadStatus* pad_status, int controller_id) { m_tas_input_dialogs[controller_id]->GetValues(pad_status); }); Movie::SetWiiInputManip([this](u8* data, WiimoteEmu::ReportFeatures rptf, int controller_id, int ext, wiimote_key key) { m_tas_input_dialogs[controller_id + 4]->GetValues(data, rptf, ext, key); }); } void CFrame::InitializeCoreCallbacks() { // Warning: this gets called from the CPU thread, so we should // only queue things to do on the proper thread State::SetOnAfterLoadCallback([this] { AddPendingEvent(wxCommandEvent{wxEVT_HOST_COMMAND, IDM_UPDATE_GUI}); }); // Warning: this gets called from the EmuThread Core::SetOnStateChangedCallback([this](Core::State state) { if (state == Core::State::Uninitialized) AddPendingEvent(wxCommandEvent{wxEVT_HOST_COMMAND, IDM_STOPPED}); }); } bool CFrame::RendererIsFullscreen() { bool fullscreen = false; if (Core::GetState() == Core::State::Running || Core::GetState() == Core::State::Paused) { fullscreen = m_render_frame->IsFullScreen(); } return fullscreen; } void CFrame::OnQuit(wxCommandEvent& WXUNUSED(event)) { Close(true); } // -------- // Events void CFrame::OnActive(wxActivateEvent& event) { m_renderer_has_focus = (event.GetActive() && event.GetEventObject() == m_render_frame); if (Core::GetState() == Core::State::Running || Core::GetState() == Core::State::Paused) { if (m_renderer_has_focus) { if (SConfig::GetInstance().bRenderToMain) m_render_parent->SetFocus(); else if (RendererIsFullscreen() && g_ActiveConfig.ExclusiveFullscreenEnabled()) DoExclusiveFullscreen(true); // Regain exclusive mode if (SConfig::GetInstance().m_PauseOnFocusLost && Core::GetState() == Core::State::Paused) DoPause(); if (SConfig::GetInstance().bHideCursor && Core::GetState() == Core::State::Running) m_render_parent->SetCursor(wxCURSOR_BLANK); } else { if (SConfig::GetInstance().m_PauseOnFocusLost && Core::GetState() == Core::State::Running) DoPause(); if (SConfig::GetInstance().bHideCursor) m_render_parent->SetCursor(wxNullCursor); } } event.Skip(); } void CFrame::OnClose(wxCloseEvent& event) { // Before closing the window we need to shut down the emulation core. // We'll try to close this window again once that is done. if (Core::GetState() != Core::State::Uninitialized) { DoStop(); if (event.CanVeto()) { event.Veto(); } // Tell OnStopped to resubmit the Close event m_is_closing = true; return; } // Stop Dolphin from saving the minimized Xpos and Ypos if (main_frame->IsIconized()) main_frame->Iconize(false); // Don't forget the skip or the window won't be destroyed event.Skip(); // Save GUI settings if (m_code_window) { SaveIniPerspectives(); } else { m_log_window->SaveSettings(); } if (m_log_window) m_log_window->RemoveAllListeners(); // Uninit m_mgr->UnInit(); } // Post events // Warning: This may cause an endless loop if the event is propagated back to its parent void CFrame::PostEvent(wxCommandEvent& event) { if (m_code_window && event.GetId() >= IDM_INTERPRETER && event.GetId() <= IDM_ADDRBOX) { event.StopPropagation(); m_code_window->GetEventHandler()->AddPendingEvent(event); } else { event.Skip(); } } void CFrame::OnMove(wxMoveEvent& event) { event.Skip(); if (!IsMaximized() && !(SConfig::GetInstance().bRenderToMain && RendererIsFullscreen())) { SConfig::GetInstance().iPosX = GetPosition().x; SConfig::GetInstance().iPosY = GetPosition().y; } } void CFrame::OnResize(wxSizeEvent& event) { event.Skip(); if (!IsMaximized() && !IsIconized() && !(SConfig::GetInstance().bRenderToMain && RendererIsFullscreen()) && !(Core::GetState() != Core::State::Uninitialized && SConfig::GetInstance().bRenderToMain && SConfig::GetInstance().bRenderWindowAutoSize)) { SConfig::GetInstance().iWidth = GetSize().GetWidth(); SConfig::GetInstance().iHeight = GetSize().GetHeight(); } // Make sure the logger pane is a sane size if (!m_code_window && m_log_window && m_mgr->GetPane("Pane 1").IsShown() && !m_mgr->GetPane("Pane 1").IsFloating() && (m_log_window->x > GetClientRect().GetWidth() || m_log_window->y > GetClientRect().GetHeight())) { ShowResizePane(); } } // Host messages #ifdef _WIN32 WXLRESULT CFrame::MSWWindowProc(WXUINT nMsg, WXWPARAM wParam, WXLPARAM lParam) { if (WM_SYSCOMMAND == nMsg && (SC_SCREENSAVE == wParam || SC_MONITORPOWER == wParam)) { return 0; } else if (nMsg == WM_QUERYENDSESSION) { // Indicate that the application will be able to close return 1; } else if (nMsg == WM_ENDSESSION) { // Actually trigger the close now Close(true); return 0; } else { return wxFrame::MSWWindowProc(nMsg, wParam, lParam); } } #endif void CFrame::InhibitScreensaver() { // Inhibit the screensaver. Depending on the operating system this may also // disable low-power states and/or screen dimming. #if defined(HAVE_X11) && HAVE_X11 if (SConfig::GetInstance().bDisableScreenSaver) { X11Utils::InhibitScreensaver(X11Utils::XDisplayFromHandle(GetHandle()), X11Utils::XWindowFromHandle(GetHandle()), true); } #endif #ifdef _WIN32 // Prevents Windows from sleeping, turning off the display, or idling EXECUTION_STATE should_screen_save = SConfig::GetInstance().bDisableScreenSaver ? ES_DISPLAY_REQUIRED : 0; SetThreadExecutionState(ES_CONTINUOUS | should_screen_save | ES_SYSTEM_REQUIRED); #endif #ifdef __APPLE__ if (SConfig::GetInstance().bDisableScreenSaver) { CFStringRef reason_for_activity = CFSTR("Emulation Running"); if (IOPMAssertionCreateWithName(kIOPMAssertionTypePreventUserIdleDisplaySleep, kIOPMAssertionLevelOn, reason_for_activity, &m_power_assertion) != kIOReturnSuccess) { m_power_assertion = kIOPMNullAssertionID; } } #endif } void CFrame::UninhibitScreensaver() { #if defined(HAVE_X11) && HAVE_X11 if (SConfig::GetInstance().bDisableScreenSaver) { X11Utils::InhibitScreensaver(X11Utils::XDisplayFromHandle(GetHandle()), X11Utils::XWindowFromHandle(GetHandle()), false); } #endif #ifdef _WIN32 // Allow windows to resume normal idling behavior SetThreadExecutionState(ES_CONTINUOUS); #endif #ifdef __APPLE__ if (m_power_assertion != kIOPMNullAssertionID) { IOPMAssertionRelease(m_power_assertion); m_power_assertion = kIOPMNullAssertionID; } #endif } void CFrame::UpdateTitle(const wxString& str) { const wxString revision_string = StrToWxStr(Common::scm_rev_str); if (SConfig::GetInstance().bRenderToMain && SConfig::GetInstance().m_InterfaceStatusbar) { GetStatusBar()->SetStatusText(str, 0); m_render_frame->SetTitle(revision_string); } else { m_render_frame->SetTitle(revision_string + StrToWxStr(" | ") + str); } } void CFrame::OnHostMessage(wxCommandEvent& event) { switch (event.GetId()) { case IDM_UPDATE_DISASM_DIALOG: // For breakpoints causing pausing if (!m_code_window || Core::GetState() != Core::State::Paused) return; // fallthrough case IDM_UPDATE_GUI: UpdateGUI(); break; case IDM_UPDATE_STATUS_BAR: if (GetStatusBar() != nullptr) GetStatusBar()->SetStatusText(event.GetString(), event.GetInt()); break; case IDM_UPDATE_TITLE: UpdateTitle(event.GetString()); break; case IDM_WINDOW_SIZE_REQUEST: { std::pair* win_size = (std::pair*)(event.GetClientData()); OnRenderWindowSizeRequest(win_size->first, win_size->second); delete win_size; } break; case WM_USER_CREATE: if (SConfig::GetInstance().bHideCursor) m_render_parent->SetCursor(wxCURSOR_BLANK); break; case IDM_PANIC: { wxString caption = event.GetString().BeforeFirst(':'); wxString text = event.GetString().AfterFirst(':'); m_panic_result = (wxYES == wxMessageBox(text, caption, wxSTAY_ON_TOP | (event.GetInt() ? wxYES_NO : wxOK), wxWindow::FindFocus())); m_panic_event.Set(); } break; case WM_USER_STOP: DoStop(); break; case IDM_STOPPED: OnStopped(); break; case IDM_UPDATE_PROGRESS_DIALOG: { int current = event.GetInt(); int total = static_cast(event.GetExtraLong()); if (total < 0 || current >= total) { if (m_progress_dialog) { delete m_progress_dialog; m_progress_dialog = nullptr; } } else if (total > 0 && current < total) { if (!m_progress_dialog) { m_progress_dialog = new wxProgressDialog( _("Operation in progress..."), event.GetString(), total, m_render_frame, wxPD_APP_MODAL | wxPD_ELAPSED_TIME | wxPD_SMOOTH | wxPD_REMAINING_TIME); m_progress_dialog->Show(); } else { if (m_progress_dialog->GetRange() != total) m_progress_dialog->SetRange(total); m_progress_dialog->Update(current, event.GetString()); } } } break; } } void CFrame::OnRenderWindowSizeRequest(int width, int height) { if (!SConfig::GetInstance().bRenderWindowAutoSize || !Core::IsRunning() || RendererIsFullscreen() || m_render_frame->IsMaximized()) return; wxSize requested_size(width, height); // Convert to window pixels, since the size is from the backend it will be in framebuffer px. requested_size *= 1.0 / m_render_frame->GetContentScaleFactor(); wxSize old_size; if (!SConfig::GetInstance().bRenderToMain) { old_size = m_render_frame->GetClientSize(); } else { // Resize for the render panel only, this implicitly retains space for everything else // (i.e. log panel, toolbar, statusbar, etc) without needing to compute for them. old_size = m_render_parent->GetSize(); } wxSize diff = requested_size - old_size; if (diff != wxSize()) m_render_frame->SetSize(m_render_frame->GetSize() + diff); } bool CFrame::RendererHasFocus() { if (m_render_parent == nullptr) return false; return m_renderer_has_focus; } void CFrame::OnGameListCtrlItemActivated(wxListEvent& WXUNUSED(event)) { // Show all platforms and regions if... // 1. All platforms are set to hide // 2. All Regions are set to hide // Otherwise call BootGame to either... // 1. Boot the selected iso // 2. Boot the default or last loaded iso. // 3. Call BrowseForDirectory if the gamelist is empty if (!m_game_list_ctrl->GetISO(0) && GameListCtrl::IsHidingItems()) { SConfig::GetInstance().m_ListGC = SConfig::GetInstance().m_ListWii = SConfig::GetInstance().m_ListWad = SConfig::GetInstance().m_ListElfDol = SConfig::GetInstance().m_ListJap = SConfig::GetInstance().m_ListUsa = SConfig::GetInstance().m_ListPal = SConfig::GetInstance().m_ListAustralia = SConfig::GetInstance().m_ListFrance = SConfig::GetInstance().m_ListGermany = SConfig::GetInstance().m_ListItaly = SConfig::GetInstance().m_ListKorea = SConfig::GetInstance().m_ListNetherlands = SConfig::GetInstance().m_ListRussia = SConfig::GetInstance().m_ListSpain = SConfig::GetInstance().m_ListTaiwan = SConfig::GetInstance().m_ListWorld = SConfig::GetInstance().m_ListUnknown = true; GetMenuBar()->FindItem(IDM_LIST_GC)->Check(true); GetMenuBar()->FindItem(IDM_LIST_WII)->Check(true); GetMenuBar()->FindItem(IDM_LIST_WAD)->Check(true); GetMenuBar()->FindItem(IDM_LIST_JAP)->Check(true); GetMenuBar()->FindItem(IDM_LIST_USA)->Check(true); GetMenuBar()->FindItem(IDM_LIST_PAL)->Check(true); GetMenuBar()->FindItem(IDM_LIST_AUSTRALIA)->Check(true); GetMenuBar()->FindItem(IDM_LIST_FRANCE)->Check(true); GetMenuBar()->FindItem(IDM_LIST_GERMANY)->Check(true); GetMenuBar()->FindItem(IDM_LIST_ITALY)->Check(true); GetMenuBar()->FindItem(IDM_LIST_KOREA)->Check(true); GetMenuBar()->FindItem(IDM_LIST_NETHERLANDS)->Check(true); GetMenuBar()->FindItem(IDM_LIST_RUSSIA)->Check(true); GetMenuBar()->FindItem(IDM_LIST_SPAIN)->Check(true); GetMenuBar()->FindItem(IDM_LIST_TAIWAN)->Check(true); GetMenuBar()->FindItem(IDM_LIST_WORLD)->Check(true); GetMenuBar()->FindItem(IDM_LIST_UNKNOWN)->Check(true); GameListRefresh(); } else if (!m_game_list_ctrl->GetISO(0)) { m_game_list_ctrl->BrowseForDirectory(); } else { // Game started by double click BootGame(""); } } static bool IsHotkey(int id, bool held = false) { return HotkeyManagerEmu::IsPressed(id, held); } static int GetMenuIDFromHotkey(unsigned int key) { switch (key) { case HK_OPEN: return wxID_OPEN; case HK_CHANGE_DISC: return IDM_CHANGE_DISC; case HK_EJECT_DISC: return IDM_EJECT_DISC; case HK_REFRESH_LIST: return wxID_REFRESH; case HK_PLAY_PAUSE: return IDM_PLAY; case HK_STOP: return IDM_STOP; case HK_RESET: return IDM_RESET; case HK_FRAME_ADVANCE: return IDM_FRAMESTEP; case HK_START_RECORDING: return IDM_RECORD; case HK_PLAY_RECORDING: return IDM_PLAY_RECORD; case HK_EXPORT_RECORDING: return IDM_RECORD_EXPORT; case HK_READ_ONLY_MODE: return IDM_RECORD_READ_ONLY; case HK_FULLSCREEN: return IDM_TOGGLE_FULLSCREEN; case HK_SCREENSHOT: return IDM_SCREENSHOT; case HK_EXIT: return wxID_EXIT; case HK_WIIMOTE1_CONNECT: return IDM_CONNECT_WIIMOTE1; case HK_WIIMOTE2_CONNECT: return IDM_CONNECT_WIIMOTE2; case HK_WIIMOTE3_CONNECT: return IDM_CONNECT_WIIMOTE3; case HK_WIIMOTE4_CONNECT: return IDM_CONNECT_WIIMOTE4; case HK_BALANCEBOARD_CONNECT: return IDM_CONNECT_BALANCEBOARD; case HK_LOAD_STATE_SLOT_1: return IDM_LOAD_SLOT_1; case HK_LOAD_STATE_SLOT_2: return IDM_LOAD_SLOT_2; case HK_LOAD_STATE_SLOT_3: return IDM_LOAD_SLOT_3; case HK_LOAD_STATE_SLOT_4: return IDM_LOAD_SLOT_4; case HK_LOAD_STATE_SLOT_5: return IDM_LOAD_SLOT_5; case HK_LOAD_STATE_SLOT_6: return IDM_LOAD_SLOT_6; case HK_LOAD_STATE_SLOT_7: return IDM_LOAD_SLOT_7; case HK_LOAD_STATE_SLOT_8: return IDM_LOAD_SLOT_8; case HK_LOAD_STATE_SLOT_9: return IDM_LOAD_SLOT_9; case HK_LOAD_STATE_SLOT_10: return IDM_LOAD_SLOT_10; case HK_SAVE_STATE_SLOT_1: return IDM_SAVE_SLOT_1; case HK_SAVE_STATE_SLOT_2: return IDM_SAVE_SLOT_2; case HK_SAVE_STATE_SLOT_3: return IDM_SAVE_SLOT_3; case HK_SAVE_STATE_SLOT_4: return IDM_SAVE_SLOT_4; case HK_SAVE_STATE_SLOT_5: return IDM_SAVE_SLOT_5; case HK_SAVE_STATE_SLOT_6: return IDM_SAVE_SLOT_6; case HK_SAVE_STATE_SLOT_7: return IDM_SAVE_SLOT_7; case HK_SAVE_STATE_SLOT_8: return IDM_SAVE_SLOT_8; case HK_SAVE_STATE_SLOT_9: return IDM_SAVE_SLOT_9; case HK_SAVE_STATE_SLOT_10: return IDM_SAVE_SLOT_10; case HK_LOAD_LAST_STATE_1: return IDM_LOAD_LAST_1; case HK_LOAD_LAST_STATE_2: return IDM_LOAD_LAST_2; case HK_LOAD_LAST_STATE_3: return IDM_LOAD_LAST_3; case HK_LOAD_LAST_STATE_4: return IDM_LOAD_LAST_4; case HK_LOAD_LAST_STATE_5: return IDM_LOAD_LAST_5; case HK_LOAD_LAST_STATE_6: return IDM_LOAD_LAST_6; case HK_LOAD_LAST_STATE_7: return IDM_LOAD_LAST_7; case HK_LOAD_LAST_STATE_8: return IDM_LOAD_LAST_8; case HK_LOAD_LAST_STATE_9: return IDM_LOAD_LAST_9; case HK_LOAD_LAST_STATE_10: return IDM_LOAD_LAST_10; case HK_SAVE_FIRST_STATE: return IDM_SAVE_FIRST_STATE; case HK_UNDO_LOAD_STATE: return IDM_UNDO_LOAD_STATE; case HK_UNDO_SAVE_STATE: return IDM_UNDO_SAVE_STATE; case HK_LOAD_STATE_FILE: return IDM_LOAD_STATE_FILE; case HK_SAVE_STATE_FILE: return IDM_SAVE_STATE_FILE; case HK_SELECT_STATE_SLOT_1: return IDM_SELECT_SLOT_1; case HK_SELECT_STATE_SLOT_2: return IDM_SELECT_SLOT_2; case HK_SELECT_STATE_SLOT_3: return IDM_SELECT_SLOT_3; case HK_SELECT_STATE_SLOT_4: return IDM_SELECT_SLOT_4; case HK_SELECT_STATE_SLOT_5: return IDM_SELECT_SLOT_5; case HK_SELECT_STATE_SLOT_6: return IDM_SELECT_SLOT_6; case HK_SELECT_STATE_SLOT_7: return IDM_SELECT_SLOT_7; case HK_SELECT_STATE_SLOT_8: return IDM_SELECT_SLOT_8; case HK_SELECT_STATE_SLOT_9: return IDM_SELECT_SLOT_9; case HK_SELECT_STATE_SLOT_10: return IDM_SELECT_SLOT_10; case HK_SAVE_STATE_SLOT_SELECTED: return IDM_SAVE_SELECTED_SLOT; case HK_LOAD_STATE_SLOT_SELECTED: return IDM_LOAD_SELECTED_SLOT; case HK_FREELOOK_DECREASE_SPEED: return IDM_FREELOOK_DECREASE_SPEED; case HK_FREELOOK_INCREASE_SPEED: return IDM_FREELOOK_INCREASE_SPEED; case HK_FREELOOK_RESET_SPEED: return IDM_FREELOOK_RESET_SPEED; case HK_FREELOOK_LEFT: return IDM_FREELOOK_LEFT; case HK_FREELOOK_RIGHT: return IDM_FREELOOK_RIGHT; case HK_FREELOOK_UP: return IDM_FREELOOK_UP; case HK_FREELOOK_DOWN: return IDM_FREELOOK_DOWN; case HK_FREELOOK_ZOOM_IN: return IDM_FREELOOK_ZOOM_IN; case HK_FREELOOK_ZOOM_OUT: return IDM_FREELOOK_ZOOM_OUT; case HK_FREELOOK_RESET: return IDM_FREELOOK_RESET; } return -1; } void CFrame::OnKeyDown(wxKeyEvent& event) { #ifdef __APPLE__ // On OS X, we claim all keyboard events while // emulation is running to avoid wxWidgets sounding // the system beep for unhandled key events when // receiving pad/Wiimote keypresses which take an // entirely different path through the HID subsystem. if (!m_renderer_has_focus) { // We do however want to pass events on when the // render window is out of focus: this allows use // of the keyboard in the rest of the UI. event.Skip(); } #else // On other platforms, we leave the key event alone // so it can be passed on to the windowing system. event.Skip(); #endif } void CFrame::OnMouse(wxMouseEvent& event) { if (!SConfig::GetInstance().bHideCursor && main_frame->RendererHasFocus() && Core::GetState() == Core::State::Running) { m_render_parent->SetCursor(wxNullCursor); m_cursor_timer.StartOnce(MOUSE_HIDE_DELAY); } // next handlers are all for FreeLook, so we don't need to check them if disabled if (!g_Config.bFreeLook) { event.Skip(); return; } // Free look variables static bool mouseLookEnabled = false; static bool mouseMoveEnabled = false; static float lastMouse[2]; if (event.MiddleDown()) { lastMouse[0] = event.GetX(); lastMouse[1] = event.GetY(); mouseMoveEnabled = true; } else if (event.RightDown()) { lastMouse[0] = event.GetX(); lastMouse[1] = event.GetY(); mouseLookEnabled = true; } else if (event.MiddleUp()) { mouseMoveEnabled = false; } else if (event.RightUp()) { mouseLookEnabled = false; } // no button, so it's a move event else if (event.GetButton() == wxMOUSE_BTN_NONE) { if (mouseLookEnabled) { VertexShaderManager::RotateView((event.GetX() - lastMouse[0]) / 200.0f, (event.GetY() - lastMouse[1]) / 200.0f); lastMouse[0] = event.GetX(); lastMouse[1] = event.GetY(); } if (mouseMoveEnabled) { VertexShaderManager::TranslateView((event.GetX() - lastMouse[0]) / 50.0f, (event.GetY() - lastMouse[1]) / 50.0f); lastMouse[0] = event.GetX(); lastMouse[1] = event.GetY(); } } event.Skip(); } void CFrame::DoFullscreen(bool enable_fullscreen) { ToggleDisplayMode(enable_fullscreen); if (SConfig::GetInstance().bRenderToMain) { m_render_frame->ShowFullScreen(enable_fullscreen, wxFULLSCREEN_ALL); if (enable_fullscreen) { // Save the current mode before going to fullscreen m_aui_current_perspective = m_mgr->SavePerspective(); m_mgr->LoadPerspective(m_aui_fullscreen_perspective, true); // Hide toolbar DoToggleToolbar(false); // Hide menubar (by having wxwidgets delete it) SetMenuBar(nullptr); // Hide the statusbar if enabled if (GetStatusBar()->IsShown()) { GetStatusBar()->Hide(); this->SendSizeEvent(); } } else { // Restore saved perspective m_mgr->LoadPerspective(m_aui_current_perspective, true); // Restore toolbar to the status it was at before going fullscreen. DoToggleToolbar(SConfig::GetInstance().m_InterfaceToolbar); // Recreate the menubar if needed. if (wxFrame::GetMenuBar() == nullptr) { SetMenuBar(CreateMenuBar()); } // Show statusbar if enabled if (SConfig::GetInstance().m_InterfaceStatusbar) { GetStatusBar()->Show(); this->SendSizeEvent(); } } } else if (g_ActiveConfig.ExclusiveFullscreenEnabled()) { if (!enable_fullscreen) DoExclusiveFullscreen(false); m_render_frame->ShowFullScreen(enable_fullscreen, wxFULLSCREEN_ALL); m_render_frame->Raise(); if (enable_fullscreen) DoExclusiveFullscreen(true); } else { m_render_frame->ShowFullScreen(enable_fullscreen, wxFULLSCREEN_ALL); m_render_frame->Raise(); } } void CFrame::DoExclusiveFullscreen(bool enable_fullscreen) { if (!g_renderer || g_renderer->IsFullscreen() == enable_fullscreen) return; Core::RunAsCPUThread([enable_fullscreen] { g_renderer->SetFullscreen(enable_fullscreen); }); } void CFrame::PollHotkeys(wxTimerEvent& event) { if (!HotkeyManagerEmu::IsEnabled()) return; if (Core::GetState() == Core::State::Uninitialized || Core::GetState() == Core::State::Paused) g_controller_interface.UpdateInput(); if (Core::GetState() != Core::State::Stopping) { HotkeyManagerEmu::GetStatus(); ParseHotkeys(); } } void CFrame::ParseHotkeys() { for (int i = 0; i < NUM_HOTKEYS; i++) { switch (i) { case HK_OPEN: case HK_CHANGE_DISC: case HK_EJECT_DISC: case HK_REFRESH_LIST: case HK_RESET: case HK_START_RECORDING: case HK_PLAY_RECORDING: case HK_EXPORT_RECORDING: case HK_READ_ONLY_MODE: case HK_LOAD_STATE_FILE: case HK_SAVE_STATE_FILE: case HK_LOAD_STATE_SLOT_SELECTED: if (IsHotkey(i)) { const int id = GetMenuIDFromHotkey(i); if (id >= 0) { wxCommandEvent evt(wxEVT_MENU, id); wxMenuItem* item = GetMenuBar()->FindItem(id); if (item && item->IsCheckable()) { item->wxMenuItemBase::Toggle(); evt.SetInt(item->IsChecked()); } GetEventHandler()->AddPendingEvent(evt); } } default: break; // do nothing } } if (!Core::IsRunningAndStarted()) { return; } // Toggle fullscreen if (IsHotkey(HK_FULLSCREEN)) DoFullscreen(!RendererIsFullscreen()); // Pause and Unpause if (IsHotkey(HK_PLAY_PAUSE)) DoPause(); // Frame advance HandleFrameSkipHotkeys(); // Stop if (IsHotkey(HK_STOP)) DoStop(); // Screenshot hotkey if (IsHotkey(HK_SCREENSHOT)) Core::SaveScreenShot(); if (IsHotkey(HK_EXIT)) wxPostEvent(this, wxCommandEvent(wxEVT_MENU, wxID_EXIT)); if (IsHotkey(HK_VOLUME_DOWN)) AudioCommon::DecreaseVolume(3); if (IsHotkey(HK_VOLUME_UP)) AudioCommon::IncreaseVolume(3); if (IsHotkey(HK_VOLUME_TOGGLE_MUTE)) AudioCommon::ToggleMuteVolume(); if (SConfig::GetInstance().m_bt_passthrough_enabled) { const auto ios = IOS::HLE::GetIOS(); auto device = ios ? ios->GetDeviceByName("/dev/usb/oh1/57e/305") : nullptr; if (device != nullptr) std::static_pointer_cast(device)->UpdateSyncButtonState( IsHotkey(HK_TRIGGER_SYNC_BUTTON, true)); } if (m_use_debugger) { if (IsHotkey(HK_STEP)) { wxCommandEvent evt(wxEVT_MENU, IDM_STEP); GetEventHandler()->AddPendingEvent(evt); } if (IsHotkey(HK_STEP_OVER)) { wxCommandEvent evt(wxEVT_MENU, IDM_STEPOVER); GetEventHandler()->AddPendingEvent(evt); } if (IsHotkey(HK_STEP_OUT)) { wxCommandEvent evt(wxEVT_MENU, IDM_STEPOUT); GetEventHandler()->AddPendingEvent(evt); } if (IsHotkey(HK_SKIP)) { wxCommandEvent evt(wxEVT_MENU, IDM_SKIP); GetEventHandler()->AddPendingEvent(evt); } if (IsHotkey(HK_SHOW_PC)) { wxCommandEvent evt(wxEVT_MENU, IDM_GOTOPC); GetEventHandler()->AddPendingEvent(evt); } if (IsHotkey(HK_SET_PC)) { wxCommandEvent evt(wxEVT_MENU, IDM_SETPC); GetEventHandler()->AddPendingEvent(evt); } if (IsHotkey(HK_BP_TOGGLE)) { wxCommandEvent evt(wxEVT_MENU, IDM_TOGGLE_BREAKPOINT); GetEventHandler()->AddPendingEvent(evt); } if (IsHotkey(HK_BP_ADD)) { BreakPointDlg bpDlg(this); if (bpDlg.ShowModal() == wxID_OK) { wxCommandEvent evt(wxEVT_HOST_COMMAND, IDM_UPDATE_BREAKPOINTS); m_code_window->GetEventHandler()->AddPendingEvent(evt); } } if (IsHotkey(HK_MBP_ADD)) { MemoryCheckDlg memDlg(this); if (memDlg.ShowModal() == wxID_OK) { wxCommandEvent evt(wxEVT_HOST_COMMAND, IDM_UPDATE_BREAKPOINTS); m_code_window->GetEventHandler()->AddPendingEvent(evt); } } } // Wiimote connect and disconnect hotkeys int WiimoteId = -1; if (IsHotkey(HK_WIIMOTE1_CONNECT)) WiimoteId = 0; if (IsHotkey(HK_WIIMOTE2_CONNECT)) WiimoteId = 1; if (IsHotkey(HK_WIIMOTE3_CONNECT)) WiimoteId = 2; if (IsHotkey(HK_WIIMOTE4_CONNECT)) WiimoteId = 3; if (IsHotkey(HK_BALANCEBOARD_CONNECT)) WiimoteId = 4; // Actually perform the Wiimote connection or disconnection if (WiimoteId >= 0 && SConfig::GetInstance().bWii) { wxCommandEvent evt; evt.SetId(IDM_CONNECT_WIIMOTE1 + WiimoteId); OnConnectWiimote(evt); } if (IsHotkey(HK_INCREASE_IR)) { OSDChoice = 1; Config::SetCurrent(Config::GFX_EFB_SCALE, Config::Get(Config::GFX_EFB_SCALE) + 1); } if (IsHotkey(HK_DECREASE_IR)) { OSDChoice = 1; if (Config::Get(Config::GFX_EFB_SCALE) > EFB_SCALE_AUTO_INTEGRAL) Config::SetCurrent(Config::GFX_EFB_SCALE, Config::Get(Config::GFX_EFB_SCALE) - 1); } if (IsHotkey(HK_TOGGLE_CROP)) { Config::SetCurrent(Config::GFX_CROP, !Config::Get(Config::GFX_CROP)); } if (IsHotkey(HK_TOGGLE_AR)) { OSDChoice = 2; // Toggle aspect ratio int aspect_ratio = Config::Get(Config::GFX_ASPECT_RATIO); aspect_ratio = (aspect_ratio + 1) & 3; Config::SetCurrent(Config::GFX_ASPECT_RATIO, aspect_ratio); } if (IsHotkey(HK_TOGGLE_EFBCOPIES)) { OSDChoice = 3; // Toggle EFB copies between EFB2RAM and EFB2Texture Config::SetCurrent(Config::GFX_HACK_SKIP_EFB_COPY_TO_RAM, !Config::Get(Config::GFX_HACK_SKIP_EFB_COPY_TO_RAM)); } if (IsHotkey(HK_TOGGLE_XFBCOPIES)) { OSDChoice = 6; // Toggle XFB copies between XFB2RAM and XFB2Texture Config::SetCurrent(Config::GFX_HACK_SKIP_XFB_COPY_TO_RAM, !Config::Get(Config::GFX_HACK_SKIP_XFB_COPY_TO_RAM)); } if (IsHotkey(HK_TOGGLE_IMMEDIATE_XFB)) { OSDChoice = 6; // Toggle immediate present of xfb Config::SetCurrent(Config::GFX_HACK_IMMEDIATE_XFB, !Config::Get(Config::GFX_HACK_IMMEDIATE_XFB)); } if (IsHotkey(HK_TOGGLE_FOG)) { OSDChoice = 4; Config::SetCurrent(Config::GFX_DISABLE_FOG, !Config::Get(Config::GFX_DISABLE_FOG)); } if (IsHotkey(HK_TOGGLE_DUMPTEXTURES)) { Config::SetCurrent(Config::GFX_DUMP_TEXTURES, !Config::Get(Config::GFX_DUMP_TEXTURES)); } if (IsHotkey(HK_TOGGLE_TEXTURES)) { Config::SetCurrent(Config::GFX_HIRES_TEXTURES, !Config::Get(Config::GFX_HIRES_TEXTURES)); } Core::SetIsThrottlerTempDisabled(IsHotkey(HK_TOGGLE_THROTTLE, true)); if (IsHotkey(HK_DECREASE_EMULATION_SPEED)) { OSDChoice = 5; if (SConfig::GetInstance().m_EmulationSpeed <= 0.0f) SConfig::GetInstance().m_EmulationSpeed = 1.0f; else if (SConfig::GetInstance().m_EmulationSpeed >= 0.2f) SConfig::GetInstance().m_EmulationSpeed -= 0.1f; else SConfig::GetInstance().m_EmulationSpeed = 0.1f; if (SConfig::GetInstance().m_EmulationSpeed >= 0.95f && SConfig::GetInstance().m_EmulationSpeed <= 1.05f) SConfig::GetInstance().m_EmulationSpeed = 1.0f; } if (IsHotkey(HK_INCREASE_EMULATION_SPEED)) { OSDChoice = 5; if (SConfig::GetInstance().m_EmulationSpeed > 0.0f) SConfig::GetInstance().m_EmulationSpeed += 0.1f; if (SConfig::GetInstance().m_EmulationSpeed >= 0.95f && SConfig::GetInstance().m_EmulationSpeed <= 1.05f) SConfig::GetInstance().m_EmulationSpeed = 1.0f; } if (IsHotkey(HK_SAVE_STATE_SLOT_SELECTED)) { State::Save(m_save_slot); } if (IsHotkey(HK_LOAD_STATE_SLOT_SELECTED)) { State::Load(m_save_slot); } if (IsHotkey(HK_TOGGLE_STEREO_SBS)) { if (g_Config.stereo_mode != StereoMode::SBS) { // Current implementation of anaglyph stereoscopy uses a // post-processing shader. Thus the shader needs to be to be // turned off when selecting other stereoscopy modes. if (g_Config.sPostProcessingShader == "dubois") { Config::SetCurrent(Config::GFX_ENHANCE_POST_SHADER, std::string("")); } Config::SetCurrent(Config::GFX_STEREO_MODE, static_cast(StereoMode::SBS)); } else { Config::SetCurrent(Config::GFX_STEREO_MODE, static_cast(StereoMode::Off)); } } if (IsHotkey(HK_TOGGLE_STEREO_TAB)) { if (g_Config.stereo_mode != StereoMode::TAB) { if (g_Config.sPostProcessingShader == "dubois") { Config::SetCurrent(Config::GFX_ENHANCE_POST_SHADER, std::string("")); } Config::SetCurrent(Config::GFX_STEREO_MODE, static_cast(StereoMode::TAB)); } else { Config::SetCurrent(Config::GFX_STEREO_MODE, static_cast(StereoMode::Off)); } } if (IsHotkey(HK_TOGGLE_STEREO_ANAGLYPH)) { if (g_Config.stereo_mode != StereoMode::Anaglyph) { // Setting the anaglyph mode also requires a specific // post-processing shader to be activated. Config::SetCurrent(Config::GFX_STEREO_MODE, static_cast(StereoMode::Anaglyph)); Config::SetCurrent(Config::GFX_ENHANCE_POST_SHADER, std::string("dubois")); } else { Config::SetCurrent(Config::GFX_STEREO_MODE, static_cast(StereoMode::Off)); Config::SetCurrent(Config::GFX_ENHANCE_POST_SHADER, std::string("")); } } if (IsHotkey(HK_TOGGLE_STEREO_3DVISION)) { if (g_Config.stereo_mode != StereoMode::Nvidia3DVision) { if (g_Config.sPostProcessingShader == "dubois") { Config::SetCurrent(Config::GFX_ENHANCE_POST_SHADER, std::string("")); } Config::SetCurrent(Config::GFX_STEREO_MODE, static_cast(StereoMode::Nvidia3DVision)); } else { Config::SetCurrent(Config::GFX_STEREO_MODE, static_cast(StereoMode::Off)); } } if (IsHotkey(HK_DECREASE_DEPTH, true)) { if (g_Config.iStereoDepth > 0) Config::SetCurrent(Config::GFX_STEREO_DEPTH, g_Config.iStereoDepth - 1); } if (IsHotkey(HK_INCREASE_DEPTH, true)) { if (g_Config.iStereoDepth < 100) Config::SetCurrent(Config::GFX_STEREO_DEPTH, g_Config.iStereoDepth + 1); } if (IsHotkey(HK_DECREASE_CONVERGENCE, true)) { int convergence = std::max(0, g_Config.iStereoConvergence - 5); Config::SetCurrent(Config::GFX_STEREO_CONVERGENCE, convergence); } if (IsHotkey(HK_INCREASE_CONVERGENCE, true)) { int convergence = std::min(500, g_Config.iStereoConvergence + 5); Config::SetCurrent(Config::GFX_STEREO_CONVERGENCE, convergence); } static float debugSpeed = 1.0f; if (IsHotkey(HK_FREELOOK_DECREASE_SPEED, true)) debugSpeed /= 1.1f; if (IsHotkey(HK_FREELOOK_INCREASE_SPEED, true)) debugSpeed *= 1.1f; if (IsHotkey(HK_FREELOOK_RESET_SPEED, true)) debugSpeed = 1.0f; if (IsHotkey(HK_FREELOOK_UP, true)) VertexShaderManager::TranslateView(0.0f, 0.0f, -debugSpeed); if (IsHotkey(HK_FREELOOK_DOWN, true)) VertexShaderManager::TranslateView(0.0f, 0.0f, debugSpeed); if (IsHotkey(HK_FREELOOK_LEFT, true)) VertexShaderManager::TranslateView(debugSpeed, 0.0f); if (IsHotkey(HK_FREELOOK_RIGHT, true)) VertexShaderManager::TranslateView(-debugSpeed, 0.0f); if (IsHotkey(HK_FREELOOK_ZOOM_IN, true)) VertexShaderManager::TranslateView(0.0f, debugSpeed); if (IsHotkey(HK_FREELOOK_ZOOM_OUT, true)) VertexShaderManager::TranslateView(0.0f, -debugSpeed); if (IsHotkey(HK_FREELOOK_RESET, true)) VertexShaderManager::ResetView(); // Savestates for (u32 i = 0; i < State::NUM_STATES; i++) { if (IsHotkey(HK_LOAD_STATE_SLOT_1 + i)) State::Load(1 + i); if (IsHotkey(HK_SAVE_STATE_SLOT_1 + i)) State::Save(1 + i); if (IsHotkey(HK_LOAD_LAST_STATE_1 + i)) State::LoadLastSaved(1 + i); if (IsHotkey(HK_SELECT_STATE_SLOT_1 + i)) { wxCommandEvent slot_event; slot_event.SetId(IDM_SELECT_SLOT_1 + i); CFrame::OnSelectSlot(slot_event); } } if (IsHotkey(HK_SAVE_FIRST_STATE)) State::SaveFirstSaved(); if (IsHotkey(HK_UNDO_LOAD_STATE)) State::UndoLoadState(); if (IsHotkey(HK_UNDO_SAVE_STATE)) State::UndoSaveState(); } void CFrame::HandleFrameSkipHotkeys() { static const int MAX_FRAME_SKIP_DELAY = 60; static int frameStepCount = 0; static const int FRAME_STEP_DELAY = 30; static int holdFrameStepDelay = 1; static int holdFrameStepDelayCount = 0; static bool holdFrameStep = false; if (IsHotkey(HK_FRAME_ADVANCE_DECREASE_SPEED)) { ++holdFrameStepDelay; if (holdFrameStepDelay > MAX_FRAME_SKIP_DELAY) holdFrameStepDelay = MAX_FRAME_SKIP_DELAY; } else if (IsHotkey(HK_FRAME_ADVANCE_INCREASE_SPEED)) { --holdFrameStepDelay; if (holdFrameStepDelay < 0) holdFrameStepDelay = 0; } else if (IsHotkey(HK_FRAME_ADVANCE_RESET_SPEED)) { holdFrameStepDelay = 1; } else if (IsHotkey(HK_FRAME_ADVANCE, true)) { if (holdFrameStepDelayCount < holdFrameStepDelay && holdFrameStep) ++holdFrameStepDelayCount; if ((frameStepCount == 0 || frameStepCount == FRAME_STEP_DELAY) && !holdFrameStep) { wxCommandEvent evt; evt.SetId(IDM_FRAMESTEP); CFrame::OnFrameStep(evt); if (holdFrameStepDelay > 0) holdFrameStep = true; } if (frameStepCount < FRAME_STEP_DELAY) { ++frameStepCount; if (holdFrameStep) holdFrameStep = false; } if (frameStepCount == FRAME_STEP_DELAY && holdFrameStep && holdFrameStepDelayCount >= holdFrameStepDelay) { holdFrameStep = false; holdFrameStepDelayCount = 0; } } else if (frameStepCount > 0) { // Reset values of frame advance to default frameStepCount = 0; holdFrameStep = false; holdFrameStepDelayCount = 0; } } void CFrame::HandleCursorTimer(wxTimerEvent& event) { if (!SConfig::GetInstance().bHideCursor && main_frame->RendererHasFocus() && Core::GetState() == Core::State::Running) m_render_parent->SetCursor(wxCURSOR_BLANK); } void CFrame::HandleSignal(wxTimerEvent& event) { if (!s_shutdown_signal_received.TestAndClear()) return; m_is_closing = true; Close(); }