mirror of
https://github.com/dolphin-emu/dolphin.git
synced 2025-01-12 09:09:12 +01:00
4288bfe0f9
Given this is actually a part of the Host interface, this should be placed with it. While we're at it, turn it into an enum class so that we don't dump its contained values into the surrounding scope. We can also make Host_Message take the enum type itself directly instead of taking a general int value. After this, it'll be trivial to divide out the rest of Common.h and remove the header from the repository entirely
528 lines
15 KiB
C++
528 lines
15 KiB
C++
// Copyright 2008 Dolphin Emulator Project
|
|
// Licensed under GPLv2+
|
|
// Refer to the license.txt file included.
|
|
|
|
#include <OptionParser.h>
|
|
#include <cstdio>
|
|
#include <cstring>
|
|
#include <optional>
|
|
#include <string>
|
|
#include <utility>
|
|
#include <wx/app.h>
|
|
#include <wx/buffer.h>
|
|
#include <wx/cmdline.h>
|
|
#include <wx/evtloop.h>
|
|
#include <wx/image.h>
|
|
#include <wx/imagpng.h>
|
|
#include <wx/intl.h>
|
|
#include <wx/language.h>
|
|
#include <wx/menu.h>
|
|
#include <wx/msgdlg.h>
|
|
#include <wx/thread.h>
|
|
#include <wx/timer.h>
|
|
#include <wx/tooltip.h>
|
|
#include <wx/utils.h>
|
|
#include <wx/window.h>
|
|
|
|
#include "Common/CPUDetect.h"
|
|
#include "Common/CommonPaths.h"
|
|
#include "Common/CommonTypes.h"
|
|
#include "Common/FileUtil.h"
|
|
#include "Common/IniFile.h"
|
|
#include "Common/Logging/LogManager.h"
|
|
#include "Common/MsgHandler.h"
|
|
#include "Common/Thread.h"
|
|
#include "Common/Version.h"
|
|
|
|
#include "Core/Analytics.h"
|
|
#include "Core/Boot/Boot.h"
|
|
#include "Core/ConfigManager.h"
|
|
#include "Core/Core.h"
|
|
#include "Core/HW/Wiimote.h"
|
|
#include "Core/Host.h"
|
|
#include "Core/Movie.h"
|
|
|
|
#include "DolphinWX/Debugger/CodeWindow.h"
|
|
#include "DolphinWX/Debugger/JitWindow.h"
|
|
#include "DolphinWX/Frame.h"
|
|
#include "DolphinWX/Globals.h"
|
|
#include "DolphinWX/Main.h"
|
|
#include "DolphinWX/NetPlay/NetWindow.h"
|
|
#include "DolphinWX/SoftwareVideoConfigDialog.h"
|
|
#include "DolphinWX/UINeedsControllerState.h"
|
|
#include "DolphinWX/VideoConfigDiag.h"
|
|
#include "DolphinWX/WxUtils.h"
|
|
|
|
#include "UICommon/CommandLineParse.h"
|
|
#include "UICommon/UICommon.h"
|
|
|
|
#include "VideoCommon/VideoBackendBase.h"
|
|
|
|
#if defined HAVE_X11 && HAVE_X11
|
|
#include <X11/Xlib.h>
|
|
#endif
|
|
|
|
// ------------
|
|
// Main window
|
|
|
|
IMPLEMENT_APP(DolphinApp)
|
|
|
|
bool wxMsgAlert(const char*, const char*, bool, MsgType);
|
|
std::string wxStringTranslator(const char*);
|
|
|
|
CFrame* main_frame = nullptr;
|
|
|
|
bool DolphinApp::Initialize(int& c, wxChar** v)
|
|
{
|
|
#if defined HAVE_X11 && HAVE_X11
|
|
XInitThreads();
|
|
#endif
|
|
return wxApp::Initialize(c, v);
|
|
}
|
|
|
|
// The 'main program' equivalent that creates the main window and return the main frame
|
|
|
|
void DolphinApp::OnInitCmdLine(wxCmdLineParser& parser)
|
|
{
|
|
parser.SetCmdLine("");
|
|
}
|
|
|
|
bool DolphinApp::OnCmdLineParsed(wxCmdLineParser& parser)
|
|
{
|
|
return true;
|
|
}
|
|
|
|
bool DolphinApp::OnInit()
|
|
{
|
|
if (!wxApp::OnInit())
|
|
return false;
|
|
|
|
Bind(wxEVT_QUERY_END_SESSION, &DolphinApp::OnEndSession, this);
|
|
Bind(wxEVT_END_SESSION, &DolphinApp::OnEndSession, this);
|
|
Bind(wxEVT_IDLE, &DolphinApp::OnIdle, this);
|
|
Bind(wxEVT_ACTIVATE_APP, &DolphinApp::OnActivate, this);
|
|
|
|
// Register message box and translation handlers
|
|
RegisterMsgAlertHandler(&wxMsgAlert);
|
|
RegisterStringTranslator(&wxStringTranslator);
|
|
|
|
#if wxUSE_ON_FATAL_EXCEPTION
|
|
wxHandleFatalExceptions(true);
|
|
#endif
|
|
|
|
#ifdef _WIN32
|
|
const bool console_attached = AttachConsole(ATTACH_PARENT_PROCESS) != FALSE;
|
|
HANDLE stdout_handle = ::GetStdHandle(STD_OUTPUT_HANDLE);
|
|
if (console_attached && stdout_handle)
|
|
{
|
|
freopen("CONOUT$", "w", stdout);
|
|
freopen("CONOUT$", "w", stderr);
|
|
}
|
|
#endif
|
|
|
|
ParseCommandLine();
|
|
|
|
#ifdef _WIN32
|
|
FreeConsole();
|
|
#endif
|
|
|
|
UICommon::SetUserDirectory(m_user_path.ToStdString());
|
|
UICommon::CreateDirectories();
|
|
InitLanguageSupport(); // The language setting is loaded from the user directory
|
|
UICommon::Init();
|
|
|
|
if (m_select_video_backend && !m_video_backend_name.empty())
|
|
SConfig::GetInstance().m_strVideoBackend = WxStrToStr(m_video_backend_name);
|
|
|
|
if (m_select_audio_emulation)
|
|
SConfig::GetInstance().bDSPHLE = (m_audio_emulation_name.Upper() == "HLE");
|
|
|
|
VideoBackendBase::ActivateBackend(SConfig::GetInstance().m_strVideoBackend);
|
|
|
|
DolphinAnalytics::Instance()->ReportDolphinStart("wx");
|
|
|
|
wxToolTip::Enable(!SConfig::GetInstance().m_DisableTooltips);
|
|
|
|
// Enable the PNG image handler for screenshots
|
|
wxImage::AddHandler(new wxPNGHandler);
|
|
|
|
// Silent PNG warnings from some homebrew banners: "iCCP: known incorrect sRGB profile"
|
|
wxImage::SetDefaultLoadFlags(wxImage::GetDefaultLoadFlags() & ~wxImage::Load_Verbose);
|
|
|
|
// We have to copy the size and position out of SConfig now because CFrame's OnMove
|
|
// handler will corrupt them during window creation (various APIs like SetMenuBar cause
|
|
// event dispatch including WM_MOVE/WM_SIZE)
|
|
wxRect window_geometry(SConfig::GetInstance().iPosX, SConfig::GetInstance().iPosY,
|
|
SConfig::GetInstance().iWidth, SConfig::GetInstance().iHeight);
|
|
main_frame = new CFrame(nullptr, wxID_ANY, StrToWxStr(Common::scm_rev_str), window_geometry,
|
|
m_use_debugger, m_batch_mode, m_use_logger);
|
|
SetTopWindow(main_frame);
|
|
|
|
AfterInit();
|
|
|
|
return true;
|
|
}
|
|
|
|
void DolphinApp::ParseCommandLine()
|
|
{
|
|
auto parser = CommandLineParse::CreateParser(CommandLineParse::ParserOptions::IncludeGUIOptions);
|
|
|
|
// Manually convert each argument to a UTF-8 std::string, because the implicit
|
|
// conversion of wxCmdLineArgsArray to char** calls ToAscii (which is undesired).
|
|
std::vector<std::string> utf8_argv;
|
|
for (int i = 1; i < argc; ++i)
|
|
utf8_argv.emplace_back(argv[i].utf8_str());
|
|
optparse::Values& options = CommandLineParse::ParseArguments(parser.get(), utf8_argv);
|
|
|
|
std::vector<std::string> args = parser->args();
|
|
|
|
if (options.is_set("exec"))
|
|
{
|
|
m_boot = BootParameters::GenerateFromFile(static_cast<const char*>(options.get("exec")));
|
|
}
|
|
else if (options.is_set("nand_title"))
|
|
{
|
|
const std::string hex_string = static_cast<const char*>(options.get("nand_title"));
|
|
if (hex_string.length() == 16)
|
|
{
|
|
const u64 title_id = std::stoull(hex_string, nullptr, 16);
|
|
m_boot = std::make_unique<BootParameters>(BootParameters::NANDTitle{title_id});
|
|
}
|
|
else
|
|
{
|
|
WxUtils::ShowErrorDialog(_("The title ID is invalid."));
|
|
}
|
|
}
|
|
else if (args.size())
|
|
{
|
|
m_boot = BootParameters::GenerateFromFile(args.front());
|
|
args.erase(args.begin());
|
|
}
|
|
|
|
m_use_debugger = options.is_set("debugger");
|
|
m_use_logger = options.is_set("logger");
|
|
m_batch_mode = options.is_set("batch");
|
|
|
|
m_confirm_stop = options.is_set("confirm");
|
|
m_confirm_setting = options.get("confirm");
|
|
|
|
m_select_video_backend = options.is_set("video_backend");
|
|
m_video_backend_name = static_cast<const char*>(options.get("video_backend"));
|
|
|
|
m_select_audio_emulation = options.is_set("audio_emulation");
|
|
m_audio_emulation_name = static_cast<const char*>(options.get("audio_emulation"));
|
|
|
|
m_play_movie = options.is_set("movie");
|
|
m_movie_file = static_cast<const char*>(options.get("movie"));
|
|
|
|
m_user_path = static_cast<const char*>(options.get("user"));
|
|
}
|
|
|
|
#ifdef __APPLE__
|
|
void DolphinApp::MacOpenFile(const wxString& fileName)
|
|
{
|
|
main_frame->StartGame(BootParameters::GenerateFromFile(fileName.ToStdString()));
|
|
}
|
|
#endif
|
|
|
|
void DolphinApp::AfterInit()
|
|
{
|
|
#if defined(USE_ANALYTICS) && USE_ANALYTICS
|
|
if (!SConfig::GetInstance().m_analytics_permission_asked)
|
|
{
|
|
int answer =
|
|
wxMessageBox(_("If authorized, Dolphin can collect data on its performance, "
|
|
"feature usage, and configuration, as well as data on your system's "
|
|
"hardware and operating system.\n\n"
|
|
"No private data is ever collected. This data helps us understand "
|
|
"how people and emulated games use Dolphin and prioritize our "
|
|
"efforts. It also helps us identify rare configurations that are "
|
|
"causing bugs, performance and stability issues.\n"
|
|
"This authorization can be revoked at any time through Dolphin's "
|
|
"settings.\n\n"
|
|
"Do you authorize Dolphin to report this information to Dolphin's "
|
|
"developers?"),
|
|
_("Usage statistics reporting"), wxYES_NO, main_frame);
|
|
|
|
SConfig::GetInstance().m_analytics_permission_asked = true;
|
|
SConfig::GetInstance().m_analytics_enabled = (answer == wxYES);
|
|
SConfig::GetInstance().SaveSettings();
|
|
|
|
DolphinAnalytics::Instance()->ReloadConfig();
|
|
}
|
|
#endif
|
|
|
|
if (m_confirm_stop)
|
|
SConfig::GetInstance().bConfirmStop = m_confirm_setting;
|
|
|
|
if (m_play_movie && !m_movie_file.empty())
|
|
{
|
|
std::optional<std::string> savestate_path;
|
|
if (Movie::PlayInput(WxStrToStr(m_movie_file), &savestate_path))
|
|
{
|
|
if (m_boot)
|
|
{
|
|
m_boot->savestate_path = savestate_path;
|
|
main_frame->StartGame(std::move(m_boot));
|
|
}
|
|
else
|
|
{
|
|
main_frame->BootGame("", savestate_path);
|
|
}
|
|
}
|
|
}
|
|
// First check if we have an exec command line.
|
|
else if (m_boot)
|
|
{
|
|
main_frame->StartGame(std::move(m_boot));
|
|
}
|
|
// If we have selected Automatic Start, start the default ISO,
|
|
// or if no default ISO exists, start the last loaded ISO
|
|
else if (m_use_debugger)
|
|
{
|
|
if (main_frame->GetMenuBar()->IsChecked(IDM_AUTOMATIC_START))
|
|
{
|
|
main_frame->BootGame("");
|
|
}
|
|
}
|
|
}
|
|
|
|
void DolphinApp::OnActivate(wxActivateEvent& ev)
|
|
{
|
|
m_is_active = ev.GetActive();
|
|
}
|
|
|
|
void DolphinApp::InitLanguageSupport()
|
|
{
|
|
std::string language_code;
|
|
{
|
|
IniFile ini;
|
|
ini.Load(File::GetUserPath(F_DOLPHINCONFIG_IDX));
|
|
ini.GetOrCreateSection("Interface")->Get("LanguageCode", &language_code, "");
|
|
}
|
|
int language = wxLANGUAGE_UNKNOWN;
|
|
if (language_code.empty())
|
|
{
|
|
language = wxLANGUAGE_DEFAULT;
|
|
}
|
|
else
|
|
{
|
|
const wxLanguageInfo* language_info = wxLocale::FindLanguageInfo(StrToWxStr(language_code));
|
|
if (language_info)
|
|
language = language_info->Language;
|
|
}
|
|
|
|
// Load language if possible, fall back to system default otherwise
|
|
if (wxLocale::IsAvailable(language))
|
|
{
|
|
m_locale.reset(new wxLocale(language));
|
|
|
|
// Specify where dolphins *.gmo files are located on each operating system
|
|
#ifdef __WXMSW__
|
|
m_locale->AddCatalogLookupPathPrefix(StrToWxStr(File::GetExeDirectory() + DIR_SEP "Languages"));
|
|
#elif defined(__WXGTK__)
|
|
m_locale->AddCatalogLookupPathPrefix(StrToWxStr(DATA_DIR "../locale"));
|
|
#elif defined(__WXOSX__)
|
|
m_locale->AddCatalogLookupPathPrefix(
|
|
StrToWxStr(File::GetBundleDirectory() + "Contents/Resources"));
|
|
#endif
|
|
|
|
m_locale->AddCatalog("dolphin-emu");
|
|
|
|
if (!m_locale->IsOk())
|
|
{
|
|
wxMessageBox(_("Error loading selected language. Falling back to system default."),
|
|
_("Error"));
|
|
m_locale.reset(new wxLocale(wxLANGUAGE_DEFAULT));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
wxMessageBox(
|
|
_("The selected language is not supported by your system. Falling back to system default."),
|
|
_("Error"));
|
|
m_locale.reset(new wxLocale(wxLANGUAGE_DEFAULT));
|
|
}
|
|
|
|
// wxWidgets sets the C locale for us, but not the C++ locale, so let's do that ourselves
|
|
UICommon::SetLocale(language_code);
|
|
}
|
|
|
|
void DolphinApp::OnEndSession(wxCloseEvent& event)
|
|
{
|
|
// Close if we've received wxEVT_END_SESSION (ignore wxEVT_QUERY_END_SESSION)
|
|
if (!event.CanVeto())
|
|
{
|
|
main_frame->Close(true);
|
|
}
|
|
}
|
|
|
|
int DolphinApp::OnExit()
|
|
{
|
|
Core::Shutdown();
|
|
UICommon::Shutdown();
|
|
|
|
return wxApp::OnExit();
|
|
}
|
|
|
|
void DolphinApp::OnFatalException()
|
|
{
|
|
WiimoteReal::Shutdown();
|
|
}
|
|
|
|
void DolphinApp::OnIdle(wxIdleEvent& ev)
|
|
{
|
|
ev.Skip();
|
|
Core::HostDispatchJobs();
|
|
}
|
|
|
|
// ------------
|
|
// Talk to GUI
|
|
|
|
bool wxMsgAlert(const char* caption, const char* text, bool yes_no, MsgType /*style*/)
|
|
{
|
|
if (wxIsMainThread())
|
|
{
|
|
NetPlayDialog*& npd = NetPlayDialog::GetInstance();
|
|
if (npd != nullptr && npd->IsShown())
|
|
{
|
|
npd->AppendChat("/!\\ " + std::string{text});
|
|
return true;
|
|
}
|
|
return wxYES == wxMessageBox(StrToWxStr(text), StrToWxStr(caption),
|
|
wxSTAY_ON_TOP | ((yes_no) ? wxYES_NO : wxOK),
|
|
wxWindow::FindFocus());
|
|
}
|
|
else
|
|
{
|
|
wxCommandEvent event(wxEVT_HOST_COMMAND, IDM_PANIC);
|
|
event.SetString(StrToWxStr(caption) + ":" + StrToWxStr(text));
|
|
event.SetInt(yes_no);
|
|
main_frame->GetEventHandler()->AddPendingEvent(event);
|
|
main_frame->m_panic_event.Wait();
|
|
return main_frame->m_panic_result;
|
|
}
|
|
}
|
|
|
|
std::string wxStringTranslator(const char* text)
|
|
{
|
|
return WxStrToStr(wxGetTranslation(wxString::FromUTF8(text)));
|
|
}
|
|
|
|
// Accessor for the main window class
|
|
CFrame* DolphinApp::GetCFrame()
|
|
{
|
|
return main_frame;
|
|
}
|
|
|
|
void Host_Message(HostMessageID id)
|
|
{
|
|
if (id == HostMessageID::WMUserJobDispatch)
|
|
{
|
|
// Trigger a wxEVT_IDLE
|
|
wxWakeUpIdle();
|
|
return;
|
|
}
|
|
wxCommandEvent event(wxEVT_HOST_COMMAND, static_cast<int>(id));
|
|
main_frame->GetEventHandler()->AddPendingEvent(event);
|
|
}
|
|
|
|
void* Host_GetRenderHandle()
|
|
{
|
|
return main_frame->GetRenderHandle();
|
|
}
|
|
|
|
// OK, this thread boundary is DANGEROUS on Linux
|
|
// wxPostEvent / wxAddPendingEvent is the solution.
|
|
void Host_NotifyMapLoaded()
|
|
{
|
|
wxCommandEvent event(wxEVT_HOST_COMMAND, IDM_NOTIFY_MAP_LOADED);
|
|
main_frame->GetEventHandler()->AddPendingEvent(event);
|
|
|
|
if (main_frame->m_code_window)
|
|
{
|
|
main_frame->m_code_window->GetEventHandler()->AddPendingEvent(event);
|
|
}
|
|
}
|
|
|
|
void Host_UpdateDisasmDialog()
|
|
{
|
|
wxCommandEvent event(wxEVT_HOST_COMMAND, IDM_UPDATE_DISASM_DIALOG);
|
|
main_frame->GetEventHandler()->AddPendingEvent(event);
|
|
|
|
if (main_frame->m_code_window)
|
|
{
|
|
main_frame->m_code_window->GetEventHandler()->AddPendingEvent(event);
|
|
}
|
|
}
|
|
|
|
void Host_UpdateMainFrame()
|
|
{
|
|
wxCommandEvent event(wxEVT_HOST_COMMAND, IDM_UPDATE_GUI);
|
|
main_frame->GetEventHandler()->AddPendingEvent(event);
|
|
|
|
if (main_frame->m_code_window)
|
|
{
|
|
main_frame->m_code_window->GetEventHandler()->AddPendingEvent(event);
|
|
}
|
|
}
|
|
|
|
void Host_UpdateTitle(const std::string& title)
|
|
{
|
|
wxCommandEvent event(wxEVT_HOST_COMMAND, IDM_UPDATE_TITLE);
|
|
event.SetString(StrToWxStr(title));
|
|
main_frame->GetEventHandler()->AddPendingEvent(event);
|
|
}
|
|
|
|
void Host_RequestRenderWindowSize(int width, int height)
|
|
{
|
|
wxCommandEvent event(wxEVT_HOST_COMMAND, IDM_WINDOW_SIZE_REQUEST);
|
|
event.SetClientData(new std::pair<int, int>(width, height));
|
|
main_frame->GetEventHandler()->AddPendingEvent(event);
|
|
}
|
|
|
|
bool Host_UINeedsControllerState()
|
|
{
|
|
return wxGetApp().IsActiveThreadsafe() && GetUINeedsControllerState();
|
|
}
|
|
|
|
bool Host_RendererHasFocus()
|
|
{
|
|
return main_frame->RendererHasFocus();
|
|
}
|
|
|
|
bool Host_RendererIsFullscreen()
|
|
{
|
|
return main_frame->RendererIsFullscreen();
|
|
}
|
|
|
|
void Host_ShowVideoConfig(void* parent, const std::string& backend_name)
|
|
{
|
|
wxWindow* const parent_window = static_cast<wxWindow*>(parent);
|
|
|
|
if (backend_name == "Software Renderer")
|
|
{
|
|
SoftwareVideoConfigDialog diag(parent_window, backend_name);
|
|
diag.ShowModal();
|
|
}
|
|
else
|
|
{
|
|
VideoConfigDiag diag(parent_window, backend_name);
|
|
diag.ShowModal();
|
|
}
|
|
}
|
|
|
|
void Host_YieldToUI()
|
|
{
|
|
wxGetApp().GetMainLoop()->YieldFor(wxEVT_CATEGORY_UI);
|
|
}
|
|
|
|
void Host_UpdateProgressDialog(const char* caption, int position, int total)
|
|
{
|
|
wxCommandEvent event(wxEVT_HOST_COMMAND, IDM_UPDATE_PROGRESS_DIALOG);
|
|
event.SetString(StrToWxStr(caption));
|
|
event.SetInt(position);
|
|
event.SetExtraLong(total);
|
|
main_frame->GetEventHandler()->AddPendingEvent(event);
|
|
}
|