patchlist.cpp 7.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. #include "patchlist.h"
  2. #include <QDebug>
  3. PatchList::PatchList(LotroDatManager *mgr, QObject *parent) : QObject(parent)
  4. {
  5. lotro_mgr_ = mgr;
  6. qInfo() << "PatchList: Initialising patchsets";
  7. texts_patch_ = new TextsPatch(mgr);
  8. graphics_patch_ = new GraphicsPatch(mgr);
  9. sounds_patch_ = new SoundsPatch(mgr);
  10. videos_patch_ = new VideosPatch(mgr);
  11. qInfo() << "PatchList: Patchsets were initialized, initializing threads";
  12. texts_patch_thread_ = new QThread(this);
  13. graphics_patch_thread_ = new QThread(this);
  14. sounds_patch_thread_ = new QThread(this);
  15. videos_patch_thread_ = new QThread(this);
  16. texts_patch_->moveToThread(texts_patch_thread_);
  17. graphics_patch_->moveToThread(graphics_patch_thread_);
  18. sounds_patch_->moveToThread(sounds_patch_thread_);
  19. videos_patch_->moveToThread(videos_patch_thread_);
  20. connect(texts_patch_, &TextsPatch::operationStarted, this, &PatchList::onPatchOperationStarted, Qt::QueuedConnection);
  21. connect(graphics_patch_, &GraphicsPatch::operationStarted, this, &PatchList::onPatchOperationStarted, Qt::QueuedConnection);
  22. connect(sounds_patch_, &SoundsPatch::operationStarted, this, &PatchList::onPatchOperationStarted, Qt::QueuedConnection);
  23. connect(videos_patch_, &VideosPatch::operationStarted, this, &PatchList::onPatchOperationStarted, Qt::QueuedConnection);
  24. connect(texts_patch_, &TextsPatch::operationFinished, this, &PatchList::onPatchOperationFinished, Qt::QueuedConnection);
  25. connect(graphics_patch_, &GraphicsPatch::operationFinished, this, &PatchList::onPatchOperationFinished, Qt::QueuedConnection);
  26. connect(sounds_patch_, &SoundsPatch::operationFinished, this, &PatchList::onPatchOperationFinished, Qt::QueuedConnection);
  27. connect(videos_patch_, &VideosPatch::operationFinished, this, &PatchList::onPatchOperationFinished, Qt::QueuedConnection);
  28. connect(texts_patch_, &TextsPatch::progressChanged, this, &PatchList::onPatchOperationProgressChanged, Qt::QueuedConnection);
  29. connect(graphics_patch_, &GraphicsPatch::progressChanged, this, &PatchList::onPatchOperationProgressChanged, Qt::QueuedConnection);
  30. connect(sounds_patch_, &SoundsPatch::progressChanged, this, &PatchList::onPatchOperationProgressChanged, Qt::QueuedConnection);
  31. connect(videos_patch_, &VideosPatch::progressChanged, this, &PatchList::onPatchOperationProgressChanged, Qt::QueuedConnection);
  32. connect(lotro_mgr_, &LotroDatManager::operationFinished, this, &PatchList::onLotroManagerOperationFinished, Qt::QueuedConnection);
  33. qInfo() << "PatchList: Patchset threads initialized, starting workers";
  34. texts_patch_thread_->start();
  35. graphics_patch_thread_->start();
  36. sounds_patch_thread_->start();
  37. videos_patch_thread_->start();
  38. patch_update_timer.setInterval(5 * 60 * 1000); // Once in 5 minutes check for updates
  39. connect(&patch_update_timer, &QTimer::timeout, this, &PatchList::update);
  40. }
  41. PatchList::~PatchList()
  42. {
  43. qDebug() << "PatchList: finishing work";
  44. texts_patch_thread_->quit();
  45. graphics_patch_thread_->quit();
  46. sounds_patch_thread_->quit();
  47. videos_patch_thread_->quit();
  48. if (!texts_patch_thread_->wait(500)) {
  49. qDebug() << "PatchList: ERROR, TEXTS PATCHSET DIDNT STOP, FORCEFULLY TERMINATING!";
  50. texts_patch_thread_->terminate();
  51. texts_patch_thread_->wait();
  52. }
  53. if (!graphics_patch_thread_->wait(500)) {
  54. qDebug() << "PatchList: ERROR, GRAPHICS PATCHSET DIDNT STOP, FORCEFULLY TERMINATING!";
  55. graphics_patch_thread_->terminate();
  56. graphics_patch_thread_->wait();
  57. }
  58. if (!sounds_patch_thread_->wait(500)) {
  59. qDebug() << "PatchList: ERROR, SOUNDS PATCHSET DIDNT STOP, FORCEFULLY TERMINATING!";
  60. sounds_patch_thread_->terminate();
  61. sounds_patch_thread_->wait();
  62. }
  63. if (!videos_patch_thread_->wait(500)) {
  64. qDebug() << "PatchList: ERROR, VIDEOS PATCHSET DIDNT STOP, FORCEFULLY TERMINATING!";
  65. videos_patch_thread_->terminate();
  66. videos_patch_thread_->wait();
  67. }
  68. qDebug() << "Patchlist: all jobs stopped, destroying";
  69. delete texts_patch_;
  70. delete graphics_patch_;
  71. delete sounds_patch_;
  72. delete videos_patch_;
  73. }
  74. QList<Patch *> PatchList::getPatchList()
  75. {
  76. return {texts_patch_, graphics_patch_, sounds_patch_, videos_patch_};
  77. }
  78. LotroDatManager *PatchList::getManager()
  79. {
  80. return lotro_mgr_;
  81. }
  82. void PatchList::onPatchOperationProgressChanged(Patch::OperationProgress operation_progress, Patch *patch)
  83. {
  84. patch_operations_status_[patch] = operation_progress;
  85. Patch::OperationProgress total_status;
  86. for (const Patch::OperationProgress &st : patch_operations_status_) {
  87. total_status = total_status + st;
  88. }
  89. emit progressChanged(total_status);
  90. }
  91. void PatchList::onLotroManagerOperationFinished(QString operation_name, QVector<QVariant>, bool result)
  92. {
  93. if (operation_name == "initializeManager") {
  94. --active_operations_num_;
  95. if (active_operations_num_ == 0) {
  96. emit patchOperationsFinished();
  97. }
  98. if (result) {
  99. startAutoUpdate();
  100. update();
  101. } else {
  102. qCritical() << "DatManager initialisation error!!!";
  103. }
  104. }
  105. if (operation_name == "createBackup" || operation_name == "restoreFromBackup" || operation_name == "removeBackup") {
  106. --active_operations_num_;
  107. if (active_operations_num_ == 0) {
  108. emit patchOperationsFinished();
  109. }
  110. }
  111. }
  112. void PatchList::startAutoUpdate()
  113. {
  114. auto_updates_enabled_ = true;
  115. patch_update_timer.start();
  116. }
  117. void PatchList::stopAutoUpdate()
  118. {
  119. patch_update_timer.stop();
  120. auto_updates_enabled_ = false;
  121. }
  122. void PatchList::initialize() {
  123. ++active_operations_num_;
  124. emit patchOperationsStarted();
  125. QMetaObject::invokeMethod(lotro_mgr_, &LotroDatManager::initializeManager, Qt::QueuedConnection);
  126. }
  127. void PatchList::createBackup()
  128. {
  129. if (active_operations_num_ > 0) {
  130. qWarning() << "Tried to start create backup operation, while others are still running!";
  131. return;
  132. }
  133. ++active_operations_num_;
  134. emit patchOperationsStarted();
  135. QMetaObject::invokeMethod(lotro_mgr_, "createBackup", Qt::QueuedConnection);
  136. }
  137. void PatchList::restoreFromBackup()
  138. {
  139. if (active_operations_num_ > 0) {
  140. qWarning() << "Tried to start restore from backup operation, while others are still running!";
  141. return;
  142. }
  143. ++active_operations_num_;
  144. emit patchOperationsStarted();
  145. QMetaObject::invokeMethod(lotro_mgr_, "restoreFromBackup", Qt::QueuedConnection);
  146. }
  147. void PatchList::removeBackup()
  148. {
  149. if (active_operations_num_ > 0) {
  150. qWarning() << "Tried to start remove backup operation, while others are still running!";
  151. return;
  152. }
  153. ++active_operations_num_;
  154. emit patchOperationsStarted();
  155. QMetaObject::invokeMethod(lotro_mgr_, "removeBackup", Qt::QueuedConnection);
  156. }
  157. void PatchList::onPatchOperationStarted(Patch::Operation, Patch*)
  158. {
  159. if (active_operations_num_ == 0) {
  160. emit patchOperationsStarted();
  161. }
  162. ++active_operations_num_;
  163. }
  164. void PatchList::onPatchOperationFinished(Patch::Operation, Patch*)
  165. {
  166. --active_operations_num_;
  167. if (active_operations_num_ == 0) {
  168. emit patchOperationsFinished();
  169. }
  170. }
  171. void PatchList::update()
  172. {
  173. if (active_operations_num_ > 0) {
  174. qWarning() << "Tried to start patch update chain, while there are already some operations on them!";
  175. return;
  176. }
  177. qInfo() << "PatchList: Starting update chain!";
  178. for (Patch* patch : getPatchList()) {
  179. QMetaObject::invokeMethod(patch, "enqueue",
  180. Q_ARG(QList<Patch::Operation>,
  181. QList<Patch::Operation>({Patch::E_CHECKFORUPDATES,
  182. Patch::E_DOWNLOAD,
  183. Patch::E_INSTALL,
  184. Patch::E_ACTIVATE})));
  185. }
  186. }