summaryrefslogtreecommitdiff
path: root/archived/projt-launcher/launcher/ui/MainWindow.cpp
diff options
context:
space:
mode:
authorMehmet Samet Duman <yongdohyun@projecttick.org>2026-04-02 18:51:45 +0300
committerMehmet Samet Duman <yongdohyun@projecttick.org>2026-04-02 18:51:45 +0300
commitd3261e64152397db2dca4d691a990c6bc2a6f4dd (patch)
treefac2f7be638651181a72453d714f0f96675c2b8b /archived/projt-launcher/launcher/ui/MainWindow.cpp
parent31b9a8949ed0a288143e23bf739f2eb64fdc63be (diff)
downloadProject-Tick-d3261e64152397db2dca4d691a990c6bc2a6f4dd.tar.gz
Project-Tick-d3261e64152397db2dca4d691a990c6bc2a6f4dd.zip
NOISSUE add archived projects
Signed-off-by: Mehmet Samet Duman <yongdohyun@projecttick.org>
Diffstat (limited to 'archived/projt-launcher/launcher/ui/MainWindow.cpp')
-rw-r--r--archived/projt-launcher/launcher/ui/MainWindow.cpp2134
1 files changed, 2134 insertions, 0 deletions
diff --git a/archived/projt-launcher/launcher/ui/MainWindow.cpp b/archived/projt-launcher/launcher/ui/MainWindow.cpp
new file mode 100644
index 0000000000..3702660c28
--- /dev/null
+++ b/archived/projt-launcher/launcher/ui/MainWindow.cpp
@@ -0,0 +1,2134 @@
+// SPDX-License-Identifier: GPL-3.0-only
+// SPDX-FileCopyrightText: 2026 Project Tick
+// SPDX-FileContributor: Project Tick Team
+/*
+ * ProjT Launcher - Minecraft Launcher
+ * Copyright (C) 2026 Project Tick
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ * === Upstream License Block (Do Not Modify) ==============================
+ *
+ *
+ *
+ * Prism Launcher - Minecraft Launcher
+ * Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
+ * Copyright (C) 2023 TheKodeToad <TheKodeToad@proton.me>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, version 3.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, write to the Free Software Foundation,
+ * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ * This file incorporates work covered by the following copyright and
+ * permission notice:
+ *
+ * Copyright 2013-2021 MultiMC Contributors
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ *
+ * ======================================================================== */
+
+#include "Application.h"
+#include "BuildConfig.h"
+#include "FileSystem.h"
+
+#include "MainWindow.h"
+#include "ui/LaunchMenu.h"
+#include "ui_MainWindow.h"
+
+#include <QDesktopServices>
+#include <QDir>
+#include <QFileInfo>
+#include <QUrl>
+#include <QUrlQuery>
+#include <QVariant>
+
+#include <QAction>
+#include <QActionGroup>
+#include <QApplication>
+#include <QButtonGroup>
+#include <QFileDialog>
+#include <QHBoxLayout>
+#include <QHeaderView>
+#include <QInputDialog>
+#include <QKeyEvent>
+#include <QLabel>
+#include <QMainWindow>
+#include <QMenu>
+#include <QMenuBar>
+#include <QMessageBox>
+#include <QProgressDialog>
+#include <QShortcut>
+#include <QStackedWidget>
+#include <QStatusBar>
+#include <QToolBar>
+#include <QToolButton>
+#include <QWidget>
+#include <QWidgetAction>
+#include <memory>
+
+#include <BaseInstance.h>
+#include <BuildConfig.h>
+#include <DesktopServices.h>
+#include <InstanceList.h>
+#include <MMCZip.h>
+#include <icons/IconList.hpp>
+#include <launch/LaunchPipeline.hpp>
+#include <minecraft/MinecraftInstance.h>
+#include <minecraft/auth/AccountList.hpp>
+#include <net/ApiDownload.h>
+#include <net/NetJob.h>
+#include <news/NewsChecker.h>
+#include <tools/BaseProfiler.h>
+#include <updater/ExternalUpdater.h>
+#include "InstanceWindow.h"
+
+#include "ui/GuiUtil.h"
+#include "ui/ViewLogWindow.h"
+#include "ui/dialogs/AboutDialog.h"
+#include "ui/dialogs/CopyInstanceDialog.h"
+#include "ui/dialogs/CreateShortcutDialog.h"
+#include "ui/dialogs/CustomMessageBox.h"
+#include "ui/dialogs/ExportInstanceDialog.h"
+#include "ui/dialogs/ExportPackDialog.h"
+#include "ui/dialogs/IconPickerDialog.h"
+#include "ui/dialogs/ImportResourceDialog.h"
+#include "ui/dialogs/NewInstanceDialog.h"
+#include "ui/dialogs/ProgressDialog.h"
+#include "ui/dialogs/NewsDialog.h"
+#include "ui/instanceview/InstanceDelegate.h"
+#include "ui/instanceview/InstanceProxyModel.h"
+#include "ui/instanceview/InstanceView.h"
+#include "ui/themes/Theme.h"
+#include "ui/themes/ThemeManager.h"
+#include "ui/widgets/LauncherHubWidget.h"
+#include "ui/widgets/LabeledToolButton.h"
+
+#include "minecraft/PackProfile.h"
+#include "minecraft/VersionFile.h"
+#include "minecraft/WorldList.h"
+#include "minecraft/mod/ModFolderModel.hpp"
+#include "minecraft/mod/ResourcePackFolderModel.hpp"
+#include "minecraft/mod/ShaderPackFolderModel.hpp"
+#include "minecraft/mod/TexturePackFolderModel.hpp"
+#include "minecraft/mod/tasks/LocalResourceParse.hpp"
+
+#include "modplatform/ModIndex.h"
+#include "modplatform/flame/FlameAPI.h"
+#include "modplatform/flame/FlameModIndex.h"
+
+#include "KonamiCode.h"
+
+#include "InstanceCopyTask.h"
+#include "InstanceDirUpdate.h"
+
+#include "Json.h"
+
+#include "MMCTime.h"
+
+namespace
+{
+ QString profileInUseFilter(const QString& profile, bool used)
+ {
+ if (used)
+ {
+ return QObject::tr("%1 (in use)").arg(profile);
+ }
+ else
+ {
+ return profile;
+ }
+ }
+} // namespace
+
+MainWindow::MainWindow(QWidget* parent) : QMainWindow(parent), ui(new Ui::MainWindow)
+{
+ ui->setupUi(this);
+
+#if defined(PROJT_DISABLE_LAUNCHER_HUB)
+ ui->actionLauncherHub->setVisible(false);
+#endif
+
+ setWindowIcon(APPLICATION->logo());
+ setWindowTitle(APPLICATION->applicationDisplayName());
+#ifndef QT_NO_ACCESSIBILITY
+ setAccessibleName(BuildConfig.LAUNCHER_DISPLAYNAME);
+#endif
+
+ // instance toolbar stuff
+ {
+ // Qt doesn't like vertical moving toolbars, so we have to force them...
+ // See https://github.com/PolyMC/PolyMC/issues/493
+ connect(ui->instanceToolBar,
+ &QToolBar::orientationChanged,
+ [this](Qt::Orientation) { ui->instanceToolBar->setOrientation(Qt::Vertical); });
+
+ // if you try to add a widget to a toolbar in a .ui file
+ // qt designer will delete it when you save the file >:(
+ changeIconButton = new LabeledToolButton(this);
+ changeIconButton->setObjectName(QStringLiteral("changeIconButton"));
+ changeIconButton->setIcon(QIcon::fromTheme("news"));
+ changeIconButton->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred);
+ connect(changeIconButton, &QToolButton::clicked, this, &MainWindow::on_actionChangeInstIcon_triggered);
+ ui->instanceToolBar->insertWidgetBefore(ui->actionLaunchInstance, changeIconButton);
+
+ renameButton = new LabeledToolButton(this);
+ renameButton->setObjectName(QStringLiteral("renameButton"));
+ renameButton->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred);
+ connect(renameButton, &QToolButton::clicked, this, &MainWindow::on_actionRenameInstance_triggered);
+ ui->instanceToolBar->insertWidgetBefore(ui->actionLaunchInstance, renameButton);
+
+ ui->instanceToolBar->insertSeparator(ui->actionLaunchInstance);
+
+ // restore the instance toolbar settings
+ auto const setting_name = QString("WideBarVisibility_%1").arg(ui->instanceToolBar->objectName());
+ instanceToolbarSetting = APPLICATION->settings()->getOrRegisterSetting(setting_name);
+
+ ui->instanceToolBar->setVisibilityState(
+ QByteArray::fromBase64(instanceToolbarSetting->get().toString().toUtf8()));
+
+ ui->instanceToolBar->addContextMenuAction(ui->newsToolBar->toggleViewAction());
+ ui->instanceToolBar->addContextMenuAction(ui->instanceToolBar->toggleViewAction());
+ ui->instanceToolBar->addContextMenuAction(ui->actionToggleStatusBar);
+ ui->instanceToolBar->addContextMenuAction(ui->actionLockToolbars);
+ }
+
+ // set the menu for the folders help, accounts, and export tool buttons
+ {
+ auto foldersMenuButton = dynamic_cast<QToolButton*>(ui->mainToolBar->widgetForAction(ui->actionFoldersButton));
+ ui->actionFoldersButton->setMenu(ui->foldersMenu);
+ foldersMenuButton->setPopupMode(QToolButton::InstantPopup);
+
+ helpMenuButton = dynamic_cast<QToolButton*>(ui->mainToolBar->widgetForAction(ui->actionHelpButton));
+ ui->actionHelpButton->setMenu(new QMenu(this));
+ ui->actionHelpButton->menu()->addActions(ui->helpMenu->actions());
+ ui->actionHelpButton->menu()->removeAction(ui->actionCheckUpdate);
+ helpMenuButton->setPopupMode(QToolButton::InstantPopup);
+
+ auto accountMenuButton = dynamic_cast<QToolButton*>(ui->mainToolBar->widgetForAction(ui->actionAccountsButton));
+ accountMenuButton->setPopupMode(QToolButton::InstantPopup);
+
+ auto exportInstanceMenu = new QMenu(this);
+ exportInstanceMenu->addAction(ui->actionExportInstanceZip);
+ exportInstanceMenu->addAction(ui->actionExportInstanceMrPack);
+ exportInstanceMenu->addAction(ui->actionExportInstanceFlamePack);
+ ui->actionExportInstance->setMenu(exportInstanceMenu);
+ }
+
+ // hide, disable and show stuff
+ {
+ ui->actionReportBug->setVisible(!BuildConfig.BUG_TRACKER_URL.isEmpty());
+ ui->actionMATRIX->setVisible(!BuildConfig.MATRIX_URL.isEmpty());
+ ui->actionDISCORD->setVisible(!BuildConfig.DISCORD_URL.isEmpty());
+ ui->actionREDDIT->setVisible(!BuildConfig.SUBREDDIT_URL.isEmpty());
+
+ ui->actionCheckUpdate->setVisible(APPLICATION->updaterEnabled());
+
+#ifndef Q_OS_MAC
+ ui->actionAddToPATH->setVisible(false);
+#endif
+
+ // disabled until we have an instance selected
+ ui->instanceToolBar->setEnabled(false);
+ setInstanceActionsEnabled(false);
+
+ // add a close button at the end of the main toolbar when running on gamescope / steam deck
+ // this is only needed on gamescope because it defaults to an X11/XWayland session and
+ // does not implement decorations
+ if (qgetenv("XDG_CURRENT_DESKTOP") == "gamescope")
+ {
+ ui->mainToolBar->addAction(ui->actionCloseWindow);
+ }
+
+ ui->actionViewJavaFolder->setEnabled(BuildConfig.JAVA_DOWNLOADER_ENABLED);
+ }
+
+ { // logs viewing
+ connect(ui->actionViewLog, &QAction::triggered, this, [] { APPLICATION->showLogWindow(); });
+ }
+
+ // add the toolbar toggles to the view menu
+ ui->viewMenu->addAction(ui->instanceToolBar->toggleViewAction());
+ ui->viewMenu->addAction(ui->newsToolBar->toggleViewAction());
+
+ updateThemeMenu();
+ updateMainToolBar();
+ // OSX magic.
+ setUnifiedTitleAndToolBarOnMac(true);
+
+ // Global shortcuts
+ {
+ // you can't set QKeySequence::StandardKey shortcuts in qt designer >:(
+ ui->actionAddInstance->setShortcut(QKeySequence::New);
+ ui->actionSettings->setShortcut(QKeySequence::Preferences);
+ ui->actionUndoTrashInstance->setShortcut(QKeySequence::Undo);
+ ui->actionDeleteInstance->setShortcuts({ QKeySequence(tr("Backspace")), QKeySequence::Delete });
+ ui->actionCloseWindow->setShortcut(QKeySequence::Close);
+ connect(ui->actionCloseWindow, &QAction::triggered, APPLICATION, &Application::closeCurrentWindow);
+
+ // Global quit shortcut (Ctrl+Q) - delegates to Application for proper cleanup
+ auto quitShortcut = new QShortcut(QKeySequence::Quit, this);
+ connect(quitShortcut, &QShortcut::activated, APPLICATION, &Application::quit);
+ }
+
+ // Konami Code
+ {
+ secretEventFilter = new KonamiCode(this);
+ connect(secretEventFilter, &KonamiCode::triggered, this, &MainWindow::konamiTriggered);
+ }
+
+ // Add the news label to the news toolbar.
+ {
+ m_newsChecker.reset(new NewsChecker(APPLICATION->network(), BuildConfig.NEWS_RSS_URL));
+ newsLabel = new QToolButton();
+ newsLabel->setIcon(QIcon::fromTheme("news"));
+ newsLabel->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Preferred);
+ newsLabel->setToolButtonStyle(Qt::ToolButtonTextBesideIcon);
+ newsLabel->setFocusPolicy(Qt::NoFocus);
+ ui->newsToolBar->insertWidget(ui->actionMoreNews, newsLabel);
+
+ connect(newsLabel, &QAbstractButton::clicked, this, &MainWindow::newsButtonClicked);
+ connect(m_newsChecker.get(), &NewsChecker::newsLoaded, this, &MainWindow::updateNewsLabel);
+ updateNewsLabel();
+ }
+
+ // Create the instance list widget
+ {
+ m_contentStack = new QStackedWidget(ui->centralWidget);
+ ui->horizontalLayout->addWidget(m_contentStack);
+
+ m_instancesPage = new QWidget(m_contentStack);
+ auto* instancesLayout = new QHBoxLayout(m_instancesPage);
+ instancesLayout->setContentsMargins(0, 0, 0, 0);
+ instancesLayout->setSpacing(0);
+
+ view = new InstanceView(m_instancesPage);
+
+ view->setSelectionMode(QAbstractItemView::SingleSelection);
+ // Delegate is owned by 'this', Qt will handle cleanup via parent-child relationship
+ auto delegate = new ListViewDelegate(this);
+ view->setItemDelegate(delegate);
+ view->setFrameShape(QFrame::NoFrame);
+ // do not show ugly blue border on the mac
+ view->setAttribute(Qt::WA_MacShowFocusRect, false);
+ connect(delegate,
+ &ListViewDelegate::textChanged,
+ this,
+ [this](QString before, QString after)
+ {
+ if (auto newRoot = askToUpdateInstanceDirName(m_selectedInstance, before, after, this);
+ !newRoot.isEmpty())
+ {
+ auto oldID = m_selectedInstance->id();
+ auto newID = QFileInfo(newRoot).fileName();
+ QString origGroup(APPLICATION->instances()->getInstanceGroup(oldID));
+ bool syncGroup = origGroup != GroupId() && oldID != newID;
+ if (syncGroup)
+ APPLICATION->instances()->setInstanceGroup(oldID, GroupId());
+
+ refreshInstances();
+ setSelectedInstanceById(newID);
+
+ if (syncGroup)
+ APPLICATION->instances()->setInstanceGroup(newID, origGroup);
+ }
+ });
+
+ view->installEventFilter(this);
+ view->setContextMenuPolicy(Qt::CustomContextMenu);
+ connect(view, &QWidget::customContextMenuRequested, this, &MainWindow::showInstanceContextMenu);
+ connect(view, &InstanceView::droppedURLs, this, &MainWindow::processURLs, Qt::QueuedConnection);
+
+ proxymodel = new InstanceProxyModel(this);
+ proxymodel->setSourceModel(APPLICATION->instances().get());
+ proxymodel->sort(0);
+ connect(proxymodel, &InstanceProxyModel::dataChanged, this, &MainWindow::instanceDataChanged);
+
+ view->setModel(proxymodel);
+ view->setSourceOfGroupCollapseStatus([](const QString& groupName) -> bool
+ { return APPLICATION->instances()->isGroupCollapsed(groupName); });
+ connect(view,
+ &InstanceView::groupStateChanged,
+ APPLICATION->instances().get(),
+ &InstanceList::on_GroupStateChanged);
+ instancesLayout->addWidget(view);
+ m_contentStack->addWidget(m_instancesPage);
+
+#if !defined(PROJT_DISABLE_LAUNCHER_HUB)
+ m_launcherHubWidget = new LauncherHubWidget(m_contentStack);
+ m_launcherHubWidget->hide();
+ m_contentStack->addWidget(m_launcherHubWidget);
+
+ connect(m_launcherHubWidget,
+ &LauncherHubWidget::selectInstanceRequested,
+ this,
+ [this](const QString& instanceId) { setSelectedInstanceById(instanceId); });
+ connect(m_launcherHubWidget,
+ &LauncherHubWidget::launchInstanceRequested,
+ this,
+ [this](const QString& instanceId)
+ {
+ setSelectedInstanceById(instanceId);
+ auto instance = APPLICATION->instances()->getInstanceById(instanceId);
+ if (instance && !instance->isRunning())
+ {
+ activateInstance(instance);
+ }
+ });
+ connect(m_launcherHubWidget,
+ &LauncherHubWidget::editInstanceRequested,
+ this,
+ [this](const QString& instanceId)
+ {
+ setSelectedInstanceById(instanceId);
+ on_actionEditInstance_triggered();
+ });
+ connect(m_launcherHubWidget,
+ &LauncherHubWidget::backupsRequested,
+ this,
+ [this](const QString& instanceId)
+ {
+ setSelectedInstanceById(instanceId);
+ on_actionManageBackups_triggered();
+ });
+ connect(m_launcherHubWidget,
+ &LauncherHubWidget::openInstanceFolderRequested,
+ this,
+ [this](const QString& instanceId)
+ {
+ setSelectedInstanceById(instanceId);
+ on_actionViewSelectedInstFolder_triggered();
+ });
+#endif
+ }
+ // The cat background
+ {
+ // set the cat action priority here so you can still see the action in qt designer
+ ui->actionCAT->setPriority(QAction::LowPriority);
+ bool cat_enable = APPLICATION->settings()->get("TheCat").toBool();
+ ui->actionCAT->setChecked(cat_enable);
+ connect(ui->actionCAT, &QAction::toggled, this, &MainWindow::onCatToggled);
+ connect(APPLICATION, &Application::currentCatChanged, this, &MainWindow::onCatChanged);
+ setCatBackground(cat_enable);
+ }
+
+ // Togglable status bar
+ {
+ bool statusBarVisible = APPLICATION->settings()->get("StatusBarVisible").toBool();
+ ui->actionToggleStatusBar->setChecked(statusBarVisible);
+ connect(ui->actionToggleStatusBar, &QAction::toggled, this, &MainWindow::setStatusBarVisibility);
+ setStatusBarVisibility(statusBarVisible);
+ }
+
+ // Lock toolbars
+ {
+ bool toolbarsLocked = APPLICATION->settings()->get("ToolbarsLocked").toBool();
+ ui->actionLockToolbars->setChecked(toolbarsLocked);
+ connect(ui->actionLockToolbars, &QAction::toggled, this, &MainWindow::lockToolbars);
+ lockToolbars(toolbarsLocked);
+ }
+ // start instance when double-clicked
+ connect(view, &InstanceView::activated, this, &MainWindow::instanceActivated);
+
+ // track the selection -- update the instance toolbar
+ connect(view->selectionModel(), &QItemSelectionModel::currentChanged, this, &MainWindow::instanceChanged);
+
+ // track icon changes and update the toolbar!
+ connect(APPLICATION->icons().get(), &projt::icons::IconList::iconUpdated, this, &MainWindow::iconUpdated);
+
+ // model reset -> selection is invalid. All the instance pointers are wrong.
+ connect(APPLICATION->instances().get(), &InstanceList::dataIsInvalid, this, &MainWindow::selectionBad);
+
+ // handle newly added instances
+ connect(APPLICATION->instances().get(),
+ &InstanceList::instanceSelectRequest,
+ this,
+ &MainWindow::instanceSelectRequest);
+
+ // When the global settings page closes, we want to know about it and update our state
+ connect(APPLICATION, &Application::globalSettingsApplied, this, &MainWindow::globalSettingsClosed);
+
+ m_statusLeft = new QLabel(tr("No instance selected"), this);
+ m_statusCenter = new QLabel(tr("Total playtime: 0s"), this);
+ statusBar()->addPermanentWidget(m_statusLeft, 1);
+ statusBar()->addPermanentWidget(m_statusCenter, 0);
+
+ // Add "manage accounts" button, right align
+ QWidget* spacer = new QWidget();
+ spacer->setSizePolicy(QSizePolicy::Expanding, QSizePolicy::Expanding);
+ ui->mainToolBar->insertWidget(ui->actionAccountsButton, spacer);
+
+ // Use undocumented property... https://stackoverflow.com/questions/7121718/create-a-scrollbar-in-a-submenu-qt
+ ui->accountsMenu->setStyleSheet("QMenu { menu-scrollable: 1; }");
+
+ repopulateAccountsMenu();
+
+ // Update the menu when the active account changes.
+ // Shouldn't have to use lambdas here like this, but if I don't, the compiler throws a fit.
+ // Template hell sucks...
+ connect(APPLICATION->accounts().get(), &AccountList::defaultAccountChanged, [this] { defaultAccountChanged(); });
+ connect(APPLICATION->accounts().get(), &AccountList::listChanged, [this] { defaultAccountChanged(); });
+
+ // Show initial account
+ defaultAccountChanged();
+
+ // load the news
+ {
+ m_newsChecker->reloadNews();
+ updateNewsLabel();
+ }
+
+ if (APPLICATION->updaterEnabled())
+ {
+ bool updatesAllowed = APPLICATION->updatesAreAllowed();
+ updatesAllowedChanged(updatesAllowed);
+
+ connect(ui->actionCheckUpdate, &QAction::triggered, this, &MainWindow::checkForUpdates);
+
+ // set up the updater object.
+ auto updater = APPLICATION->updater();
+
+ if (updater)
+ {
+ connect(updater.get(),
+ &ExternalUpdater::canCheckForUpdatesChanged,
+ this,
+ &MainWindow::updatesAllowedChanged);
+ }
+ }
+
+ connect(ui->actionUndoTrashInstance, &QAction::triggered, this, &MainWindow::undoTrashInstance);
+
+ setSelectedInstanceById(APPLICATION->settings()->get("SelectedInstance").toString());
+
+ // removing this looks stupid
+ view->setFocus();
+
+ retranslateUi();
+}
+
+// macOS always has a native menu bar, so these fixes are not applicable
+// Other systems may or may not have a native menu bar (most do not - it seems like only Ubuntu Unity does)
+#ifndef Q_OS_MAC
+void MainWindow::keyReleaseEvent(QKeyEvent* event)
+{
+ if (event->key() == Qt::Key_Alt && !APPLICATION->settings()->get("MenuBarInsteadOfToolBar").toBool())
+ ui->menuBar->setVisible(!ui->menuBar->isVisible());
+ else
+ QMainWindow::keyReleaseEvent(event);
+}
+#endif
+
+void MainWindow::retranslateUi()
+{
+ if (m_selectedInstance)
+ {
+ m_statusLeft->setText(m_selectedInstance->getStatusbarDescription());
+ }
+ else
+ {
+ m_statusLeft->setText(tr("No instance selected"));
+ }
+
+ ui->retranslateUi(this);
+
+ MinecraftAccountPtr defaultAccount = APPLICATION->accounts()->defaultAccount();
+ if (defaultAccount)
+ {
+ auto profileLabel = profileInUseFilter(defaultAccount->displayName(), defaultAccount->isInUse());
+ ui->actionAccountsButton->setText(profileLabel);
+ }
+
+ changeIconButton->setToolTip(ui->actionChangeInstIcon->toolTip());
+ renameButton->setToolTip(ui->actionRenameInstance->toolTip());
+
+ // replace the %1 with the launcher display name in some actions
+ if (helpMenuButton->toolTip().contains("%1"))
+ helpMenuButton->setToolTip(helpMenuButton->toolTip().arg(BuildConfig.LAUNCHER_DISPLAYNAME));
+
+ for (auto action : ui->helpMenu->actions())
+ {
+ if (action->text().contains("%1"))
+ action->setText(action->text().arg(BuildConfig.LAUNCHER_DISPLAYNAME));
+ if (action->toolTip().contains("%1"))
+ action->setToolTip(action->toolTip().arg(BuildConfig.LAUNCHER_DISPLAYNAME));
+ }
+}
+
+MainWindow::~MainWindow()
+{}
+
+QMenu* MainWindow::createPopupMenu()
+{
+ QMenu* filteredMenu = QMainWindow::createPopupMenu();
+ filteredMenu->removeAction(ui->mainToolBar->toggleViewAction());
+
+ filteredMenu->addAction(ui->actionToggleStatusBar);
+ filteredMenu->addAction(ui->actionLockToolbars);
+
+ return filteredMenu;
+}
+void MainWindow::setStatusBarVisibility(bool state)
+{
+ statusBar()->setVisible(state);
+ APPLICATION->settings()->set("StatusBarVisible", state);
+}
+void MainWindow::lockToolbars(bool state)
+{
+ ui->mainToolBar->setMovable(!state);
+ ui->instanceToolBar->setMovable(!state);
+ ui->newsToolBar->setMovable(!state);
+ APPLICATION->settings()->set("ToolbarsLocked", state);
+}
+
+void MainWindow::konamiTriggered()
+{
+ QString gradient =
+ " stop:0 rgba(125, 0, 0, 255), stop:0.166 rgba(125, 125, 0, 255), stop:0.333 rgba(0, 125, 0, 255), stop:0.5 "
+ "rgba(0, 125, 125, "
+ "255), stop:0.666 rgba(0, 0, 125, 255), stop:0.833 rgba(125, 0, 125, 255), stop:1 rgba(125, 0, 0, 255));";
+ QString stylesheet = "background-color: qlineargradient(spread:pad, x1:0, y1:0, x2:1, y2:0," + gradient;
+ if (ui->mainToolBar->styleSheet() == stylesheet)
+ {
+ ui->mainToolBar->setStyleSheet("");
+ ui->instanceToolBar->setStyleSheet("");
+ ui->centralWidget->setStyleSheet("");
+ ui->newsToolBar->setStyleSheet("");
+ ui->statusBar->setStyleSheet("");
+ qDebug() << "Super Secret Mode DEACTIVATED!";
+ }
+ else
+ {
+ ui->mainToolBar->setStyleSheet(stylesheet);
+ ui->instanceToolBar->setStyleSheet("background-color: qlineargradient(spread:pad, x1:0, y1:0, x2:0, y2:1,"
+ + gradient);
+ ui->centralWidget->setStyleSheet("background-color: qlineargradient(spread:pad, x1:0, y1:0, x2:1, y2:1,"
+ + gradient);
+ ui->newsToolBar->setStyleSheet(stylesheet);
+ ui->statusBar->setStyleSheet(stylesheet);
+ qDebug() << "Super Secret Mode ACTIVATED!";
+ }
+}
+
+void MainWindow::showInstanceContextMenu(const QPoint& pos)
+{
+ QList<QAction*> actions;
+
+ QAction* actionSep = new QAction("", this);
+ actionSep->setSeparator(true);
+
+ bool onInstance = view->indexAt(pos).isValid();
+ if (onInstance)
+ {
+ // reuse the file menu actions
+ actions = ui->fileMenu->actions();
+
+ // remove the add instance action, launcher settings action and close action
+ actions.removeFirst();
+ actions.removeLast();
+ actions.removeLast();
+
+ // add backup action after export actions
+ int exportIndex = actions.indexOf(ui->actionExportInstance);
+ if (exportIndex >= 0)
+ {
+ actions.insert(exportIndex + 1, ui->actionManageBackups);
+ }
+
+ actions.prepend(ui->actionChangeInstIcon);
+ actions.prepend(ui->actionRenameInstance);
+
+ // add header
+ actions.prepend(actionSep);
+ QAction* actionVoid = new QAction(m_selectedInstance->name(), this);
+ actionVoid->setEnabled(false);
+ actions.prepend(actionVoid);
+ }
+ else
+ {
+ auto group = view->groupNameAt(pos);
+
+ QAction* actionVoid = new QAction(group.isNull() ? BuildConfig.LAUNCHER_DISPLAYNAME : group, this);
+ actionVoid->setEnabled(false);
+
+ QAction* actionCreateInstance = new QAction(tr("&Create instance"), this);
+ actionCreateInstance->setToolTip(ui->actionAddInstance->toolTip());
+ if (!group.isNull())
+ {
+ QVariantMap instance_action_data;
+ instance_action_data["group"] = group;
+ actionCreateInstance->setData(instance_action_data);
+ }
+
+ connect(actionCreateInstance, &QAction::triggered, this, &MainWindow::on_actionAddInstance_triggered);
+
+ actions.prepend(actionSep);
+ actions.prepend(actionVoid);
+ actions.append(actionCreateInstance);
+ if (!group.isNull())
+ {
+ QAction* actionDeleteGroup = new QAction(tr("&Delete group"), this);
+ connect(actionDeleteGroup, &QAction::triggered, this, [this, group] { deleteGroup(group); });
+ actions.append(actionDeleteGroup);
+
+ QAction* actionRenameGroup = new QAction(tr("&Rename group"), this);
+ connect(actionRenameGroup, &QAction::triggered, this, [this, group] { renameGroup(group); });
+ actions.append(actionRenameGroup);
+ }
+ }
+ QMenu myMenu;
+ myMenu.addActions(actions);
+ /*
+ if (onInstance)
+ myMenu.setEnabled(m_selectedInstance->canLaunch());
+ */
+ myMenu.exec(view->mapToGlobal(pos));
+}
+
+void MainWindow::updateMainToolBar()
+{
+ ui->menuBar->setVisible(APPLICATION->settings()->get("MenuBarInsteadOfToolBar").toBool());
+ ui->mainToolBar->setVisible(ui->menuBar->isNativeMenuBar()
+ || !APPLICATION->settings()->get("MenuBarInsteadOfToolBar").toBool());
+}
+
+void MainWindow::updateLaunchButton()
+{
+ QMenu* launchMenu = ui->actionLaunchInstance->menu();
+ if (launchMenu)
+ launchMenu->clear();
+ else
+ launchMenu = new QMenu(this);
+ if (m_selectedInstance)
+ LaunchMenu::populate(m_selectedInstance, launchMenu);
+ ui->actionLaunchInstance->setMenu(launchMenu);
+}
+
+void MainWindow::updateThemeMenu()
+{
+ QMenu* themeMenu = ui->actionChangeTheme->menu();
+
+ if (themeMenu)
+ {
+ themeMenu->clear();
+ }
+ else
+ {
+ themeMenu = new QMenu(this);
+ }
+
+ auto themes = APPLICATION->themeManager()->getValidApplicationThemes();
+
+ QActionGroup* themesGroup = new QActionGroup(this);
+
+ for (auto* theme : themes)
+ {
+ QAction* themeAction = themeMenu->addAction(theme->name());
+
+ themeAction->setCheckable(true);
+ if (APPLICATION->settings()->get("ApplicationTheme").toString() == theme->id())
+ {
+ themeAction->setChecked(true);
+ }
+ themeAction->setActionGroup(themesGroup);
+
+ connect(themeAction,
+ &QAction::triggered,
+ [theme]()
+ {
+ APPLICATION->themeManager()->setApplicationTheme(theme->id());
+ APPLICATION->settings()->set("ApplicationTheme", theme->id());
+ });
+ }
+
+ ui->actionChangeTheme->setMenu(themeMenu);
+}
+
+void MainWindow::repopulateAccountsMenu()
+{
+ ui->accountsMenu->clear();
+
+ // NOTE: this is done so the accounts button text is not set to the accounts menu title
+ QMenu* accountsButtonMenu = ui->actionAccountsButton->menu();
+ if (accountsButtonMenu)
+ {
+ accountsButtonMenu->clear();
+ }
+ else
+ {
+ accountsButtonMenu = new QMenu(this);
+ ui->actionAccountsButton->setMenu(accountsButtonMenu);
+ }
+
+ auto accounts = APPLICATION->accounts();
+ MinecraftAccountPtr defaultAccount = accounts->defaultAccount();
+
+ QString active_profileId = "";
+ if (defaultAccount)
+ {
+ // this can be called before accountMenuButton exists
+ if (ui->actionAccountsButton)
+ {
+ auto profileLabel = profileInUseFilter(defaultAccount->displayName(), defaultAccount->isInUse());
+ ui->actionAccountsButton->setText(profileLabel);
+ }
+ }
+
+ QActionGroup* accountsGroup = new QActionGroup(this);
+
+ if (accounts->count() <= 0)
+ {
+ ui->actionNoAccountsAdded->setEnabled(false);
+ ui->accountsMenu->addAction(ui->actionNoAccountsAdded);
+ }
+ else
+ {
+ for (int i = 0; i < accounts->count(); i++)
+ {
+ MinecraftAccountPtr account = accounts->at(i);
+ auto profileLabel = profileInUseFilter(account->displayName(), account->isInUse());
+ QAction* action = new QAction(profileLabel, this);
+ action->setData(i);
+ action->setCheckable(true);
+ action->setActionGroup(accountsGroup);
+ if (defaultAccount == account)
+ {
+ action->setChecked(true);
+ }
+
+ auto face = account->getFace();
+ if (!face.isNull())
+ {
+ action->setIcon(face);
+ }
+ else
+ {
+ action->setIcon(QIcon::fromTheme("noaccount"));
+ }
+
+ const int highestNumberKey = 9;
+ if (i < highestNumberKey)
+ {
+ action->setShortcut(QKeySequence(tr("Ctrl+%1").arg(i + 1)));
+ }
+
+ ui->accountsMenu->addAction(action);
+ connect(action, &QAction::triggered, this, &MainWindow::changeActiveAccount);
+ }
+ }
+
+ ui->accountsMenu->addSeparator();
+
+ ui->actionNoDefaultAccount->setData(-1);
+ ui->actionNoDefaultAccount->setChecked(!defaultAccount);
+ ui->actionNoDefaultAccount->setActionGroup(accountsGroup);
+
+ ui->accountsMenu->addAction(ui->actionNoDefaultAccount);
+
+ connect(ui->actionNoDefaultAccount, &QAction::triggered, this, &MainWindow::changeActiveAccount);
+
+ ui->accountsMenu->addSeparator();
+ ui->accountsMenu->addAction(ui->actionManageAccounts);
+
+ accountsButtonMenu->addActions(ui->accountsMenu->actions());
+}
+
+void MainWindow::updatesAllowedChanged(bool allowed)
+{
+ if (!APPLICATION->updaterEnabled())
+ {
+ return;
+ }
+ ui->actionCheckUpdate->setEnabled(allowed);
+}
+
+/*
+ * Assumes the sender is a QAction
+ */
+void MainWindow::changeActiveAccount()
+{
+ QAction* sAction = (QAction*)sender();
+
+ // Profile's associated Mojang username
+ if (sAction->data().typeId() != QMetaType::Int)
+ return;
+
+ QVariant action_data = sAction->data();
+ bool valid = false;
+ int index = action_data.toInt(&valid);
+ if (!valid)
+ {
+ index = -1;
+ }
+ auto accounts = APPLICATION->accounts();
+ accounts->setDefaultAccount(index == -1 ? nullptr : accounts->at(index));
+ defaultAccountChanged();
+}
+
+void MainWindow::defaultAccountChanged()
+{
+ repopulateAccountsMenu();
+
+ MinecraftAccountPtr account = APPLICATION->accounts()->defaultAccount();
+
+ if (account && account->profileName() != "")
+ {
+ auto profileLabel = profileInUseFilter(account->displayName(), account->isInUse());
+ ui->actionAccountsButton->setText(profileLabel);
+ auto face = account->getFace();
+ if (face.isNull())
+ {
+ ui->actionAccountsButton->setIcon(QIcon::fromTheme("noaccount"));
+ }
+ else
+ {
+ ui->actionAccountsButton->setIcon(face);
+ }
+ return;
+ }
+
+ // Set the icon to the "no account" icon.
+ ui->actionAccountsButton->setIcon(QIcon::fromTheme("noaccount"));
+ ui->actionAccountsButton->setText(tr("Accounts"));
+}
+
+bool MainWindow::eventFilter(QObject* obj, QEvent* ev)
+{
+ if (obj == view)
+ {
+ if (ev->type() == QEvent::KeyPress)
+ {
+ secretEventFilter->input(ev);
+ QKeyEvent* keyEvent = static_cast<QKeyEvent*>(ev);
+ switch (keyEvent->key())
+ {
+ /*
+ case Qt::Key_Enter:
+ case Qt::Key_Return:
+ activateInstance(m_selectedInstance);
+ return true;
+ */
+ case Qt::Key_Delete: on_actionDeleteInstance_triggered(); return true;
+ case Qt::Key_F5: refreshInstances(); return true;
+ case Qt::Key_F2: on_actionRenameInstance_triggered(); return true;
+ default: break;
+ }
+ }
+ }
+ return QMainWindow::eventFilter(obj, ev);
+}
+
+void MainWindow::updateNewsLabel()
+{
+ if (m_newsChecker->isLoadingNews())
+ {
+ newsLabel->setText(tr("Loading news..."));
+ newsLabel->setEnabled(false);
+ ui->actionMoreNews->setVisible(false);
+ }
+ else
+ {
+ QList<NewsEntryPtr> entries = m_newsChecker->getNewsEntries();
+ if (entries.length() > 0)
+ {
+ newsLabel->setText(entries[0]->title);
+ newsLabel->setEnabled(true);
+ ui->actionMoreNews->setVisible(true);
+ }
+ else
+ {
+ newsLabel->setText(tr("No news available."));
+ newsLabel->setEnabled(false);
+ ui->actionMoreNews->setVisible(false);
+ }
+ }
+}
+
+QList<int> stringToIntList(const QString& string)
+{
+ QStringList split = string.split(',', Qt::SkipEmptyParts);
+ QList<int> out;
+ for (int i = 0; i < split.size(); ++i)
+ {
+ out.append(split.at(i).toInt());
+ }
+ return out;
+}
+QString intListToString(const QList<int>& list)
+{
+ QStringList slist;
+ for (int i = 0; i < list.size(); ++i)
+ {
+ slist.append(QString::number(list.at(i)));
+ }
+ return slist.join(',');
+}
+
+void MainWindow::onCatToggled(bool state)
+{
+ setCatBackground(state);
+ APPLICATION->settings()->set("TheCat", state);
+}
+
+void MainWindow::setCatBackground(bool enabled)
+{
+ view->setPaintCat(enabled);
+ view->viewport()->repaint();
+}
+
+void MainWindow::runModalTask(Task* task)
+{
+ if (!task)
+ return;
+ connect(task,
+ &Task::failed,
+ [this](QString reason)
+ { CustomMessageBox::selectable(this, tr("Error"), reason, QMessageBox::Critical)->show(); });
+ connect(
+ task,
+ &Task::succeeded,
+ [this, task]()
+ {
+ QStringList warnings = task->warnings();
+ if (warnings.count())
+ {
+ CustomMessageBox::selectable(this, tr("Warnings"), warnings.join('\n'), QMessageBox::Warning)->show();
+ }
+ });
+ connect(task,
+ &Task::aborted,
+ [this]
+ {
+ CustomMessageBox::selectable(this,
+ tr("Task aborted"),
+ tr("The task has been aborted by the user."),
+ QMessageBox::Information)
+ ->show();
+ });
+ ProgressDialog loadDialog(this);
+ loadDialog.setSkipButton(true, tr("Abort"));
+ loadDialog.execWithTask(*task);
+}
+
+void MainWindow::instanceFromInstanceTask(InstanceTask* rawTask)
+{
+ unique_qobject_ptr<Task> task(APPLICATION->instances()->wrapInstanceTask(rawTask));
+ runModalTask(task.get());
+}
+
+void MainWindow::on_actionCopyInstance_triggered()
+{
+ if (!m_selectedInstance)
+ return;
+
+ CopyInstanceDialog copyInstDlg(m_selectedInstance, this);
+ if (!copyInstDlg.exec())
+ return;
+
+ auto copyTask = new InstanceCopyTask(m_selectedInstance, copyInstDlg.getChosenOptions());
+ copyTask->setName(copyInstDlg.instName());
+ copyTask->setGroup(copyInstDlg.instGroup());
+ copyTask->setIcon(copyInstDlg.iconKey());
+ unique_qobject_ptr<Task> task(APPLICATION->instances()->wrapInstanceTask(copyTask));
+ runModalTask(task.get());
+}
+
+void MainWindow::addInstance(const QString& url, const QMap<QString, QString>& extra_info)
+{
+ QString groupName;
+ do
+ {
+ QObject* obj = sender();
+ if (!obj)
+ break;
+ QAction* action = qobject_cast<QAction*>(obj);
+ if (!action)
+ break;
+ auto map = action->data().toMap();
+ if (!map.contains("group"))
+ break;
+ groupName = map["group"].toString();
+ }
+ while (0);
+
+ if (groupName.isEmpty())
+ {
+ groupName = APPLICATION->settings()->get("LastUsedGroupForNewInstance").toString();
+ }
+
+ NewInstanceDialog newInstDlg(groupName, url, extra_info, this);
+ if (!newInstDlg.exec())
+ return;
+
+ APPLICATION->settings()->set("LastUsedGroupForNewInstance", newInstDlg.instGroup());
+
+ InstanceTask* creationTask = newInstDlg.extractTask();
+ if (creationTask)
+ {
+ instanceFromInstanceTask(creationTask);
+ }
+}
+
+void MainWindow::on_actionAddInstance_triggered()
+{
+ addInstance();
+}
+
+void MainWindow::processURLs(QList<QUrl> urls)
+{
+ // NOTE: This loop only processes one dropped file!
+ for (auto& url : urls)
+ {
+ qDebug() << "Processing" << url;
+
+ // The isLocalFile() check below doesn't work as intended without an explicit scheme.
+ if (url.scheme().isEmpty())
+ url.setScheme("file");
+
+ ModPlatform::IndexedVersion version;
+ QMap<QString, QString> extra_info;
+ QUrl local_url;
+ if (!url.isLocalFile())
+ { // download the remote resource and identify
+ const bool isExternalURLImport =
+ (url.host().toLower() == "import") || (url.path().startsWith("/import", Qt::CaseInsensitive));
+
+ QUrl dl_url;
+ if (url.scheme() == "curseforge" || (url.scheme() == BuildConfig.LAUNCHER_APP_BINARY_NAME && url.host() == "install"))
+ {
+ // need to find the download link for the modpack / resource
+ // format of url curseforge://install?addonId=IDHERE&fileId=IDHERE
+ // format of url binaryname://install?platform=curseforge&addonId=IDHERE&fileId=IDHERE
+ QUrlQuery query(url);
+
+ if (url.scheme() == BuildConfig.LAUNCHER_APP_BINARY_NAME)
+ {
+ if (query.queryItemValue("platform").toLower() != "curseforge")
+ {
+ qDebug() << "Invalid mod distribution platform:" << query.queryItemValue("platform");
+ continue;
+ }
+ }
+
+ if (query.allQueryItemValues("addonId").isEmpty() || query.allQueryItemValues("fileId").isEmpty())
+ {
+ qDebug() << "Invalid curseforge link:" << url;
+ continue;
+ }
+
+ auto addonId = query.allQueryItemValues("addonId")[0];
+ auto fileId = query.allQueryItemValues("fileId")[0];
+
+ extra_info.insert("pack_id", addonId);
+ extra_info.insert("pack_version_id", fileId);
+
+ auto array = std::make_shared<QByteArray>();
+
+ auto api = FlameAPI();
+ auto job = api.getFile(addonId, fileId, array);
+
+ connect(job.get(),
+ &Task::failed,
+ this,
+ [this](QString reason)
+ { CustomMessageBox::selectable(this, tr("Error"), reason, QMessageBox::Critical)->show(); });
+ connect(job.get(),
+ &Task::succeeded,
+ this,
+ [this, array, addonId, fileId, &dl_url, &version]
+ {
+ qDebug() << "Returned CFURL Json:\n" << array->toStdString().c_str();
+ auto doc = Json::requireDocument(*array);
+ auto data = Json::ensureObject(Json::ensureObject(doc.object()), "data");
+ // No way to find out if it's a mod or a modpack before here
+ // And also we need to check if it ends with .zip, instead of any better way
+ version = FlameMod::loadIndexedPackVersion(data);
+ auto fileName = version.fileName;
+
+ // Have to use ensureString then use QUrl to get proper url encoding
+ dl_url = QUrl(version.downloadUrl);
+ if (!dl_url.isValid())
+ {
+ CustomMessageBox::selectable(this,
+ tr("Error"),
+ tr("The modpack, mod, or resource %1 is blocked for "
+ "third-parties! Please download it manually.")
+ .arg(fileName),
+ QMessageBox::Critical)
+ ->show();
+ return;
+ }
+
+ QFileInfo dl_file(dl_url.fileName());
+ });
+
+ { // drop stack
+ ProgressDialog dlUrlDialod(this);
+ dlUrlDialod.setSkipButton(true, tr("Abort"));
+ dlUrlDialod.execWithTask(*job);
+ }
+ }
+ else if (url.scheme() == BuildConfig.LAUNCHER_APP_BINARY_NAME && !isExternalURLImport)
+ {
+ QVariantMap receivedData;
+ const QUrlQuery query(url.query());
+ const auto items = query.queryItems();
+ for (auto it = items.begin(), end = items.end(); it != end; ++it)
+ receivedData.insert(it->first, it->second);
+ emit APPLICATION->oauthReplyRecieved(receivedData);
+ continue;
+ }
+ else if ((url.scheme() == "prismlauncher" || url.scheme() == BuildConfig.LAUNCHER_APP_BINARY_NAME)
+ && isExternalURLImport)
+ {
+ const auto host = url.host().toLower();
+ const auto path = url.path();
+
+ QString encodedTarget;
+ {
+ QUrlQuery query(url);
+ const auto values = query.allQueryItemValues("url");
+ if (!values.isEmpty())
+ {
+ encodedTarget = values.first();
+ }
+ }
+
+ if (encodedTarget.isEmpty())
+ {
+ QString p = path;
+ if (p.startsWith("/import/", Qt::CaseInsensitive))
+ {
+ p = p.mid(QString("/import/").size());
+ }
+ else if (host == "import" && p.startsWith("/"))
+ {
+ p = p.mid(1);
+ }
+
+ if (!p.isEmpty() && p != "/import")
+ {
+ encodedTarget = p;
+ }
+ }
+
+ if (encodedTarget.isEmpty())
+ {
+ CustomMessageBox::selectable(
+ this, tr("Error"), tr("Invalid import link: missing 'url' parameter."), QMessageBox::Critical)
+ ->show();
+ continue;
+ }
+
+ const QString decodedStr = QUrl::fromPercentEncoding(encodedTarget.toUtf8()).trimmed();
+ QUrl target = QUrl::fromUserInput(decodedStr);
+
+ if (!target.isValid() || (target.scheme() != "https" && target.scheme() != "http"))
+ {
+ CustomMessageBox::selectable(
+ this, tr("Error"), tr("Invalid import link: URL must be http(s)."), QMessageBox::Critical)
+ ->show();
+ continue;
+ }
+
+ const auto res = QMessageBox::question(this,
+ tr("Install modpack"),
+ tr("Do you want to download and import a modpack from:\n%1\n\nURL:\n%2")
+ .arg(target.host(), target.toString()),
+ QMessageBox::Yes | QMessageBox::No,
+ QMessageBox::Yes);
+ if (res != QMessageBox::Yes)
+ {
+ continue;
+ }
+
+ dl_url = target;
+ }
+ else
+ {
+ dl_url = url;
+ }
+
+ if (!dl_url.isValid())
+ {
+ continue; // no valid url to download this resource
+ }
+
+ const QString path = dl_url.host() + '/' + dl_url.path();
+ auto entry = APPLICATION->metacache()->resolveEntry("general", path);
+ entry->setStale(true);
+ auto dl_job = unique_qobject_ptr<NetJob>(new NetJob(tr("Modpack download"), APPLICATION->network()));
+ dl_job->addNetAction(Net::ApiDownload::makeCached(dl_url, entry));
+ auto archivePath = entry->getFullPath();
+
+ bool dl_success = false;
+ connect(dl_job.get(),
+ &Task::failed,
+ this,
+ [this](QString reason)
+ { CustomMessageBox::selectable(this, tr("Error"), reason, QMessageBox::Critical)->show(); });
+ connect(dl_job.get(), &Task::succeeded, this, [&dl_success] { dl_success = true; });
+
+ { // drop stack
+ ProgressDialog dlUrlDialod(this);
+ dlUrlDialod.setSkipButton(true, tr("Abort"));
+ dlUrlDialod.execWithTask(*dl_job);
+ }
+
+ if (!dl_success)
+ {
+ continue; // no local file to identify
+ }
+ local_url = QUrl::fromLocalFile(archivePath);
+ }
+ else
+ {
+ local_url = url;
+ }
+
+ auto localFileName = QDir::toNativeSeparators(local_url.toLocalFile());
+ QFileInfo localFileInfo(localFileName);
+
+ auto type = ResourceUtils::identify(localFileInfo);
+
+ if (ModPlatform::ResourceTypeUtils::VALID_RESOURCES.count(type) == 0)
+ { // probably instance/modpack
+ addInstance(localFileName, extra_info);
+ continue;
+ }
+
+ if (APPLICATION->instances()->count() <= 0)
+ {
+ CustomMessageBox::selectable(
+ this,
+ tr("No instance!"),
+ tr("No instance available to add the resource to.\nPlease create a new instance before "
+ "attempting to install this resource again."),
+ QMessageBox::Critical)
+ ->show();
+ continue;
+ }
+ ImportResourceDialog dlg(localFileName, type, this);
+
+ if (dlg.exec() != QDialog::Accepted)
+ continue;
+
+ qDebug() << "Adding resource" << localFileName << "to" << dlg.selectedInstanceKey;
+
+ auto inst = APPLICATION->instances()->getInstanceById(dlg.selectedInstanceKey);
+ auto minecraftInst = std::dynamic_pointer_cast<MinecraftInstance>(inst);
+
+ switch (type)
+ {
+ case ModPlatform::ResourceType::ResourcePack:
+ minecraftInst->resourcePackList()->installResourceWithFlameMetadata(localFileName, version);
+ break;
+ case ModPlatform::ResourceType::TexturePack:
+ minecraftInst->texturePackList()->installResourceWithFlameMetadata(localFileName, version);
+ break;
+ case ModPlatform::ResourceType::DataPack:
+ qWarning() << "Importing of Data Packs not supported at this time. Ignoring" << localFileName;
+ break;
+ case ModPlatform::ResourceType::Mod:
+ minecraftInst->loaderModList()->installResourceWithFlameMetadata(localFileName, version);
+ break;
+ case ModPlatform::ResourceType::ShaderPack:
+ minecraftInst->shaderPackList()->installResourceWithFlameMetadata(localFileName, version);
+ break;
+ case ModPlatform::ResourceType::World: minecraftInst->worldList()->installWorld(localFileInfo); break;
+ case ModPlatform::ResourceType::Unknown:
+ default: qDebug() << "Can't Identify" << localFileName << "Ignoring it."; break;
+ }
+ }
+}
+
+void MainWindow::on_actionREDDIT_triggered()
+{
+ openLauncherHub(QUrl(BuildConfig.SUBREDDIT_URL));
+}
+
+void MainWindow::on_actionDISCORD_triggered()
+{
+ openLauncherHub(QUrl(BuildConfig.DISCORD_URL));
+}
+
+void MainWindow::on_actionMATRIX_triggered()
+{
+ openLauncherHub(QUrl(BuildConfig.MATRIX_URL));
+}
+
+void MainWindow::on_actionChangeInstIcon_triggered()
+{
+ if (!m_selectedInstance)
+ return;
+
+ IconPickerDialog dlg(this);
+ dlg.execWithSelection(m_selectedInstance->iconKey());
+ if (dlg.result() == QDialog::Accepted)
+ {
+ m_selectedInstance->setIconKey(dlg.selectedIconKey);
+ auto icon = APPLICATION->icons()->getIcon(dlg.selectedIconKey);
+ ui->actionChangeInstIcon->setIcon(icon);
+ changeIconButton->setIcon(icon);
+ }
+}
+
+void MainWindow::iconUpdated(QString icon)
+{
+ if (icon == m_currentInstIcon)
+ {
+ auto new_icon = APPLICATION->icons()->getIcon(m_currentInstIcon);
+ ui->actionChangeInstIcon->setIcon(new_icon);
+ changeIconButton->setIcon(new_icon);
+ }
+}
+
+void MainWindow::updateInstanceToolIcon(QString new_icon)
+{
+ m_currentInstIcon = new_icon;
+ auto icon = APPLICATION->icons()->getIcon(m_currentInstIcon);
+ ui->actionChangeInstIcon->setIcon(icon);
+ changeIconButton->setIcon(icon);
+}
+
+void MainWindow::setSelectedInstanceById(const QString& id)
+{
+ if (id.isNull())
+ return;
+ const QModelIndex index = APPLICATION->instances()->getInstanceIndexById(id);
+ if (index.isValid())
+ {
+ QModelIndex selectionIndex = proxymodel->mapFromSource(index);
+ view->selectionModel()->setCurrentIndex(selectionIndex, QItemSelectionModel::ClearAndSelect);
+ updateStatusCenter();
+ }
+}
+
+void MainWindow::on_actionChangeInstGroup_triggered()
+{
+ if (!m_selectedInstance)
+ return;
+
+ InstanceId instId = m_selectedInstance->id();
+ QString src(APPLICATION->instances()->getInstanceGroup(instId));
+
+ QStringList groups = APPLICATION->instances()->getGroups();
+ groups.prepend("");
+ int index = groups.indexOf(src);
+ bool ok = false;
+ QString dst =
+ QInputDialog::getItem(this, tr("Group name"), tr("Enter a new group name."), groups, index, true, &ok);
+ dst = dst.simplified();
+
+ if (ok)
+ {
+ APPLICATION->instances()->setInstanceGroup(instId, dst);
+ }
+}
+
+void MainWindow::deleteGroup(QString group)
+{
+ Q_ASSERT(!group.isEmpty());
+
+ const int reply = QMessageBox::question(this,
+ tr("Delete group"),
+ tr("Are you sure you want to delete the group '%1'?").arg(group),
+ QMessageBox::Yes | QMessageBox::No);
+ if (reply == QMessageBox::Yes)
+ APPLICATION->instances()->deleteGroup(group);
+}
+
+void MainWindow::renameGroup(QString group)
+{
+ Q_ASSERT(!group.isEmpty());
+
+ QString name =
+ QInputDialog::getText(this, tr("Rename group"), tr("Enter a new group name."), QLineEdit::Normal, group);
+ name = name.simplified();
+ if (name.isNull() || name == group)
+ return;
+
+ const bool empty = name.isEmpty();
+ const bool duplicate =
+ APPLICATION->instances()->getGroups().contains(name, Qt::CaseInsensitive) && group.toLower() != name.toLower();
+
+ if (empty || duplicate)
+ {
+ QMessageBox::warning(this,
+ tr("Cannot rename group"),
+ empty ? tr("Cannot set empty name.") : tr("Group already exists. :/"));
+ return;
+ }
+
+ APPLICATION->instances()->renameGroup(group, name);
+}
+
+void MainWindow::undoTrashInstance()
+{
+ if (!APPLICATION->instances()->undoTrashInstance())
+ QMessageBox::warning(this,
+ tr("Failed to undo trashing instance"),
+ tr("Some instances and shortcuts could not be restored.\nPlease check your trashbin to "
+ "manually restore them."));
+ ui->actionUndoTrashInstance->setEnabled(APPLICATION->instances()->trashedSomething());
+}
+
+void MainWindow::on_actionViewLauncherRootFolder_triggered()
+{
+ DesktopServices::openPath(".");
+}
+
+void MainWindow::on_actionViewInstanceFolder_triggered()
+{
+ QString str = APPLICATION->settings()->get("InstanceDir").toString();
+ DesktopServices::openPath(str);
+}
+
+void MainWindow::on_actionViewCentralModsFolder_triggered()
+{
+ DesktopServices::openPath(APPLICATION->settings()->get("CentralModsDir").toString(), true);
+}
+
+void MainWindow::on_actionViewSkinsFolder_triggered()
+{
+ DesktopServices::openPath(APPLICATION->settings()->get("SkinsDir").toString(), true);
+}
+
+void MainWindow::on_actionViewIconThemeFolder_triggered()
+{
+ DesktopServices::openPath(APPLICATION->themeManager()->getIconThemesFolder().path(), true);
+}
+
+void MainWindow::on_actionViewWidgetThemeFolder_triggered()
+{
+ DesktopServices::openPath(APPLICATION->themeManager()->getApplicationThemesFolder().path(), true);
+}
+
+void MainWindow::on_actionViewCatPackFolder_triggered()
+{
+ DesktopServices::openPath(APPLICATION->themeManager()->getCatPacksFolder().path(), true);
+}
+
+void MainWindow::on_actionViewIconsFolder_triggered()
+{
+ DesktopServices::openPath(APPLICATION->icons()->getDirectory(), true);
+}
+
+void MainWindow::on_actionViewLogsFolder_triggered()
+{
+ DesktopServices::openPath("logs", true);
+}
+
+void MainWindow::on_actionViewJavaFolder_triggered()
+{
+ DesktopServices::openPath(APPLICATION->javaPath(), true);
+}
+
+void MainWindow::refreshInstances()
+{
+ APPLICATION->instances()->loadList();
+}
+
+void MainWindow::checkForUpdates()
+{
+ if (APPLICATION->updaterEnabled())
+ {
+ APPLICATION->triggerUpdateCheck();
+ }
+ else
+ {
+ qWarning() << "Updater not set up. Cannot check for updates.";
+ }
+}
+
+void MainWindow::on_actionSettings_triggered()
+{
+ APPLICATION->ShowGlobalSettings(this, "global-settings");
+}
+
+void MainWindow::globalSettingsClosed()
+{
+ // Batch UI updates to prevent multiple layout recalculations
+ setUpdatesEnabled(false);
+
+ // Only reload instances if instance-related settings changed
+ // (instance directory, group settings, etc.)
+ APPLICATION->instances()->loadList();
+ proxymodel->invalidate();
+ proxymodel->sort(0);
+
+ // Update UI components that depend on settings
+ updateMainToolBar();
+ updateLaunchButton();
+ updateThemeMenu();
+ updateStatusCenter();
+
+ // Persist window state to prevent loss on abnormal exit
+ APPLICATION->settings()->set("MainWindowState", QString::fromUtf8(saveState().toBase64()));
+
+ setUpdatesEnabled(true);
+ update();
+}
+
+void MainWindow::on_actionEditInstance_triggered()
+{
+ if (!m_selectedInstance)
+ return;
+
+ if (m_selectedInstance->canEdit())
+ {
+ APPLICATION->showInstanceWindow(m_selectedInstance);
+ }
+ else
+ {
+ CustomMessageBox::selectable(
+ this,
+ tr("Instance not editable"),
+ tr("This instance is not editable. It may be broken, invalid, or too old. Check logs for details."),
+ QMessageBox::Critical)
+ ->show();
+ }
+}
+
+void MainWindow::on_actionManageAccounts_triggered()
+{
+ APPLICATION->ShowGlobalSettings(this, "accounts");
+}
+
+void MainWindow::on_actionReportBug_triggered()
+{
+ DesktopServices::openUrl(QUrl(BuildConfig.BUG_TRACKER_URL));
+}
+
+void MainWindow::on_actionClearMetadata_triggered()
+{
+ // This if contains side effects!
+ if (!APPLICATION->metacache()->evictAll())
+ {
+ CustomMessageBox::selectable(
+ this,
+ tr("Error"),
+ tr("Metadata cache clear Failed!\nTo clear the metadata cache manually, press Folders -> View "
+ "Launcher Root Folder, and after closing the launcher delete the folder named \"meta\"\n"),
+ QMessageBox::Warning)
+ ->show();
+ }
+
+ APPLICATION->metacache()->SaveNow();
+}
+
+#ifdef Q_OS_MAC
+void MainWindow::on_actionAddToPATH_triggered()
+{
+ auto binaryPath = APPLICATION->applicationFilePath();
+ auto targetPath = QString("/usr/local/bin/%1").arg(BuildConfig.LAUNCHER_APP_BINARY_NAME);
+ qDebug() << "Symlinking" << binaryPath << "to" << targetPath;
+
+ QStringList args;
+ args << "-e";
+ args << QString("do shell script \"mkdir -p /usr/local/bin && ln -sf '%1' '%2'\" with administrator privileges")
+ .arg(binaryPath, targetPath);
+ auto outcome = QProcess::execute("/usr/bin/osascript", args);
+ if (!outcome)
+ {
+ QMessageBox::information(this,
+ tr("Successfully added %1 to PATH").arg(BuildConfig.LAUNCHER_DISPLAYNAME),
+ tr("%1 was successfully added to your PATH. You can now start it by running `%2`.")
+ .arg(BuildConfig.LAUNCHER_DISPLAYNAME, BuildConfig.LAUNCHER_APP_BINARY_NAME));
+ }
+ else
+ {
+ QMessageBox::critical(
+ this,
+ tr("Failed to add %1 to PATH").arg(BuildConfig.LAUNCHER_DISPLAYNAME),
+ tr("An error occurred while trying to add %1 to PATH").arg(BuildConfig.LAUNCHER_DISPLAYNAME));
+ }
+}
+#endif
+
+void MainWindow::on_actionOpenWiki_triggered()
+{
+ openLauncherHub(QUrl(BuildConfig.WIKI_URL));
+}
+
+void MainWindow::on_actionMoreNews_triggered()
+{
+ auto entries = m_newsChecker->getNewsEntries();
+ NewsDialog news_dialog(entries, this);
+ connect(&news_dialog, &NewsDialog::openHubRequested, this, &MainWindow::openLauncherHub);
+ news_dialog.exec();
+}
+
+void MainWindow::on_actionLauncherHub_triggered()
+{
+ if (isLauncherHubVisible())
+ {
+ showMainContent();
+ return;
+ }
+ openLauncherHub();
+}
+
+void MainWindow::newsButtonClicked()
+{
+ auto entries = m_newsChecker->getNewsEntries();
+ NewsDialog news_dialog(entries, this);
+ connect(&news_dialog, &NewsDialog::openHubRequested, this, &MainWindow::openLauncherHub);
+ news_dialog.toggleArticleList();
+ news_dialog.exec();
+}
+
+void MainWindow::showMainContent()
+{
+ if (!m_contentStack || !m_instancesPage)
+ {
+ return;
+ }
+ m_contentStack->setCurrentWidget(m_instancesPage);
+}
+
+void MainWindow::ensureLauncherHubPage()
+{
+#if defined(PROJT_DISABLE_LAUNCHER_HUB)
+ return;
+#else
+ if (!m_contentStack || !m_launcherHubWidget)
+ {
+ return;
+ }
+
+ m_launcherHubWidget->setSelectedInstanceId(m_selectedInstance ? m_selectedInstance->id() : QString());
+ m_launcherHubWidget->refreshCockpit();
+ m_contentStack->setCurrentWidget(m_launcherHubWidget);
+#endif
+}
+
+bool MainWindow::isLauncherHubVisible() const
+{
+ return m_contentStack && m_launcherHubWidget && m_contentStack->currentWidget() == m_launcherHubWidget;
+}
+
+void MainWindow::openLauncherHub(const QUrl& url)
+{
+#if defined(PROJT_DISABLE_LAUNCHER_HUB)
+ QMessageBox::information(this, tr("Launcher Hub"), tr("Launcher Hub is not available in this version."));
+ if (url.isValid())
+ {
+ QDesktopServices::openUrl(url);
+ }
+ return;
+#endif
+
+ ensureLauncherHubPage();
+ if (!m_launcherHubWidget)
+ {
+ return;
+ }
+ if (url.isValid())
+ {
+ m_launcherHubWidget->openUrl(url);
+ }
+}
+
+void MainWindow::onCatChanged(int)
+{
+ setCatBackground(APPLICATION->settings()->get("TheCat").toBool());
+}
+
+void MainWindow::on_actionAbout_triggered()
+{
+ AboutDialog dialog(this);
+ dialog.exec();
+}
+
+void MainWindow::on_actionDeleteInstance_triggered()
+{
+ if (!m_selectedInstance)
+ {
+ return;
+ }
+
+ if (m_selectedInstance->isRunning())
+ {
+ CustomMessageBox::selectable(
+ this,
+ tr("Cannot Delete Running Instance"),
+ tr("The selected instance is currently running and cannot be deleted. Please stop the instance before "
+ "attempting to delete it."),
+ QMessageBox::Warning,
+ QMessageBox::Ok)
+ ->exec();
+ return;
+ }
+ auto id = m_selectedInstance->id();
+
+ QString shortcutStr;
+ auto shortcuts = m_selectedInstance->shortcuts();
+ if (!shortcuts.isEmpty())
+ shortcutStr = tr(" and its %n registered shortcut(s)", "", shortcuts.size());
+ auto response = CustomMessageBox::selectable(this,
+ tr("Confirm Deletion"),
+ tr("You are about to delete \"%1\"%2.\n"
+ "This may be permanent and will completely delete the instance.\n\n"
+ "Are you sure?")
+ .arg(m_selectedInstance->name(), shortcutStr),
+ QMessageBox::Warning,
+ QMessageBox::Yes | QMessageBox::No,
+ QMessageBox::No)
+ ->exec();
+
+ if (response != QMessageBox::Yes)
+ return;
+
+ if (!checkLinkedInstances(id, this, tr("Deleting")))
+ return;
+
+ if (APPLICATION->instances()->trashInstance(id))
+ {
+ ui->actionUndoTrashInstance->setEnabled(APPLICATION->instances()->trashedSomething());
+ }
+ else
+ {
+ APPLICATION->instances()->deleteInstance(id);
+ }
+ APPLICATION->settings()->set("SelectedInstance", QString());
+ selectionBad();
+}
+
+void MainWindow::on_actionExportInstanceZip_triggered()
+{
+ if (m_selectedInstance)
+ {
+ ExportInstanceDialog dlg(m_selectedInstance, this);
+ dlg.exec();
+ }
+}
+
+void MainWindow::on_actionExportInstanceMrPack_triggered()
+{
+ if (m_selectedInstance)
+ {
+ auto instance = std::dynamic_pointer_cast<MinecraftInstance>(m_selectedInstance);
+ if (instance != nullptr)
+ {
+ ExportPackDialog dlg(instance, this);
+ dlg.exec();
+ }
+ }
+}
+
+void MainWindow::on_actionExportInstanceFlamePack_triggered()
+{
+ if (m_selectedInstance)
+ {
+ auto instance = std::dynamic_pointer_cast<MinecraftInstance>(m_selectedInstance);
+ if (instance)
+ {
+ if (auto cmp = instance->getPackProfile()->getComponent("net.minecraft");
+ cmp && cmp->getVersionFile() && cmp->getVersionFile()->type == "snapshot")
+ {
+ QMessageBox msgBox(this);
+ msgBox.setText("Snapshots are currently not supported by CurseForge modpacks.");
+ msgBox.exec();
+ return;
+ }
+ ExportPackDialog dlg(instance, this, ModPlatform::ResourceProvider::FLAME);
+ dlg.exec();
+ }
+ }
+}
+
+void MainWindow::on_actionManageBackups_triggered()
+{
+ if (m_selectedInstance)
+ {
+ APPLICATION->showInstanceWindow(m_selectedInstance, "backups");
+ }
+}
+
+void MainWindow::on_actionRenameInstance_triggered()
+{
+ if (m_selectedInstance)
+ {
+ view->edit(view->currentIndex());
+ }
+}
+
+void MainWindow::on_actionViewSelectedInstFolder_triggered()
+{
+ if (m_selectedInstance)
+ {
+ QString str = m_selectedInstance->instanceRoot();
+ DesktopServices::openPath(QFileInfo(str));
+ }
+}
+
+void MainWindow::closeEvent(QCloseEvent* event)
+{
+ // Save the window state and geometry.
+ APPLICATION->settings()->set("MainWindowState", QString::fromUtf8(saveState().toBase64()));
+ APPLICATION->settings()->set("MainWindowGeometry", QString::fromUtf8(saveGeometry().toBase64()));
+ instanceToolbarSetting->set(QString::fromUtf8(ui->instanceToolBar->getVisibilityState().toBase64()));
+ event->accept();
+ emit isClosing();
+}
+
+void MainWindow::changeEvent(QEvent* event)
+{
+ if (event->type() == QEvent::LanguageChange)
+ {
+ retranslateUi();
+ }
+ QMainWindow::changeEvent(event);
+}
+
+void MainWindow::instanceActivated(QModelIndex index)
+{
+ if (!index.isValid())
+ return;
+ QString id = index.data(InstanceList::InstanceIDRole).toString();
+ InstancePtr inst = APPLICATION->instances()->getInstanceById(id);
+ if (!inst)
+ return;
+
+ activateInstance(inst);
+}
+
+void MainWindow::on_actionLaunchInstance_triggered()
+{
+ if (m_selectedInstance && !m_selectedInstance->isRunning())
+ {
+ APPLICATION->launch(m_selectedInstance);
+ }
+}
+
+void MainWindow::activateInstance(InstancePtr instance)
+{
+ APPLICATION->launch(instance);
+}
+
+void MainWindow::on_actionKillInstance_triggered()
+{
+ if (m_selectedInstance && m_selectedInstance->isRunning())
+ {
+ APPLICATION->kill(m_selectedInstance);
+ }
+}
+
+void MainWindow::on_actionCreateInstanceShortcut_triggered()
+{
+ if (!m_selectedInstance)
+ return;
+
+ CreateShortcutDialog shortcutDlg(m_selectedInstance, this);
+ if (!shortcutDlg.exec())
+ return;
+ shortcutDlg.createShortcut();
+}
+
+void MainWindow::taskEnd()
+{
+ QObject* sender = QObject::sender();
+ if (sender == m_versionLoadTask)
+ m_versionLoadTask = NULL;
+
+ sender->deleteLater();
+}
+
+void MainWindow::startTask(Task* task)
+{
+ connect(task, &Task::succeeded, this, &MainWindow::taskEnd);
+ connect(task, &Task::failed, this, &MainWindow::taskEnd);
+ task->start();
+}
+
+void MainWindow::instanceChanged(const QModelIndex& current, [[maybe_unused]] const QModelIndex& previous)
+{
+ if (!current.isValid())
+ {
+ APPLICATION->settings()->set("SelectedInstance", QString());
+ selectionBad();
+ return;
+ }
+ if (m_selectedInstance)
+ {
+ disconnect(m_selectedInstance.get(),
+ &BaseInstance::runningStatusChanged,
+ this,
+ &MainWindow::refreshCurrentInstance);
+ disconnect(m_selectedInstance.get(), &BaseInstance::profilerChanged, this, &MainWindow::refreshCurrentInstance);
+ }
+ QString id = current.data(InstanceList::InstanceIDRole).toString();
+ m_selectedInstance = APPLICATION->instances()->getInstanceById(id);
+ if (m_selectedInstance)
+ {
+ ui->instanceToolBar->setEnabled(true);
+ setInstanceActionsEnabled(true);
+ ui->actionLaunchInstance->setEnabled(m_selectedInstance->canLaunch());
+
+ ui->actionKillInstance->setEnabled(m_selectedInstance->isRunning());
+ ui->actionExportInstance->setEnabled(m_selectedInstance->canExport());
+ renameButton->setText(m_selectedInstance->name());
+ m_statusLeft->setText(m_selectedInstance->getStatusbarDescription());
+ updateStatusCenter();
+ updateInstanceToolIcon(m_selectedInstance->iconKey());
+
+ updateLaunchButton();
+
+ APPLICATION->settings()->set("SelectedInstance", m_selectedInstance->id());
+
+ connect(m_selectedInstance.get(),
+ &BaseInstance::runningStatusChanged,
+ this,
+ &MainWindow::refreshCurrentInstance);
+ connect(m_selectedInstance.get(), &BaseInstance::profilerChanged, this, &MainWindow::refreshCurrentInstance);
+ if (m_launcherHubWidget)
+ {
+ m_launcherHubWidget->setSelectedInstanceId(m_selectedInstance->id());
+ m_launcherHubWidget->refreshCockpit();
+ }
+ }
+ else
+ {
+ APPLICATION->settings()->set("SelectedInstance", QString());
+ selectionBad();
+ return;
+ }
+}
+
+void MainWindow::instanceSelectRequest(QString id)
+{
+ setSelectedInstanceById(id);
+}
+
+void MainWindow::instanceDataChanged(const QModelIndex& topLeft, const QModelIndex& bottomRight)
+{
+ auto current = view->selectionModel()->currentIndex();
+ QItemSelection test(topLeft, bottomRight);
+ if (test.contains(current))
+ {
+ instanceChanged(current, current);
+ }
+}
+
+void MainWindow::selectionBad()
+{
+ // start by reseting everything...
+ m_selectedInstance = nullptr;
+ m_statusLeft->setText(tr("No instance selected"));
+
+ statusBar()->clearMessage();
+ ui->instanceToolBar->setEnabled(false);
+ setInstanceActionsEnabled(false);
+ updateLaunchButton();
+ renameButton->setText(tr("Rename Instance"));
+ updateInstanceToolIcon("grass");
+ if (m_launcherHubWidget)
+ {
+ m_launcherHubWidget->setSelectedInstanceId(QString());
+ m_launcherHubWidget->refreshCockpit();
+ }
+
+ // ...and then see if we can enable the previously selected instance
+ setSelectedInstanceById(APPLICATION->settings()->get("SelectedInstance").toString());
+}
+
+void MainWindow::checkInstancePathForProblems()
+{
+ QString instanceFolder = APPLICATION->settings()->get("InstanceDir").toString();
+ if (FS::checkProblemticPathJava(QDir(instanceFolder)))
+ {
+ QMessageBox warning(this);
+ warning.setText(tr("Your instance folder contains \'!\' and this is known to cause Java problems!"));
+ warning.setInformativeText(tr("You have now two options: <br/>"
+ " - change the instance folder in the settings <br/>"
+ " - move this installation of %1 to a different folder")
+ .arg(BuildConfig.LAUNCHER_DISPLAYNAME));
+ warning.setDefaultButton(QMessageBox::Ok);
+ warning.exec();
+ }
+ auto tempFolderText = tr("This is a problem: <br/>"
+ " - The launcher will likely be deleted without warning by the operating system <br/>"
+ " - close the launcher now and extract it to a real location, not a temporary folder");
+ QString pathfoldername = QDir(instanceFolder).absolutePath();
+ if (pathfoldername.contains("Rar$", Qt::CaseInsensitive))
+ {
+ QMessageBox warning(this);
+ warning.setText(
+ tr("Your instance folder contains \'Rar$\' - that means you haven't extracted the launcher archive!"));
+ warning.setInformativeText(tempFolderText);
+ warning.setDefaultButton(QMessageBox::Ok);
+ warning.exec();
+ }
+ else if (pathfoldername.startsWith(QDir::tempPath()) || pathfoldername.contains("/TempState/"))
+ {
+ QMessageBox warning(this);
+ warning.setText(tr("Your instance folder is in a temporary folder: \'%1\'!").arg(QDir::tempPath()));
+ warning.setInformativeText(tempFolderText);
+ warning.setDefaultButton(QMessageBox::Ok);
+ warning.exec();
+ }
+}
+
+void MainWindow::updateStatusCenter()
+{
+ m_statusCenter->setVisible(APPLICATION->settings()->get("ShowGlobalGameTime").toBool());
+
+ int timePlayed = APPLICATION->instances()->getTotalPlayTime();
+ if (timePlayed > 0)
+ {
+ m_statusCenter->setText(
+ tr("Total playtime: %1")
+ .arg(Time::prettifyDuration(timePlayed,
+ APPLICATION->settings()->get("ShowGameTimeWithoutDays").toBool())));
+ }
+}
+// "Instance actions" are actions that require an instance to be selected (i.e. "new instance" is not here)
+// Actions that also require other conditions (e.g. a running instance) won't be changed.
+void MainWindow::setInstanceActionsEnabled(bool enabled)
+{
+ ui->actionEditInstance->setEnabled(enabled);
+ ui->actionChangeInstGroup->setEnabled(enabled);
+ ui->actionViewSelectedInstFolder->setEnabled(enabled);
+ ui->actionExportInstance->setEnabled(enabled);
+ ui->actionManageBackups->setEnabled(enabled);
+ ui->actionDeleteInstance->setEnabled(enabled);
+ ui->actionCopyInstance->setEnabled(enabled);
+ ui->actionCreateInstanceShortcut->setEnabled(enabled);
+}
+
+void MainWindow::refreshCurrentInstance()
+{
+ auto current = view->selectionModel()->currentIndex();
+ instanceChanged(current, current);
+}