mainwindow_moc.cpp 9.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358
  1. /*
  2. * mainwindow_moc.cpp, part of VCMI engine
  3. *
  4. * Authors: listed in file AUTHORS in main folder
  5. *
  6. * License: GNU General Public License v2.0 or later
  7. * Full text of license available in license.txt file, in main folder
  8. *
  9. */
  10. #include "StdInc.h"
  11. #include "mainwindow_moc.h"
  12. #include "ui_mainwindow_moc.h"
  13. #include <QDir>
  14. #include "../lib/CConfigHandler.h"
  15. #include "../lib/VCMIDirs.h"
  16. #include "../lib/filesystem/Filesystem.h"
  17. #include "../lib/logging/CBasicLogConfigurator.h"
  18. #include "../lib/texts/Languages.h"
  19. #include "../lib/ExceptionsCommon.h"
  20. #include "updatedialog_moc.h"
  21. #include "main.h"
  22. #include "helper.h"
  23. void MainWindow::load()
  24. {
  25. // Set current working dir to executable folder.
  26. // This is important on Mac for relative paths to work inside DMG.
  27. QDir::setCurrent(QApplication::applicationDirPath());
  28. #ifndef VCMI_MOBILE
  29. console = new CConsoleHandler();
  30. #endif
  31. CBasicLogConfigurator logConfig(VCMIDirs::get().userLogsPath() / "VCMI_Launcher_log.txt", console);
  32. logConfig.configureDefault();
  33. try
  34. {
  35. CResourceHandler::initialize();
  36. CResourceHandler::load("config/filesystem.json");
  37. }
  38. catch (const DataLoadingException & e)
  39. {
  40. QMessageBox::critical(this, tr("Error starting executable"), QString::fromStdString(e.what()));
  41. }
  42. Helper::loadSettings();
  43. }
  44. void MainWindow::computeSidePanelSizes()
  45. {
  46. QVector<QToolButton*> widgets = {
  47. ui->modslistButton,
  48. ui->settingsButton,
  49. ui->aboutButton,
  50. ui->startGameButton
  51. };
  52. for(auto & widget : widgets)
  53. {
  54. QFontMetrics metrics(widget->font());
  55. QSize iconSize = widget->iconSize();
  56. // this is minimal space that is needed for our button to avoid text clipping
  57. int buttonHeight = iconSize.height() + metrics.height() + 4;
  58. widget->setMinimumHeight(buttonHeight);
  59. widget->setMaximumHeight(buttonHeight * 1.2);
  60. }
  61. }
  62. MainWindow::MainWindow(QWidget * parent)
  63. : QMainWindow(parent), ui(new Ui::MainWindow)
  64. {
  65. load(); // load FS before UI
  66. bool setupCompleted = settings["launcher"]["setupCompleted"].Bool();
  67. if (!setupCompleted)
  68. detectPreferredLanguage();
  69. updateTranslation(); // load translation
  70. ui->setupUi(this);
  71. setAcceptDrops(true);
  72. setWindowIcon(QIcon{":/icons/menu-game.png"});
  73. ui->modslistButton->setIcon(QIcon{":/icons/menu-mods.png"});
  74. ui->settingsButton->setIcon(QIcon{":/icons/menu-settings.png"});
  75. ui->aboutButton->setIcon(QIcon{":/icons/about-project.png"});
  76. ui->startGameButton->setIcon(QIcon{":/icons/menu-game.png"});
  77. #ifndef VCMI_MOBILE
  78. //load window settings
  79. QSettings s(Ui::teamName, Ui::appName);
  80. auto size = s.value("MainWindow/Size").toSize();
  81. if(size.isValid())
  82. {
  83. resize(size);
  84. }
  85. auto position = s.value("MainWindow/Position").toPoint();
  86. if(!position.isNull())
  87. {
  88. move(position);
  89. }
  90. #endif
  91. computeSidePanelSizes();
  92. bool h3DataFound = CResourceHandler::get()->existsResource(ResourcePath("DATA/GENRLTXT.TXT"));
  93. if (h3DataFound && setupCompleted)
  94. ui->tabListWidget->setCurrentIndex(TabRows::START);
  95. else
  96. enterSetup();
  97. ui->settingsView->setDisplayList();
  98. if(settings["launcher"]["updateOnStartup"].Bool())
  99. UpdateDialog::showUpdateDialog(false);
  100. }
  101. void MainWindow::detectPreferredLanguage()
  102. {
  103. auto preferredLanguages = QLocale::system().uiLanguages();
  104. std::string selectedLanguage;
  105. for (auto const & userLang : preferredLanguages)
  106. {
  107. logGlobal->info("Preferred language: %s", userLang.toStdString());
  108. for (auto const & vcmiLang : Languages::getLanguageList())
  109. if (vcmiLang.tagIETF == userLang.toStdString())
  110. selectedLanguage = vcmiLang.identifier;
  111. if (!selectedLanguage.empty())
  112. {
  113. logGlobal->info("Selected language: %s", selectedLanguage);
  114. Settings node = settings.write["general"]["language"];
  115. node->String() = selectedLanguage;
  116. return;
  117. }
  118. }
  119. }
  120. void MainWindow::enterSetup()
  121. {
  122. ui->startGameButton->setEnabled(false);
  123. ui->settingsButton->setEnabled(false);
  124. ui->aboutButton->setEnabled(false);
  125. ui->modslistButton->setEnabled(false);
  126. ui->tabListWidget->setCurrentIndex(TabRows::SETUP);
  127. }
  128. void MainWindow::exitSetup()
  129. {
  130. Settings writer = settings.write["launcher"]["setupCompleted"];
  131. writer->Bool() = true;
  132. ui->startGameButton->setEnabled(true);
  133. ui->settingsButton->setEnabled(true);
  134. ui->aboutButton->setEnabled(true);
  135. ui->modslistButton->setEnabled(true);
  136. ui->tabListWidget->setCurrentIndex(TabRows::MODS);
  137. }
  138. void MainWindow::switchToStartTab()
  139. {
  140. ui->startGameButton->setEnabled(true);
  141. ui->startGameButton->setChecked(true);
  142. ui->tabListWidget->setCurrentIndex(TabRows::START);
  143. auto* startGameTabWidget = qobject_cast<StartGameTab*>(ui->tabListWidget->widget(TabRows::START));
  144. if(startGameTabWidget)
  145. startGameTabWidget->refreshState();
  146. }
  147. void MainWindow::switchToModsTab()
  148. {
  149. ui->startGameButton->setEnabled(true);
  150. ui->modslistButton->setChecked(true);
  151. ui->tabListWidget->setCurrentIndex(TabRows::MODS);
  152. }
  153. void MainWindow::changeEvent(QEvent * event)
  154. {
  155. if(event->type() == QEvent::LanguageChange)
  156. {
  157. ui->retranslateUi(this);
  158. }
  159. QMainWindow::changeEvent(event);
  160. }
  161. MainWindow::~MainWindow()
  162. {
  163. #ifndef VCMI_MOBILE
  164. //save window settings
  165. QSettings s(Ui::teamName, Ui::appName);
  166. s.setValue("MainWindow/Size", size());
  167. s.setValue("MainWindow/Position", pos());
  168. #endif
  169. delete ui;
  170. }
  171. void MainWindow::on_startGameButton_clicked()
  172. {
  173. switchToStartTab();
  174. }
  175. CModListView * MainWindow::getModView()
  176. {
  177. return ui->modlistView;
  178. }
  179. void MainWindow::on_modslistButton_clicked()
  180. {
  181. switchToModsTab();
  182. }
  183. void MainWindow::on_settingsButton_clicked()
  184. {
  185. ui->startGameButton->setEnabled(true);
  186. ui->tabListWidget->setCurrentIndex(TabRows::SETTINGS);
  187. }
  188. void MainWindow::on_aboutButton_clicked()
  189. {
  190. ui->startGameButton->setEnabled(true);
  191. ui->tabListWidget->setCurrentIndex(TabRows::ABOUT);
  192. }
  193. void MainWindow::dragEnterEvent(QDragEnterEvent* event)
  194. {
  195. if(event->mimeData()->hasUrls())
  196. for(const auto & url : event->mimeData()->urls())
  197. for(const auto & ending : QStringList({".zip", ".h3m", ".h3c", ".vmap", ".vcmp", ".json", ".exe"}))
  198. if(url.fileName().endsWith(ending, Qt::CaseInsensitive))
  199. {
  200. event->acceptProposedAction();
  201. return;
  202. }
  203. }
  204. void MainWindow::dropEvent(QDropEvent* event)
  205. {
  206. const QMimeData* mimeData = event->mimeData();
  207. if(mimeData->hasUrls())
  208. {
  209. const QList<QUrl> urlList = mimeData->urls();
  210. for (const auto & url : urlList)
  211. manualInstallFile(url.toLocalFile());
  212. }
  213. }
  214. void MainWindow::manualInstallFile(QString filePath)
  215. {
  216. if(filePath.endsWith(".zip", Qt::CaseInsensitive) || filePath.endsWith(".exe", Qt::CaseInsensitive))
  217. switchToModsTab();
  218. QString fileName = QFileInfo{filePath}.fileName();
  219. if(filePath.endsWith(".zip", Qt::CaseInsensitive))
  220. {
  221. QString filenameClean = fileName.toLower()
  222. // mod name currently comes from zip file -> remove suffixes from github zip download
  223. .replace(QRegularExpression("-[0-9a-f]{40}"), "")
  224. .replace(QRegularExpression("-vcmi-.+\\.zip"), ".zip")
  225. .replace("-main.zip", ".zip");
  226. getModView()->downloadFile(filenameClean, QUrl::fromLocalFile(filePath), "mods");
  227. }
  228. else if(filePath.endsWith(".json", Qt::CaseInsensitive))
  229. {
  230. QDir configDir(QString::fromStdString(VCMIDirs::get().userConfigPath().string()));
  231. QStringList configFile = configDir.entryList({fileName}, QDir::Filter::Files); // case insensitive check
  232. if(!configFile.empty())
  233. {
  234. auto dialogResult = QMessageBox::warning(this, tr("Replace config file?"), tr("Do you want to replace %1?").arg(configFile[0]), QMessageBox::Yes | QMessageBox::No, QMessageBox::No);
  235. if(dialogResult == QMessageBox::Yes)
  236. {
  237. const auto configFilePath = configDir.filePath(configFile[0]);
  238. QFile::remove(configFilePath);
  239. QFile::copy(filePath, configFilePath);
  240. // reload settings
  241. Helper::loadSettings();
  242. for(const auto widget : qApp->allWidgets())
  243. if(auto settingsView = qobject_cast<CSettingsView *>(widget))
  244. settingsView->loadSettings();
  245. getModView()->reload();
  246. }
  247. }
  248. }
  249. else
  250. getModView()->installFiles(QStringList{filePath});
  251. }
  252. ETranslationStatus MainWindow::getTranslationStatus()
  253. {
  254. QString preferredlanguage = QString::fromStdString(settings["general"]["language"].String());
  255. QString installedlanguage = QString::fromStdString(settings["session"]["language"].String());
  256. if (preferredlanguage == installedlanguage)
  257. return ETranslationStatus::ACTIVE;
  258. QString modName = getModView()->getTranslationModName(preferredlanguage);
  259. if (modName.isEmpty())
  260. return ETranslationStatus::NOT_AVAILABLE;
  261. if (!getModView()->isModInstalled(modName))
  262. return ETranslationStatus::NOT_INSTALLLED;
  263. if (!getModView()->isModEnabled(modName))
  264. return ETranslationStatus::DISABLED;
  265. return ETranslationStatus::ACTIVE;
  266. }
  267. void MainWindow::updateTranslation()
  268. {
  269. #ifdef ENABLE_QT_TRANSLATIONS
  270. const std::string translationFile = settings["general"]["language"].String()+ ".qm";
  271. QString translationFileResourcePath = QString{":/translation/%1"}.arg(translationFile.c_str());
  272. logGlobal->info("Loading translation %s", translationFile);
  273. if(!QFile::exists(translationFileResourcePath))
  274. {
  275. logGlobal->debug("Translation file %s does not exist", translationFileResourcePath.toStdString());
  276. return;
  277. }
  278. if (!translator.load(translationFileResourcePath))
  279. {
  280. logGlobal->error("Failed to load translation file %s", translationFileResourcePath.toStdString());
  281. return;
  282. }
  283. if(translationFile == "english.qm")
  284. {
  285. // translator doesn't need to be installed for English
  286. return;
  287. }
  288. if (!qApp->installTranslator(&translator))
  289. {
  290. logGlobal->error("Failed to install translator for translation file %s", translationFileResourcePath.toStdString());
  291. }
  292. #endif
  293. }