graphicspatch.cpp 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340
  1. #include "graphicspatch.h"
  2. #include "LotroDat/LotroDat.h"
  3. #include "LotroDat/Database.h"
  4. #include "models/lotrodatmanager.h"
  5. #include <QUrlQuery>
  6. #include <QSet>
  7. GraphicsPatch::GraphicsPatch(LotroDatManager *mgr, QObject *parent) : Patch("GraphicsPatch", mgr, parent)
  8. {
  9. connect(lotro_mgr_, &LotroDatManager::operationStarted, this, &GraphicsPatch::onLotroDatManagerOperationStarted);
  10. connect(lotro_mgr_, &LotroDatManager::operationFinished, this, &GraphicsPatch::onLotroDatManagerOperationFinished);
  11. connect(lotro_mgr_, &LotroDatManager::statusChanged, this, &GraphicsPatch::onLotroDatManagerStatusChanged);
  12. }
  13. void GraphicsPatch::checkForUpdates()
  14. {
  15. QUrlQuery query; // query for building GET-request aka patch-version
  16. foreach (QString db_name, databases_names) {
  17. query.addQueryItem(db_name, "100");
  18. }
  19. QUrl target_url;
  20. target_url.setUrl(Settings::getValue("Network/patch_updates_url").toString());
  21. target_url.setQuery(query);
  22. QByteArray target_array;
  23. Downloader downloader;
  24. downloader.setUrl(target_url);
  25. downloader.targetBytearray = &target_array;
  26. downloader.start();
  27. downloader.waitForDownloaded();
  28. if (target_array.isEmpty()) {
  29. qWarning() << *this << "Cannot check for updates, target_array is empty!";
  30. emit errorOccured(E_CHECKFORUPDATES, this, "");
  31. emit operationFinished(E_CHECKFORUPDATES, this);
  32. return;
  33. }
  34. QStringList patch_info = QString(target_array).split('|');
  35. if (patch_info.size() != databases_names.size()) {
  36. qCritical() << __FUNCTION__ << "Incorrect patches number! Data: " << patch_info;
  37. emit errorOccured(E_CHECKFORUPDATES, this, "");
  38. emit operationFinished(E_CHECKFORUPDATES, this);
  39. return;
  40. }
  41. for (int i = 0; i < databases_names.size(); ++i) {
  42. QStringList patch_data = patch_info[i].split(":::");
  43. if (patch_data.size() != 3) {
  44. qCritical() << __FUNCTION__ << "Incorrect patch entry size! Entry: " << patch_data;
  45. emit errorOccured(E_CHECKFORUPDATES, this, "");
  46. emit operationFinished(E_CHECKFORUPDATES, this);
  47. return;
  48. }
  49. QString patch_filename = Settings::getValue("General/PatchDownloadDir").toString() + "/" + QUrl(patch_data[0]).fileName();
  50. Settings::setValue("PatchDatabases/" + databases_names[i] + "/url", patch_data[0]);
  51. Settings::setValue("PatchDatabases/" + databases_names[i] + "/hashsum", patch_data[1]);
  52. Settings::setValue("PatchDatabases/" + databases_names[i] + "/datetime", patch_data[2]);
  53. Settings::setValue("PatchDatabases/" + databases_names[i] + "/path", patch_filename);
  54. }
  55. emit operationFinished(E_CHECKFORUPDATES, this);
  56. }
  57. void GraphicsPatch::download()
  58. {
  59. foreach (QString db_name, databases_names) {
  60. QString settings_prefix = "PatchDatabases/" + db_name;
  61. QString target_filename = QApplication::applicationDirPath() + "/" + Settings::getValue(settings_prefix + "/path").toString();
  62. qDebug() << patch_name_ << ": Checking if file " << target_filename << " matches its hashsum";
  63. if (FileSystem::fileHash(target_filename) == Settings::getValue(settings_prefix + "/hashsum").toString()) {
  64. qInfo() << *this << ": file " << target_filename << " is up-to-date, no need to download";
  65. continue;
  66. }
  67. FileSystem::createFilePath(target_filename);
  68. QFile* target_file = new QFile(target_filename);
  69. if (!target_file->open(QIODevice::ReadWrite | QIODevice::Truncate)) {
  70. qWarning() << *this << "Cannot open file " << target_filename;
  71. continue;
  72. }
  73. qInfo() << *this << ": beginning download of file " << target_filename;
  74. Downloader* downloader = new Downloader();
  75. downloader->setUrl(Settings::getValue(settings_prefix + "/url").toUrl());
  76. downloader->targetFile = target_file;
  77. connect(downloader, &Downloader::progressChanged, this, &GraphicsPatch::onDownloaderProgressChanged);
  78. connect(downloader, &Downloader::downloadFinished, this, &GraphicsPatch::onDownloaderFinished);
  79. downloaders_.insert(downloader);
  80. downloader->start();
  81. }
  82. if (downloaders_.empty()) {
  83. emit operationFinished(E_DOWNLOAD, this);
  84. }
  85. }
  86. void GraphicsPatch::install()
  87. {
  88. emit operationFinished(E_INSTALL, this);
  89. return;
  90. installLoadscreens();
  91. foreach (QString db_name, QStringList({"image", "texture"})) {
  92. ++elapsed_patches_to_install_;
  93. QMetaObject::invokeMethod(lotro_mgr_, "installPatch", Qt::QueuedConnection,
  94. Q_ARG(QString, getPatchName() + "_" + db_name),
  95. Q_ARG(QString, Settings::getValue("PatchDatabases/" + db_name + "/path").toString())
  96. );
  97. }
  98. }
  99. void GraphicsPatch::activate()
  100. {
  101. emit operationFinished(E_ACTIVATE, this);
  102. return;
  103. if (Settings::getValue("Components/loadscreens").toBool()) {
  104. enableLoadscreens();
  105. } else {
  106. disableLoadscreens();
  107. }
  108. QString operation_name;
  109. operation_name = Settings::getValue("Components/maps").toBool() ? "enableCategory" : "disableCategory";
  110. ++elapsed_patches_to_install_;
  111. QMetaObject::invokeMethod(lotro_mgr_, operation_name.toLocal8Bit().constData(), Qt::QueuedConnection,
  112. Q_ARG(QString, getPatchName() + "_maps"),
  113. Q_ARG(LotroDatManager::Category, LotroDatManager::Category::E_MAPS_COMMON)
  114. );
  115. operation_name = Settings::getValue("Components/textures").toBool() ? "enableCategory" : "disableCategory";
  116. ++elapsed_patches_to_install_;
  117. QMetaObject::invokeMethod(lotro_mgr_, operation_name.toLocal8Bit().constData(), Qt::QueuedConnection,
  118. Q_ARG(QString, getPatchName() + "_textures"),
  119. Q_ARG(LotroDatManager::Category, LotroDatManager::Category::E_TEXTURES_COMMON)
  120. );
  121. }
  122. void GraphicsPatch::onDownloaderProgressChanged(Downloader *, Downloader::Status)
  123. {
  124. Downloader::Status all_downloads_status;
  125. foreach (Downloader* downloader, downloaders_) {
  126. all_downloads_status = all_downloads_status + downloader->getDownloadStatus();
  127. }
  128. emit progressChanged(OperationProgress(all_downloads_status), this);
  129. }
  130. void GraphicsPatch::onDownloaderFinished(Downloader *ptr)
  131. {
  132. ptr->targetFile->close();
  133. ptr->targetFile->deleteLater();
  134. Downloader::Status all_downloads_status;
  135. for (const Downloader* downloader : downloaders_) {
  136. all_downloads_status = all_downloads_status + downloader->getDownloadStatus();
  137. }
  138. if (!all_downloads_status.running) {
  139. for (Downloader* downloader : downloaders_) {
  140. downloader->deleteLater();
  141. }
  142. downloaders_.clear();
  143. emit operationFinished(E_DOWNLOAD, this);
  144. }
  145. }
  146. void GraphicsPatch::onLotroDatManagerOperationFinished(QString operation_name, QVector<QVariant> args, bool successful)
  147. {
  148. if (args.size() == 0 || !args[0].toString().contains(getPatchName())) {
  149. return; // This means, that message from LotroManager is addressed to another patchset
  150. }
  151. if (operation_name.contains("installPatch")) {
  152. if (!successful) {
  153. qCritical() << *this << "Database " + args[0].toString() + " (" + args[1].toString() + ") was not installed due to Legacy core error.";
  154. emit errorOccured(E_INSTALL, this, "Database " + args[0].toString() + " (" + args[1].toString() + ") was not installed due to Legacy core error.");
  155. }
  156. --elapsed_patches_to_install_;
  157. if (!elapsed_patches_to_install_) {
  158. is_being_patched_by_lotro_dat_manager_ = false;
  159. emit operationFinished(E_INSTALL, this);
  160. }
  161. }
  162. if (operation_name.contains("enableCategory") || operation_name.contains("disableCategory")) {
  163. if (!successful) {
  164. qCritical() << *this << "Error in patch " + args[0].toString() + ": activating category " + QString::number(args[1].toInt()) + " failed.";
  165. emit errorOccured(E_ACTIVATE, this, "Error in patch " + args[0].toString() + ": activating category " + QString::number(args[1].toInt()) + " failed.");
  166. }
  167. --elapsed_patches_to_install_;
  168. if (!elapsed_patches_to_install_) {
  169. is_being_patched_by_lotro_dat_manager_ = false;
  170. emit operationFinished(E_ACTIVATE, this);
  171. }
  172. }
  173. }
  174. void GraphicsPatch::onLotroDatManagerOperationStarted(QString operation_name, QVector<QVariant> args)
  175. {
  176. if (args.size() == 0 || !args[0].toString().contains(getPatchName())) {
  177. return;
  178. }
  179. is_being_patched_by_lotro_dat_manager_ = true;
  180. }
  181. void GraphicsPatch::onLotroDatManagerStatusChanged(LotroDatManager::Status status)
  182. {
  183. if (!is_being_patched_by_lotro_dat_manager_) {
  184. return;
  185. }
  186. emit progressChanged(OperationProgress(status), this);
  187. }
  188. void GraphicsPatch::installLoadscreens()
  189. {
  190. QString locale_prefix = Settings::getValue("Lotro/original_locale").toString();
  191. const QStringList loadscreens_filenames = {
  192. locale_prefix == "English" ? "lotro_ad_pregame.jpg" : "lotro_ad_pregame_" + locale_prefix + ".jpg",
  193. "lotro_generic_teleport_screen_01.jpg",
  194. "lotro_generic_teleport_screen_02.jpg",
  195. "lotro_generic_teleport_screen_03.jpg",
  196. "lotro_generic_teleport_screen_04.jpg",
  197. "lotro_generic_teleport_screen_05.jpg",
  198. "lotro_generic_teleport_screen_06.jpg",
  199. "lotro_generic_teleport_screen_07.jpg",
  200. "lotro_generic_teleport_screen_08.jpg",
  201. "lotro_generic_teleport_screen_09.jpg",
  202. "lotro_generic_teleport_screen_10.jpg"
  203. };
  204. OperationProgress progress;
  205. progress.install_finished_parts = 0;
  206. progress.install_total_parts = loadscreens_filenames.size() * 2;
  207. LOTRO_DAT::Database database;
  208. if (!database.InitDatabase(Settings::getValue("PatchDatabases/Loadscreen/path").toString().toStdString())) {
  209. qCritical() << *this << "database.InitDatabase() of " << Settings::getValue("PatchDatabases/Loadscreen/path").toString() << " FAILED!";
  210. return;
  211. }
  212. LOTRO_DAT::SubfileData data;
  213. QString logo_path = Settings::getValue("Lotro/game_path").toString() + "/raw/" + (locale_prefix == "English" ? "en" : locale_prefix) + "/logo/";
  214. for (size_t i = 0; i < qMin(size_t(loadscreens_filenames.size()), database.CountRows()); ++i) {
  215. QFile::remove(logo_path + loadscreens_filenames[i] + "_ru");
  216. if (!data.binary_data.WriteToFile((logo_path + loadscreens_filenames[i] + "_ru").toStdString())) {
  217. qWarning() << patch_name_ << "Cannot write to file " << logo_path + loadscreens_filenames[i];
  218. }
  219. progress.install_finished_parts++;
  220. emit progressChanged(progress, this);
  221. }
  222. for (int i = 0; i < loadscreens_filenames.size(); ++i) {
  223. QFile::copy(logo_path + loadscreens_filenames[i], logo_path + loadscreens_filenames[i] + "_orig"); // Will not copy if _orig file already exists
  224. progress.install_finished_parts++;
  225. emit progressChanged(progress, this);
  226. }
  227. }
  228. void GraphicsPatch::enableLoadscreens()
  229. {
  230. QString locale_prefix = Settings::getValue("Lotro/original_locale").toString();
  231. QString logo_path = Settings::getValue("Lotro/game_path").toString() + "/raw/" + (locale_prefix == "English" ? "en" : locale_prefix) + "/logo/";
  232. const QStringList loadscreens_filenames = {
  233. locale_prefix == "English" ? "lotro_ad_pregame.jpg" : "lotro_ad_pregame_" + locale_prefix + ".jpg",
  234. "lotro_generic_teleport_screen_01.jpg",
  235. "lotro_generic_teleport_screen_02.jpg",
  236. "lotro_generic_teleport_screen_03.jpg",
  237. "lotro_generic_teleport_screen_04.jpg",
  238. "lotro_generic_teleport_screen_05.jpg",
  239. "lotro_generic_teleport_screen_06.jpg",
  240. "lotro_generic_teleport_screen_07.jpg",
  241. "lotro_generic_teleport_screen_08.jpg",
  242. "lotro_generic_teleport_screen_09.jpg",
  243. "lotro_generic_teleport_screen_10.jpg"
  244. };
  245. OperationProgress progress;
  246. progress.install_total_parts = loadscreens_filenames.size();
  247. for (int i = 0; i < loadscreens_filenames.size(); ++i) {
  248. QFile::remove(logo_path + loadscreens_filenames[i]);
  249. QFile::copy(logo_path + loadscreens_filenames[i] + "_ru", logo_path + loadscreens_filenames[i]);
  250. progress.install_finished_parts++;
  251. emit progressChanged(progress, this);
  252. }
  253. }
  254. void GraphicsPatch::disableLoadscreens()
  255. {
  256. QString locale_prefix = Settings::getValue("Lotro/original_locale").toString();
  257. QString logo_path = Settings::getValue("Lotro/game_path").toString() + "/raw/" + (locale_prefix == "English" ? "en" : locale_prefix) + "/logo/";
  258. const QStringList loadscreens_filenames = {
  259. locale_prefix == "English" ? "lotro_ad_pregame.jpg" : "lotro_ad_pregame_" + locale_prefix + ".jpg",
  260. "lotro_generic_teleport_screen_01.jpg",
  261. "lotro_generic_teleport_screen_02.jpg",
  262. "lotro_generic_teleport_screen_03.jpg",
  263. "lotro_generic_teleport_screen_04.jpg",
  264. "lotro_generic_teleport_screen_05.jpg",
  265. "lotro_generic_teleport_screen_06.jpg",
  266. "lotro_generic_teleport_screen_07.jpg",
  267. "lotro_generic_teleport_screen_08.jpg",
  268. "lotro_generic_teleport_screen_09.jpg",
  269. "lotro_generic_teleport_screen_10.jpg"
  270. };
  271. OperationProgress progress;
  272. progress.install_total_parts = loadscreens_filenames.size();
  273. for (int i = 0; i < loadscreens_filenames.size(); ++i) {
  274. QFile::remove(logo_path + loadscreens_filenames[i]);
  275. QFile::copy(logo_path + loadscreens_filenames[i] + "_orig", logo_path + loadscreens_filenames[i]);
  276. progress.install_finished_parts++;
  277. emit progressChanged(progress, this);
  278. }
  279. }