2015-11-27 00:33:07 -08:00
|
|
|
// Copyright 2015 Dolphin Emulator Project
|
|
|
|
// Licensed under GPLv2+
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-04-29 22:20:03 +02:00
|
|
|
#ifdef _WIN32
|
|
|
|
#include <Windows.h>
|
|
|
|
#include <cstdio>
|
|
|
|
#endif
|
|
|
|
|
2017-06-14 14:31:30 +02:00
|
|
|
#include <OptionParser.h>
|
2016-05-12 01:18:30 +00:00
|
|
|
#include <QAbstractEventDispatcher>
|
2015-11-27 00:33:07 -08:00
|
|
|
#include <QApplication>
|
2017-05-20 23:41:02 +02:00
|
|
|
#include <QObject>
|
2017-10-05 18:20:18 +02:00
|
|
|
#include <QPushButton>
|
2018-03-17 17:22:05 -07:00
|
|
|
#include <QWidget>
|
2015-11-27 00:33:07 -08:00
|
|
|
|
2017-07-25 21:11:47 -07:00
|
|
|
#include "Common/MsgHandler.h"
|
2020-03-31 20:51:34 +02:00
|
|
|
#include "Common/ScopeGuard.h"
|
2019-03-04 20:49:00 +01:00
|
|
|
|
2017-05-20 23:41:02 +02:00
|
|
|
#include "Core/Analytics.h"
|
2017-10-02 00:09:07 +02:00
|
|
|
#include "Core/Boot/Boot.h"
|
2018-07-14 23:07:23 -04:00
|
|
|
#include "Core/ConfigManager.h"
|
2015-11-27 00:33:07 -08:00
|
|
|
#include "Core/Core.h"
|
2019-03-04 20:49:00 +01:00
|
|
|
|
2018-07-07 00:40:15 +02:00
|
|
|
#include "DolphinQt/Host.h"
|
|
|
|
#include "DolphinQt/MainWindow.h"
|
2019-03-04 20:49:00 +01:00
|
|
|
#include "DolphinQt/QtUtils/ModalMessageBox.h"
|
2018-07-07 00:40:15 +02:00
|
|
|
#include "DolphinQt/QtUtils/RunOnObject.h"
|
|
|
|
#include "DolphinQt/Resources.h"
|
|
|
|
#include "DolphinQt/Settings.h"
|
|
|
|
#include "DolphinQt/Translation.h"
|
|
|
|
#include "DolphinQt/Updater.h"
|
2019-03-04 20:49:00 +01:00
|
|
|
|
2017-06-14 14:31:30 +02:00
|
|
|
#include "UICommon/CommandLineParse.h"
|
2015-11-27 00:33:07 -08:00
|
|
|
#include "UICommon/UICommon.h"
|
|
|
|
|
2019-06-16 23:45:37 -04:00
|
|
|
static bool QtMsgAlertHandler(const char* caption, const char* text, bool yes_no,
|
|
|
|
Common::MsgType style)
|
2017-07-25 21:11:47 -07:00
|
|
|
{
|
2020-03-31 20:51:34 +02:00
|
|
|
const bool called_from_cpu_thread = Core::IsCPUThread();
|
|
|
|
|
2018-05-21 15:27:12 -07:00
|
|
|
std::optional<bool> r = RunOnObject(QApplication::instance(), [&] {
|
2020-03-31 20:51:34 +02:00
|
|
|
Common::ScopeGuard scope_guard(&Core::UndeclareAsCPUThread);
|
|
|
|
if (called_from_cpu_thread)
|
|
|
|
{
|
|
|
|
// Temporarily declare this as the CPU thread to avoid getting a deadlock if any DolphinQt
|
|
|
|
// code calls RunAsCPUThread while the CPU thread is blocked on this function returning.
|
|
|
|
// Notably, if the panic alert steals focus from RenderWidget, Host::SetRenderFocus gets
|
|
|
|
// called, which can attempt to use RunAsCPUThread to get us out of exclusive fullscreen.
|
|
|
|
Core::DeclareAsCPUThread();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
scope_guard.Dismiss();
|
|
|
|
}
|
|
|
|
|
2019-08-01 19:52:28 +02:00
|
|
|
ModalMessageBox message_box(QApplication::activeWindow(), Qt::ApplicationModal);
|
2017-07-25 21:11:47 -07:00
|
|
|
message_box.setWindowTitle(QString::fromUtf8(caption));
|
|
|
|
message_box.setText(QString::fromUtf8(text));
|
2017-10-05 18:20:18 +02:00
|
|
|
|
2017-07-25 21:11:47 -07:00
|
|
|
message_box.setStandardButtons(yes_no ? QMessageBox::Yes | QMessageBox::No : QMessageBox::Ok);
|
2019-06-16 23:45:37 -04:00
|
|
|
if (style == Common::MsgType::Warning)
|
2017-10-05 18:20:18 +02:00
|
|
|
message_box.addButton(QMessageBox::Ignore)->setText(QObject::tr("Ignore for this session"));
|
|
|
|
|
2017-07-25 21:11:47 -07:00
|
|
|
message_box.setIcon([&] {
|
|
|
|
switch (style)
|
|
|
|
{
|
2019-06-16 23:45:37 -04:00
|
|
|
case Common::MsgType::Information:
|
2017-07-25 21:11:47 -07:00
|
|
|
return QMessageBox::Information;
|
2019-06-16 23:45:37 -04:00
|
|
|
case Common::MsgType::Question:
|
2017-07-25 21:11:47 -07:00
|
|
|
return QMessageBox::Question;
|
2019-06-16 23:45:37 -04:00
|
|
|
case Common::MsgType::Warning:
|
2017-07-25 21:11:47 -07:00
|
|
|
return QMessageBox::Warning;
|
2019-06-16 23:45:37 -04:00
|
|
|
case Common::MsgType::Critical:
|
2017-07-25 21:11:47 -07:00
|
|
|
return QMessageBox::Critical;
|
|
|
|
}
|
|
|
|
// appease MSVC
|
|
|
|
return QMessageBox::NoIcon;
|
|
|
|
}());
|
|
|
|
|
2017-10-05 18:20:18 +02:00
|
|
|
const int button = message_box.exec();
|
|
|
|
if (button == QMessageBox::Yes)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (button == QMessageBox::Ignore)
|
2019-07-20 21:04:27 +02:00
|
|
|
{
|
2019-06-16 23:45:37 -04:00
|
|
|
Common::SetEnableAlert(false);
|
2019-07-20 21:04:27 +02:00
|
|
|
return true;
|
|
|
|
}
|
2017-10-05 18:20:18 +02:00
|
|
|
|
|
|
|
return false;
|
2017-07-25 21:11:47 -07:00
|
|
|
});
|
2018-05-21 15:27:12 -07:00
|
|
|
if (r.has_value())
|
|
|
|
return *r;
|
|
|
|
return false;
|
2017-07-25 21:11:47 -07:00
|
|
|
}
|
|
|
|
|
2017-06-23 00:40:55 -07:00
|
|
|
// N.B. On Windows, this should be called from WinMain. Link against qtmain and specify
|
|
|
|
// /SubSystem:Windows
|
2015-11-27 00:33:07 -08:00
|
|
|
int main(int argc, char* argv[])
|
|
|
|
{
|
2018-04-29 22:20:03 +02:00
|
|
|
#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
|
|
|
|
|
2020-03-22 17:49:32 +13:00
|
|
|
auto parser = CommandLineParse::CreateParser(CommandLineParse::ParserOptions::IncludeGUIOptions);
|
|
|
|
const optparse::Values& options = CommandLineParse::ParseArguments(parser.get(), argc, argv);
|
|
|
|
const std::vector<std::string> args = parser->args();
|
|
|
|
|
2017-05-30 22:42:21 +02:00
|
|
|
QCoreApplication::setAttribute(Qt::AA_EnableHighDpiScaling);
|
|
|
|
QCoreApplication::setAttribute(Qt::AA_UseHighDpiPixmaps);
|
2017-07-05 17:35:47 -07:00
|
|
|
QCoreApplication::setOrganizationName(QStringLiteral("Dolphin Emulator"));
|
|
|
|
QCoreApplication::setOrganizationDomain(QStringLiteral("dolphin-emu.org"));
|
2018-01-01 12:31:53 +00:00
|
|
|
QCoreApplication::setApplicationName(QStringLiteral("dolphin-emu"));
|
2017-05-30 22:42:21 +02:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
QApplication app(argc, argv);
|
2015-11-27 00:33:07 -08:00
|
|
|
|
2018-05-05 11:56:42 +02:00
|
|
|
#ifdef _WIN32
|
2020-02-17 00:45:34 -08:00
|
|
|
// On Windows, Qt 5's default system font (MS Shell Dlg 2) is outdated.
|
|
|
|
// Interestingly, the QMenu font is correct and comes from lfMenuFont
|
|
|
|
// (Segoe UI on English computers).
|
|
|
|
// So use it for the entire application.
|
|
|
|
// This code will become unnecessary and obsolete once we switch to Qt 6.
|
|
|
|
QApplication::setFont(QApplication::font("QMenu"));
|
2018-05-05 11:56:42 +02:00
|
|
|
#endif
|
|
|
|
|
2018-04-29 22:20:03 +02:00
|
|
|
#ifdef _WIN32
|
|
|
|
FreeConsole();
|
|
|
|
#endif
|
|
|
|
|
2017-06-14 14:31:30 +02:00
|
|
|
UICommon::SetUserDirectory(static_cast<const char*>(options.get("user")));
|
2016-06-24 10:43:46 +02:00
|
|
|
UICommon::CreateDirectories();
|
|
|
|
UICommon::Init();
|
|
|
|
Resources::Init();
|
2019-10-27 19:03:10 +01:00
|
|
|
Settings::Instance().SetBatchModeEnabled(options.is_set("batch"));
|
2015-11-27 00:33:07 -08:00
|
|
|
|
2017-07-25 21:11:47 -07:00
|
|
|
// Hook up alerts from core
|
2019-06-16 23:45:37 -04:00
|
|
|
Common::RegisterMsgAlertHandler(QtMsgAlertHandler);
|
2017-07-25 21:11:47 -07:00
|
|
|
|
2017-07-17 14:50:40 -07:00
|
|
|
// Hook up translations
|
|
|
|
Translation::Initialize();
|
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
// Whenever the event loop is about to go to sleep, dispatch the jobs
|
|
|
|
// queued in the Core first.
|
|
|
|
QObject::connect(QAbstractEventDispatcher::instance(), &QAbstractEventDispatcher::aboutToBlock,
|
|
|
|
&app, &Core::HostDispatchJobs);
|
2016-05-12 01:18:30 +00:00
|
|
|
|
2020-05-06 22:10:30 -05:00
|
|
|
std::optional<std::string> save_state_path;
|
|
|
|
if (options.is_set("save_state"))
|
|
|
|
{
|
|
|
|
save_state_path = static_cast<const char*>(options.get("save_state"));
|
|
|
|
}
|
|
|
|
|
2017-10-02 00:09:07 +02:00
|
|
|
std::unique_ptr<BootParameters> boot;
|
2019-10-27 19:03:10 +01:00
|
|
|
bool game_specified = false;
|
2017-11-26 18:32:28 +01:00
|
|
|
if (options.is_set("exec"))
|
|
|
|
{
|
2018-11-05 19:20:45 +01:00
|
|
|
const std::list<std::string> paths_list = options.all("exec");
|
|
|
|
const std::vector<std::string> paths{std::make_move_iterator(std::begin(paths_list)),
|
|
|
|
std::make_move_iterator(std::end(paths_list))};
|
2020-05-06 22:10:30 -05:00
|
|
|
boot = BootParameters::GenerateFromFile(paths, save_state_path);
|
2019-10-27 19:03:10 +01:00
|
|
|
game_specified = true;
|
2017-11-26 18:32:28 +01:00
|
|
|
}
|
|
|
|
else if (options.is_set("nand_title"))
|
2017-10-02 00:09:07 +02:00
|
|
|
{
|
|
|
|
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);
|
|
|
|
boot = std::make_unique<BootParameters>(BootParameters::NANDTitle{title_id});
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-03-04 20:49:00 +01:00
|
|
|
ModalMessageBox::critical(nullptr, QObject::tr("Error"), QObject::tr("Invalid title ID."));
|
2017-10-02 00:09:07 +02:00
|
|
|
}
|
2019-10-27 19:03:10 +01:00
|
|
|
game_specified = true;
|
2017-10-02 00:09:07 +02:00
|
|
|
}
|
2018-03-16 09:14:24 +01:00
|
|
|
else if (!args.empty())
|
|
|
|
{
|
2020-05-06 22:10:30 -05:00
|
|
|
boot = BootParameters::GenerateFromFile(args.front(), save_state_path);
|
2019-10-27 19:03:10 +01:00
|
|
|
game_specified = true;
|
2018-03-16 09:14:24 +01:00
|
|
|
}
|
2017-10-02 00:09:07 +02:00
|
|
|
|
2018-03-20 00:41:47 -06:00
|
|
|
int retval;
|
2017-06-22 15:11:53 -07:00
|
|
|
|
2020-05-06 22:10:30 -05:00
|
|
|
if (save_state_path && !game_specified)
|
|
|
|
{
|
|
|
|
ModalMessageBox::critical(
|
|
|
|
nullptr, QObject::tr("Error"),
|
|
|
|
QObject::tr("A save state cannot be loaded without specifying a game to launch."));
|
|
|
|
retval = 1;
|
|
|
|
}
|
|
|
|
else if (Settings::Instance().IsBatchModeEnabled() && !game_specified)
|
2019-10-27 19:03:10 +01:00
|
|
|
{
|
|
|
|
ModalMessageBox::critical(
|
|
|
|
nullptr, QObject::tr("Error"),
|
|
|
|
QObject::tr("Batch mode cannot be used without specifying a game to launch."));
|
|
|
|
retval = 1;
|
|
|
|
}
|
2020-05-06 22:10:30 -05:00
|
|
|
else if (!boot && (Settings::Instance().IsBatchModeEnabled() || save_state_path))
|
2019-10-27 19:03:10 +01:00
|
|
|
{
|
|
|
|
// A game to launch was specified, but it was invalid.
|
|
|
|
// An error has already been shown by code above, so exit without showing another error.
|
|
|
|
retval = 1;
|
|
|
|
}
|
|
|
|
else
|
2016-07-13 05:52:45 +00:00
|
|
|
{
|
2019-06-23 19:26:07 +02:00
|
|
|
DolphinAnalytics::Instance().ReportDolphinStart("qt");
|
2017-05-20 23:41:02 +02:00
|
|
|
|
2019-03-27 14:26:17 +01:00
|
|
|
MainWindow win{std::move(boot), static_cast<const char*>(options.get("movie"))};
|
2018-05-04 04:02:05 -04:00
|
|
|
if (options.is_set("debugger"))
|
|
|
|
Settings::Instance().SetDebugModeEnabled(true);
|
2018-05-13 15:03:48 +02:00
|
|
|
win.Show();
|
2017-05-20 23:41:02 +02:00
|
|
|
|
|
|
|
#if defined(USE_ANALYTICS) && USE_ANALYTICS
|
2017-06-22 15:11:53 -07:00
|
|
|
if (!SConfig::GetInstance().m_analytics_permission_asked)
|
2017-05-20 23:41:02 +02:00
|
|
|
{
|
2019-03-04 20:49:00 +01:00
|
|
|
ModalMessageBox analytics_prompt(&win);
|
2017-05-20 23:41:02 +02:00
|
|
|
|
|
|
|
analytics_prompt.setIcon(QMessageBox::Question);
|
|
|
|
analytics_prompt.setStandardButtons(QMessageBox::Yes | QMessageBox::No);
|
2018-09-08 15:19:01 +02:00
|
|
|
analytics_prompt.setWindowTitle(QObject::tr("Allow Usage Statistics Reporting"));
|
2017-07-23 11:41:39 +02:00
|
|
|
analytics_prompt.setText(
|
|
|
|
QObject::tr("Do you authorize Dolphin to report information to Dolphin's developers?"));
|
2017-05-20 23:41:02 +02:00
|
|
|
analytics_prompt.setInformativeText(
|
|
|
|
QObject::tr("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 "
|
2017-07-23 11:41:39 +02:00
|
|
|
"settings."));
|
2017-05-20 23:41:02 +02:00
|
|
|
|
|
|
|
const int answer = analytics_prompt.exec();
|
|
|
|
|
2017-06-22 15:11:53 -07:00
|
|
|
SConfig::GetInstance().m_analytics_permission_asked = true;
|
2018-03-23 23:25:17 +01:00
|
|
|
Settings::Instance().SetAnalyticsEnabled(answer == QMessageBox::Yes);
|
2017-05-20 23:41:02 +02:00
|
|
|
|
2019-06-23 19:26:07 +02:00
|
|
|
DolphinAnalytics::Instance().ReloadConfig();
|
2017-05-20 23:41:02 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2019-08-11 20:58:04 +02:00
|
|
|
if (!Settings::Instance().IsBatchModeEnabled())
|
|
|
|
{
|
|
|
|
auto* updater = new Updater(&win);
|
|
|
|
updater->start();
|
|
|
|
}
|
2018-03-17 17:22:05 -07:00
|
|
|
|
2016-07-13 05:52:45 +00:00
|
|
|
retval = app.exec();
|
|
|
|
}
|
2015-11-27 00:33:07 -08:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
Core::Shutdown();
|
|
|
|
UICommon::Shutdown();
|
|
|
|
Host::GetInstance()->deleteLater();
|
2015-11-27 00:33:07 -08:00
|
|
|
|
2016-06-24 10:43:46 +02:00
|
|
|
return retval;
|
2015-11-27 00:33:07 -08:00
|
|
|
}
|