Merge branch 'develop' of https://github.com/PrismLauncher/PrismLauncher into import_zip
Signed-off-by: Trial97 <alexandru.tripon97@gmail.com>
This commit is contained in:
		
							
								
								
									
										30
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										30
									
								
								.github/workflows/build.yml
									
									
									
									
										vendored
									
									
								
							| @@ -586,33 +586,3 @@ jobs: | ||||
|         with: | ||||
|           bundle: "Prism Launcher.flatpak" | ||||
|           manifest-path: flatpak/org.prismlauncher.PrismLauncher.yml  | ||||
|  | ||||
|   nix: | ||||
|     runs-on: ubuntu-latest | ||||
|     strategy: | ||||
|       matrix: | ||||
|         package: | ||||
|           - prismlauncher | ||||
|           - prismlauncher-qt5 | ||||
|     steps: | ||||
|       - name: Clone repository | ||||
|         if: inputs.build_type == 'Debug' | ||||
|         uses: actions/checkout@v3 | ||||
|         with: | ||||
|           submodules: 'true' | ||||
|       - name: Install nix | ||||
|         if: inputs.build_type == 'Debug' | ||||
|         uses: cachix/install-nix-action@v22 | ||||
|         with: | ||||
|           install_url: https://nixos.org/nix/install | ||||
|           extra_nix_config: | | ||||
|             auto-optimise-store = true | ||||
|             experimental-features = nix-command flakes | ||||
|       - uses: cachix/cachix-action@v12 | ||||
|         if: inputs.build_type == 'Debug' | ||||
|         with: | ||||
|           name: prismlauncher | ||||
|           authToken: '${{ secrets.CACHIX_AUTH_TOKEN }}' | ||||
|       - name: Build | ||||
|         if: inputs.build_type == 'Debug' | ||||
|         run: nix build .#${{ matrix.package }} --print-build-logs | ||||
|   | ||||
| @@ -82,14 +82,16 @@ Thanks to the awesome people over at [MacStadium](https://www.macstadium.com/), | ||||
|  | ||||
| ## Forking/Redistributing/Custom builds policy | ||||
|  | ||||
| We don't care what you do with your fork/custom build as long as you follow the terms of the [license](LICENSE) (this is a legal responsibility), and if you made code changes rather than just packaging a custom build, please do the following as a basic courtesy: | ||||
| You are free to fork, redistribute and provide custom builds as long as you follow the terms of the [license](LICENSE) (this is a legal responsibility), and if you made code changes rather than just packaging a custom build, please do the following as a basic courtesy: | ||||
|  | ||||
| - Make it clear that your fork is not Prism Launcher and is not endorsed by or affiliated with the Prism Launcher project (<https://prismlauncher.org>). | ||||
| - Go through [CMakeLists.txt](CMakeLists.txt) and change Prism Launcher's API keys to your own or set them to empty strings (`""`) to disable them (this way the program will still compile but the functionality requiring those keys will be disabled). | ||||
|  | ||||
| If you have any questions or want any clarification on the above conditions please make an issue and ask us. | ||||
|  | ||||
| Be aware that if you build this software without removing the provided API keys in [CMakeLists.txt](CMakeLists.txt) you are accepting the following terms and conditions: | ||||
| If you are just building Prism Launcher for your distribution, please make sure to set the `Launcher_BUILD_PLATFORM` to a slug representing your distribution. Examples are `archlinux`, `fedora` and `nixpkgs`. | ||||
|  | ||||
| Note that if you build this software without removing the provided API keys in [CMakeLists.txt](CMakeLists.txt) you are accepting the following terms and conditions: | ||||
|  | ||||
| - [Microsoft Identity Platform Terms of Use](https://docs.microsoft.com/en-us/legal/microsoft-identity-platform/terms-of-use) | ||||
| - [CurseForge 3rd Party API Terms and Conditions](https://support.curseforge.com/en/support/solutions/articles/9000207405-curse-forge-3rd-party-api-terms-and-conditions) | ||||
|   | ||||
							
								
								
									
										18
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										18
									
								
								flake.lock
									
									
									
										generated
									
									
									
								
							| @@ -76,11 +76,11 @@ | ||||
|     "libnbtplusplus": { | ||||
|       "flake": false, | ||||
|       "locked": { | ||||
|         "lastModified": 1650031308, | ||||
|         "narHash": "sha256-TvVOjkUobYJD9itQYueELJX3wmecvEdCbJ0FinW2mL4=", | ||||
|         "lastModified": 1690036783, | ||||
|         "narHash": "sha256-A5kTgICnx+Qdq3Fir/bKTfdTt/T1NQP2SC+nhN1ENug=", | ||||
|         "owner": "PrismLauncher", | ||||
|         "repo": "libnbtplusplus", | ||||
|         "rev": "2203af7eeb48c45398139b583615134efd8d407f", | ||||
|         "rev": "a5e8fd52b8bf4ab5d5bcc042b2a247867589985f", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
| @@ -91,11 +91,11 @@ | ||||
|     }, | ||||
|     "nixpkgs": { | ||||
|       "locked": { | ||||
|         "lastModified": 1689413807, | ||||
|         "narHash": "sha256-exuzOvOhGAEKWQKwDuZAL4N8a1I837hH5eocaTcIbLc=", | ||||
|         "lastModified": 1690026219, | ||||
|         "narHash": "sha256-oOduRk/kzQxOBknZXTLSEYd7tk+GoKvr8wV6Ab+t4AU=", | ||||
|         "owner": "nixos", | ||||
|         "repo": "nixpkgs", | ||||
|         "rev": "46ed466081b9cad1125b11f11a2af5cc40b942c7", | ||||
|         "rev": "f465da166263bc0d4b39dfd4ca28b777c92d4b73", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
| @@ -138,11 +138,11 @@ | ||||
|         ] | ||||
|       }, | ||||
|       "locked": { | ||||
|         "lastModified": 1689328505, | ||||
|         "narHash": "sha256-9B3+OeUn1a/CvzE3GW6nWNwS5J7PDHTyHGlpL3wV5oA=", | ||||
|         "lastModified": 1689668210, | ||||
|         "narHash": "sha256-XAATwDkaUxH958yXLs1lcEOmU6pSEIkatY3qjqk8X0E=", | ||||
|         "owner": "cachix", | ||||
|         "repo": "pre-commit-hooks.nix", | ||||
|         "rev": "5e28316db471d1ac234beb70031b635437421dd6", | ||||
|         "rev": "eb433bff05b285258be76513add6f6c57b441775", | ||||
|         "type": "github" | ||||
|       }, | ||||
|       "original": { | ||||
|   | ||||
							
								
								
									
										5
									
								
								garnix.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								garnix.yaml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | ||||
| builds: | ||||
|   exclude: [] | ||||
|   include: | ||||
|     - "devShells.*-linux.*" | ||||
|     - "packages.*-linux.*" | ||||
| @@ -1186,6 +1186,16 @@ QIcon Application::getThemedIcon(const QString& name) | ||||
|     return QIcon::fromTheme(name); | ||||
| } | ||||
|  | ||||
| QList<CatPack*> Application::getValidCatPacks() | ||||
| { | ||||
|     return m_themeManager->getValidCatPacks(); | ||||
| } | ||||
|  | ||||
| QString Application::getCatPack(QString catName) | ||||
| { | ||||
|     return m_themeManager->getCatPack(catName); | ||||
| } | ||||
|  | ||||
| bool Application::openJsonEditor(const QString& filename) | ||||
| { | ||||
|     const QString file = QDir::current().absoluteFilePath(filename); | ||||
|   | ||||
| @@ -48,6 +48,7 @@ | ||||
| #include <BaseInstance.h> | ||||
|  | ||||
| #include "minecraft/launch/MinecraftServerTarget.h" | ||||
| #include "ui/themes/CatPack.h" | ||||
|  | ||||
| class LaunchController; | ||||
| class LocalPeer; | ||||
| @@ -126,9 +127,11 @@ public: | ||||
|  | ||||
|     void setApplicationTheme(const QString& name); | ||||
|  | ||||
|     shared_qobject_ptr<ExternalUpdater> updater() { | ||||
|         return m_updater; | ||||
|     } | ||||
|     QList<CatPack*> getValidCatPacks(); | ||||
|  | ||||
|     QString getCatPack(QString catName = ""); | ||||
|  | ||||
|     shared_qobject_ptr<ExternalUpdater> updater() { return m_updater; } | ||||
|  | ||||
|     void triggerUpdateCheck(); | ||||
|  | ||||
|   | ||||
| @@ -761,6 +761,8 @@ SET(LAUNCHER_SOURCES | ||||
|     ui/themes/SystemTheme.h | ||||
|     ui/themes/ThemeManager.cpp | ||||
|     ui/themes/ThemeManager.h | ||||
|     ui/themes/CatPack.cpp | ||||
|     ui/themes/CatPack.h | ||||
|  | ||||
|     # Processes | ||||
|     LaunchController.h | ||||
|   | ||||
| @@ -92,7 +92,7 @@ void InstanceImportTask::executeTask() | ||||
|  | ||||
|         connect(filesNetJob.get(), &NetJob::succeeded, this, &InstanceImportTask::processZipPack); | ||||
|         connect(filesNetJob.get(), &NetJob::progress, this, &InstanceImportTask::setProgress); | ||||
|         connect(filesNetJob.get(), &NetJob::stepProgress, this, &InstanceImportTask::propogateStepProgress); | ||||
|         connect(filesNetJob.get(), &NetJob::stepProgress, this, &InstanceImportTask::propagateStepProgress); | ||||
|         connect(filesNetJob.get(), &NetJob::failed, this, &InstanceImportTask::emitFailed); | ||||
|         connect(filesNetJob.get(), &NetJob::aborted, this, &InstanceImportTask::emitAborted); | ||||
|         task.reset(filesNetJob); | ||||
| @@ -192,7 +192,7 @@ void InstanceImportTask::processZipPack() | ||||
|         stepProgress(*progressStep); | ||||
|         emitFailed(reason); | ||||
|     }); | ||||
|     connect(zipTask.get(), &Task::stepProgress, this, &InstanceImportTask::propogateStepProgress); | ||||
|     connect(zipTask.get(), &Task::stepProgress, this, &InstanceImportTask::propagateStepProgress); | ||||
|  | ||||
|     connect(zipTask.get(), &Task::progress, this, [this, progressStep](qint64 current, qint64 total) { | ||||
|         progressStep->update(current, total); | ||||
| @@ -287,7 +287,7 @@ void InstanceImportTask::processFlame() | ||||
|     }); | ||||
|     connect(inst_creation_task.get(), &Task::failed, this, &InstanceImportTask::emitFailed); | ||||
|     connect(inst_creation_task.get(), &Task::progress, this, &InstanceImportTask::setProgress); | ||||
|     connect(inst_creation_task.get(), &Task::stepProgress, this, &InstanceImportTask::propogateStepProgress); | ||||
|     connect(inst_creation_task.get(), &Task::stepProgress, this, &InstanceImportTask::propagateStepProgress); | ||||
|     connect(inst_creation_task.get(), &Task::status, this, &InstanceImportTask::setStatus); | ||||
|     connect(inst_creation_task.get(), &Task::details, this, &InstanceImportTask::setDetails); | ||||
|  | ||||
| @@ -380,7 +380,7 @@ void InstanceImportTask::processModrinth() | ||||
|     }); | ||||
|     connect(inst_creation_task, &Task::failed, this, &InstanceImportTask::emitFailed); | ||||
|     connect(inst_creation_task, &Task::progress, this, &InstanceImportTask::setProgress); | ||||
|     connect(inst_creation_task, &Task::stepProgress, this, &InstanceImportTask::propogateStepProgress); | ||||
|     connect(inst_creation_task, &Task::stepProgress, this, &InstanceImportTask::propagateStepProgress); | ||||
|     connect(inst_creation_task, &Task::status, this, &InstanceImportTask::setStatus); | ||||
|     connect(inst_creation_task, &Task::details, this, &InstanceImportTask::setDetails); | ||||
|     connect(inst_creation_task, &Task::finished, inst_creation_task, &InstanceCreationTask::deleteLater); | ||||
|   | ||||
| @@ -799,7 +799,7 @@ class InstanceStaging : public Task { | ||||
|         connect(child, &Task::status, this, &InstanceStaging::setStatus); | ||||
|         connect(child, &Task::details, this, &InstanceStaging::setDetails); | ||||
|         connect(child, &Task::progress, this, &InstanceStaging::setProgress); | ||||
|         connect(child, &Task::stepProgress, this, &InstanceStaging::propogateStepProgress); | ||||
|         connect(child, &Task::stepProgress, this, &InstanceStaging::propagateStepProgress); | ||||
|         connect(&m_backoffTimer, &QTimer::timeout, this, &InstanceStaging::childSucceded); | ||||
|     } | ||||
|  | ||||
|   | ||||
| @@ -54,7 +54,7 @@ ResourceDownloadTask::ResourceDownloadTask(ModPlatform::IndexedPack::Ptr pack, | ||||
|     m_filesNetJob->addNetAction(Net::Download::makeFile(m_pack_version.downloadUrl, dir.absoluteFilePath(getFilename()))); | ||||
|     connect(m_filesNetJob.get(), &NetJob::succeeded, this, &ResourceDownloadTask::downloadSucceeded); | ||||
|     connect(m_filesNetJob.get(), &NetJob::progress, this, &ResourceDownloadTask::downloadProgressChanged); | ||||
|     connect(m_filesNetJob.get(), &NetJob::stepProgress, this, &ResourceDownloadTask::propogateStepProgress); | ||||
|     connect(m_filesNetJob.get(), &NetJob::stepProgress, this, &ResourceDownloadTask::propagateStepProgress); | ||||
|     connect(m_filesNetJob.get(), &NetJob::failed, this, &ResourceDownloadTask::downloadFailed); | ||||
|  | ||||
|     addTask(m_filesNetJob); | ||||
|   | ||||
| @@ -28,7 +28,7 @@ void Update::executeTask() | ||||
|     { | ||||
|         connect(m_updateTask.get(), &Task::finished, this, &Update::updateFinished); | ||||
|         connect(m_updateTask.get(), &Task::progress, this, &Update::setProgress); | ||||
|         connect(m_updateTask.get(), &Task::stepProgress, this, &Update::propogateStepProgress); | ||||
|         connect(m_updateTask.get(), &Task::stepProgress, this, &Update::propagateStepProgress); | ||||
|         connect(m_updateTask.get(), &Task::status, this, &Update::setStatus); | ||||
|         connect(m_updateTask.get(), &Task::details, this, &Update::setDetails); | ||||
|         emit progressReportingRequest(); | ||||
|   | ||||
| @@ -843,7 +843,7 @@ QMap<QString, QString> MinecraftInstance::createCensorFilterFromSession(AuthSess | ||||
|     { | ||||
|         addToFilter(sessionRef.session, tr("<SESSION ID>")); | ||||
|     } | ||||
|     if (sessionRef.access_token != "offline") { | ||||
|     if (sessionRef.access_token != "0") { | ||||
|         addToFilter(sessionRef.access_token, tr("<ACCESS TOKEN>")); | ||||
|     } | ||||
|     if(sessionRef.client_token.size()) { | ||||
|   | ||||
| @@ -22,7 +22,7 @@ void MinecraftLoadAndCheck::executeTask() | ||||
|     connect(m_task.get(), &Task::failed, this, &MinecraftLoadAndCheck::subtaskFailed); | ||||
|     connect(m_task.get(), &Task::aborted, this, [this]{ subtaskFailed(tr("Aborted")); }); | ||||
|     connect(m_task.get(), &Task::progress, this, &MinecraftLoadAndCheck::progress); | ||||
|     connect(m_task.get(), &Task::stepProgress, this, &MinecraftLoadAndCheck::propogateStepProgress); | ||||
|     connect(m_task.get(), &Task::stepProgress, this, &MinecraftLoadAndCheck::propagateStepProgress); | ||||
|     connect(m_task.get(), &Task::status, this, &MinecraftLoadAndCheck::setStatus); | ||||
| } | ||||
|  | ||||
|   | ||||
| @@ -100,7 +100,7 @@ void MinecraftUpdate::next() | ||||
|         disconnect(task.get(), &Task::failed, this, &MinecraftUpdate::subtaskFailed); | ||||
|         disconnect(task.get(), &Task::aborted, this, &Task::abort); | ||||
|         disconnect(task.get(), &Task::progress, this, &MinecraftUpdate::progress); | ||||
|         disconnect(task.get(), &Task::stepProgress, this, &MinecraftUpdate::propogateStepProgress); | ||||
|         disconnect(task.get(), &Task::stepProgress, this, &MinecraftUpdate::propagateStepProgress); | ||||
|         disconnect(task.get(), &Task::status, this, &MinecraftUpdate::setStatus); | ||||
|         disconnect(task.get(), &Task::details, this, &MinecraftUpdate::setDetails); | ||||
|     } | ||||
| @@ -120,7 +120,7 @@ void MinecraftUpdate::next() | ||||
|     connect(task.get(), &Task::failed, this, &MinecraftUpdate::subtaskFailed); | ||||
|     connect(task.get(), &Task::aborted, this, &Task::abort); | ||||
|     connect(task.get(), &Task::progress, this, &MinecraftUpdate::progress); | ||||
|     connect(task.get(), &Task::stepProgress, this, &MinecraftUpdate::propogateStepProgress); | ||||
|     connect(task.get(), &Task::stepProgress, this, &MinecraftUpdate::propagateStepProgress); | ||||
|     connect(task.get(), &Task::status, this, &MinecraftUpdate::setStatus); | ||||
|     connect(task.get(), &Task::details, this, &MinecraftUpdate::setDetails); | ||||
|     // if the task is already running, do not start it again | ||||
|   | ||||
| @@ -374,6 +374,10 @@ bool AccountData::resumeStateFromV3(QJsonObject data) { | ||||
|     } | ||||
|  | ||||
|     yggdrasilToken = tokenFromJSONV3(data, "ygg"); | ||||
|     // versions before 7.2 used "offline" as the offline token | ||||
|     if (yggdrasilToken.token == "offline") | ||||
|         yggdrasilToken.token = "0"; | ||||
|  | ||||
|     minecraftProfile = profileFromJSONV3(data, "profile"); | ||||
|     if(!entitlementFromJSONV3(data, minecraftEntitlement)) { | ||||
|         if(minecraftProfile.validity != Katabasis::Validity::None) { | ||||
|   | ||||
| @@ -37,6 +37,7 @@ | ||||
|  | ||||
| #include "MinecraftAccount.h" | ||||
|  | ||||
| #include <QCryptographicHash> | ||||
| #include <QUuid> | ||||
| #include <QJsonObject> | ||||
| #include <QJsonArray> | ||||
| @@ -93,14 +94,14 @@ MinecraftAccountPtr MinecraftAccount::createOffline(const QString &username) | ||||
| { | ||||
|     auto account = makeShared<MinecraftAccount>(); | ||||
|     account->data.type = AccountType::Offline; | ||||
|     account->data.yggdrasilToken.token = "offline"; | ||||
|     account->data.yggdrasilToken.token = "0"; | ||||
|     account->data.yggdrasilToken.validity = Katabasis::Validity::Certain; | ||||
|     account->data.yggdrasilToken.issueInstant = QDateTime::currentDateTimeUtc(); | ||||
|     account->data.yggdrasilToken.extra["userName"] = username; | ||||
|     account->data.yggdrasilToken.extra["clientToken"] = QUuid::createUuid().toString().remove(QRegularExpression("[{}-]")); | ||||
|     account->data.minecraftEntitlement.ownsMinecraft = true; | ||||
|     account->data.minecraftEntitlement.canPlayMinecraft = true; | ||||
|     account->data.minecraftProfile.id = QUuid::createUuid().toString().remove(QRegularExpression("[{}-]")); | ||||
|     account->data.minecraftProfile.id = uuidFromUsername(username).toString().remove(QRegularExpression("[{}-]")); | ||||
|     account->data.minecraftProfile.name = username; | ||||
|     account->data.minecraftProfile.validity = Katabasis::Validity::Certain; | ||||
|     return account; | ||||
| @@ -334,3 +335,32 @@ void MinecraftAccount::incrementUses() | ||||
|         qWarning() << "Profile" << data.profileId() << "is now in use."; | ||||
|     } | ||||
| } | ||||
|  | ||||
| QUuid MinecraftAccount::uuidFromUsername(QString username) { | ||||
|     auto input = QString("OfflinePlayer:%1").arg(username).toUtf8(); | ||||
|  | ||||
|     // basically a reimplementation of Java's UUID#nameUUIDFromBytes | ||||
|     QByteArray digest = QCryptographicHash::hash(input, QCryptographicHash::Md5); | ||||
|  | ||||
| #if QT_VERSION < QT_VERSION_CHECK(6, 0, 0) | ||||
|     auto bOr = [](QByteArray& array, int index, char value) { | ||||
|         array[index] = array.at(index) | value; | ||||
|     }; | ||||
|     auto bAnd = [](QByteArray& array, int index, char value) { | ||||
|         array[index] = array.at(index) & value; | ||||
|     }; | ||||
| #else | ||||
|     auto bOr = [](QByteArray& array, qsizetype index, char value) { | ||||
|         array[index] |= value; | ||||
|     }; | ||||
|     auto bAnd = [](QByteArray& array, qsizetype index, char value) { | ||||
|         array[index] &= value; | ||||
|     }; | ||||
| #endif | ||||
|     bAnd(digest, 6, (char) 0x0f); // clear version | ||||
|     bOr(digest, 6, (char) 0x30); // set to version 3 | ||||
|     bAnd(digest, 8, (char) 0x3f); // clear variant | ||||
|     bOr(digest, 8, (char) 0x80); // set to IETF variant | ||||
|  | ||||
|     return QUuid::fromRfc4122(digest); | ||||
| } | ||||
|   | ||||
| @@ -98,6 +98,8 @@ public: /* construction */ | ||||
|     static MinecraftAccountPtr loadFromJsonV2(const QJsonObject &json); | ||||
|     static MinecraftAccountPtr loadFromJsonV3(const QJsonObject &json); | ||||
|  | ||||
|     static QUuid uuidFromUsername(QString username); | ||||
|  | ||||
|     //! Saves a MinecraftAccount to a JSON object and returns it. | ||||
|     QJsonObject saveToJson() const; | ||||
|  | ||||
|   | ||||
| @@ -45,7 +45,7 @@ void AssetUpdateTask::executeTask() | ||||
|     connect(downloadJob.get(), &NetJob::failed, this, &AssetUpdateTask::assetIndexFailed); | ||||
|     connect(downloadJob.get(), &NetJob::aborted, this, [this]{ emitFailed(tr("Aborted")); }); | ||||
|     connect(downloadJob.get(), &NetJob::progress, this, &AssetUpdateTask::progress); | ||||
|     connect(downloadJob.get(), &NetJob::stepProgress, this, &AssetUpdateTask::propogateStepProgress); | ||||
|     connect(downloadJob.get(), &NetJob::stepProgress, this, &AssetUpdateTask::propagateStepProgress); | ||||
|  | ||||
|     qDebug() << m_inst->name() << ": Starting asset index download"; | ||||
|     downloadJob->start(); | ||||
| @@ -84,7 +84,7 @@ void AssetUpdateTask::assetIndexFinished() | ||||
|         connect(downloadJob.get(), &NetJob::failed, this, &AssetUpdateTask::assetsFailed); | ||||
|         connect(downloadJob.get(), &NetJob::aborted, this, [this]{ emitFailed(tr("Aborted")); }); | ||||
|         connect(downloadJob.get(), &NetJob::progress, this, &AssetUpdateTask::progress); | ||||
|         connect(downloadJob.get(), &NetJob::stepProgress, this, &AssetUpdateTask::propogateStepProgress); | ||||
|         connect(downloadJob.get(), &NetJob::stepProgress, this, &AssetUpdateTask::propagateStepProgress); | ||||
|         downloadJob->start(); | ||||
|         return; | ||||
|     } | ||||
|   | ||||
| @@ -75,7 +75,7 @@ void FMLLibrariesTask::executeTask() | ||||
|     connect(dljob.get(), &NetJob::failed, this, &FMLLibrariesTask::fmllibsFailed); | ||||
|     connect(dljob.get(), &NetJob::aborted, this, [this]{ emitFailed(tr("Aborted")); }); | ||||
|     connect(dljob.get(), &NetJob::progress, this, &FMLLibrariesTask::progress); | ||||
|     connect(dljob.get(), &NetJob::stepProgress, this, &FMLLibrariesTask::propogateStepProgress); | ||||
|     connect(dljob.get(), &NetJob::stepProgress, this, &FMLLibrariesTask::propagateStepProgress); | ||||
|     downloadJob.reset(dljob); | ||||
|     downloadJob->start(); | ||||
| } | ||||
|   | ||||
| @@ -70,7 +70,7 @@ void LibrariesTask::executeTask() | ||||
|     connect(downloadJob.get(), &NetJob::failed, this, &LibrariesTask::jarlibFailed); | ||||
|     connect(downloadJob.get(), &NetJob::aborted, this, [this]{ emitFailed(tr("Aborted")); }); | ||||
|     connect(downloadJob.get(), &NetJob::progress, this, &LibrariesTask::progress); | ||||
|     connect(downloadJob.get(), &NetJob::stepProgress, this, &LibrariesTask::propogateStepProgress); | ||||
|     connect(downloadJob.get(), &NetJob::stepProgress, this, &LibrariesTask::propagateStepProgress); | ||||
|  | ||||
|     downloadJob->start(); | ||||
| } | ||||
|   | ||||
| @@ -684,7 +684,7 @@ void PackInstallTask::installConfigs() | ||||
|         abortable = true; | ||||
|         setProgress(current, total); | ||||
|     }); | ||||
|     connect(jobPtr.get(), &NetJob::stepProgress, this, &PackInstallTask::propogateStepProgress); | ||||
|     connect(jobPtr.get(), &NetJob::stepProgress, this, &PackInstallTask::propagateStepProgress); | ||||
|     connect(jobPtr.get(), &NetJob::aborted, [&]{ | ||||
|         abortable = false; | ||||
|         jobPtr.reset(); | ||||
| @@ -852,7 +852,7 @@ void PackInstallTask::downloadMods() | ||||
|         abortable = true; | ||||
|         setProgress(current, total); | ||||
|     }); | ||||
|     connect(jobPtr.get(), &NetJob::stepProgress, this, &PackInstallTask::propogateStepProgress); | ||||
|     connect(jobPtr.get(), &NetJob::stepProgress, this, &PackInstallTask::propagateStepProgress); | ||||
|     connect(jobPtr.get(), &NetJob::aborted, [&] | ||||
|     { | ||||
|         abortable = false; | ||||
|   | ||||
| @@ -52,7 +52,7 @@ void Flame::FileResolvingTask::executeTask() | ||||
|         stepProgress(*step_progress); | ||||
|         emitFailed(reason); | ||||
|     }); | ||||
|     connect(m_dljob.get(), &NetJob::stepProgress, this, &FileResolvingTask::propogateStepProgress); | ||||
|     connect(m_dljob.get(), &NetJob::stepProgress, this, &FileResolvingTask::propagateStepProgress); | ||||
|     connect(m_dljob.get(), &NetJob::progress, this, [this, step_progress](qint64 current, qint64 total) { | ||||
|         qDebug() << "Resolve slug progress" << current << total; | ||||
|         step_progress->update(current, total); | ||||
| @@ -118,7 +118,7 @@ void Flame::FileResolvingTask::netJobFinished() | ||||
|         stepProgress(*step_progress); | ||||
|         emitFailed(reason); | ||||
|     }); | ||||
|     connect(m_checkJob.get(), &NetJob::stepProgress, this, &FileResolvingTask::propogateStepProgress); | ||||
|     connect(m_checkJob.get(), &NetJob::stepProgress, this, &FileResolvingTask::propagateStepProgress); | ||||
|     connect(m_checkJob.get(), &NetJob::progress, this, [this, step_progress](qint64 current, qint64 total) { | ||||
|         qDebug() << "Resolve slug progress" << current << total; | ||||
|         step_progress->update(current, total); | ||||
| @@ -195,7 +195,7 @@ void Flame::FileResolvingTask::modrinthCheckFinished() | ||||
|             stepProgress(*step_progress); | ||||
|             emitFailed(reason); | ||||
|         }); | ||||
|         connect(m_slugJob.get(), &NetJob::stepProgress, this, &FileResolvingTask::propogateStepProgress); | ||||
|         connect(m_slugJob.get(), &NetJob::stepProgress, this, &FileResolvingTask::propagateStepProgress); | ||||
|         connect(m_slugJob.get(), &NetJob::progress, this, [this, step_progress](qint64 current, qint64 total) { | ||||
|             qDebug() << "Resolve slug progress" << current << total; | ||||
|             step_progress->update(current, total); | ||||
|   | ||||
| @@ -57,15 +57,11 @@ | ||||
| #include <QDebug> | ||||
| #include <QFileInfo> | ||||
|  | ||||
| #include "meta/Index.h" | ||||
| #include "meta/VersionList.h" | ||||
| #include "minecraft/World.h" | ||||
| #include "minecraft/mod/tasks/LocalResourceParse.h" | ||||
|  | ||||
|  | ||||
| const static QMap<QString, QString> forgemap = { { "1.2.5", "3.4.9.171" }, | ||||
|                                                  { "1.4.2", "6.0.1.355" }, | ||||
|                                                  { "1.4.7", "6.6.2.534" }, | ||||
|                                                  { "1.5.2", "7.8.1.737" } }; | ||||
|  | ||||
| static const FlameAPI api; | ||||
|  | ||||
| bool FlameCreationTask::abort() | ||||
| @@ -259,6 +255,56 @@ bool FlameCreationTask::updateInstance() | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| QString FlameCreationTask::getVersionForLoader(QString uid, QString loaderType, QString loaderVersion, QString mcVersion) | ||||
| { | ||||
|     if (loaderVersion == "recommended") { | ||||
|         auto vlist = APPLICATION->metadataIndex()->get(uid); | ||||
|         if (!vlist) { | ||||
|             setError(tr("Failed to get local metadata index for %1").arg(uid)); | ||||
|             return {}; | ||||
|         } | ||||
|  | ||||
|         if (!vlist->isLoaded()) { | ||||
|             QEventLoop loadVersionLoop; | ||||
|             auto task = vlist->getLoadTask(); | ||||
|             connect(task.get(), &Task::finished, &loadVersionLoop, &QEventLoop::quit); | ||||
|             if (!task->isRunning()) | ||||
|                 task->start(); | ||||
|  | ||||
|             loadVersionLoop.exec(); | ||||
|         } | ||||
|  | ||||
|         for (auto version : vlist->versions()) { | ||||
|             // first recommended build we find, we use. | ||||
|             if (!version->isRecommended()) | ||||
|                 continue; | ||||
|             auto reqs = version->requiredSet(); | ||||
|  | ||||
|             // filter by minecraft version, if the loader depends on a certain version. | ||||
|             // not all mod loaders depend on a given Minecraft version, so we won't do this | ||||
|             // filtering for those loaders. | ||||
|             if (loaderType == "forge") { | ||||
|                 auto iter = std::find_if(reqs.begin(), reqs.end(), [mcVersion](const Meta::Require& req) { | ||||
|                     return req.uid == "net.minecraft" && req.equalsVersion == mcVersion; | ||||
|                 }); | ||||
|                 if (iter == reqs.end()) | ||||
|                     continue; | ||||
|             } | ||||
|             return version->descriptor(); | ||||
|         } | ||||
|  | ||||
|         setError(tr("Failed to find version for %1 loader").arg(loaderType)); | ||||
|         return {}; | ||||
|     } | ||||
|  | ||||
|     if (loaderVersion.isEmpty()) { | ||||
|         emitFailed(tr("No loader version set for modpack!")); | ||||
|         return {}; | ||||
|     } | ||||
|  | ||||
|     return loaderVersion; | ||||
| } | ||||
|  | ||||
| bool FlameCreationTask::createInstance() | ||||
| { | ||||
|     QEventLoop loop; | ||||
| @@ -297,22 +343,29 @@ bool FlameCreationTask::createInstance() | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     QString forgeVersion; | ||||
|     QString fabricVersion; | ||||
|     // TODO: is Quilt relevant here? | ||||
|     QString loaderType; | ||||
|     QString loaderUid; | ||||
|     QString loaderVersion; | ||||
|  | ||||
|     for (auto& loader : m_pack.minecraft.modLoaders) { | ||||
|         auto id = loader.id; | ||||
|         if (id.startsWith("forge-")) { | ||||
|             id.remove("forge-"); | ||||
|             forgeVersion = id; | ||||
|             continue; | ||||
|         } | ||||
|         if (id.startsWith("fabric-")) { | ||||
|             loaderType = "forge"; | ||||
|             loaderUid = "net.minecraftforge"; | ||||
|         } else if (loaderType == "fabric") { | ||||
|             id.remove("fabric-"); | ||||
|             fabricVersion = id; | ||||
|             loaderType = "fabric"; | ||||
|             loaderUid = "net.fabricmc.fabric-loader"; | ||||
|         } else if (loaderType == "quilt") { | ||||
|             id.remove("quilt-"); | ||||
|             loaderType = "quilt"; | ||||
|             loaderUid = "org.quiltmc.quilt-loader"; | ||||
|         } else { | ||||
|             logWarning(tr("Unknown mod loader in manifest: %1").arg(id)); | ||||
|             continue; | ||||
|         } | ||||
|         logWarning(tr("Unknown mod loader in manifest: %1").arg(id)); | ||||
|         loaderVersion = id; | ||||
|     } | ||||
|  | ||||
|     QString configPath = FS::PathCombine(m_stagingPath, "instance.cfg"); | ||||
| @@ -329,19 +382,12 @@ bool FlameCreationTask::createInstance() | ||||
|     auto components = instance.getPackProfile(); | ||||
|     components->buildingFromScratch(); | ||||
|     components->setComponentVersion("net.minecraft", mcVersion, true); | ||||
|     if (!forgeVersion.isEmpty()) { | ||||
|         // FIXME: dirty, nasty, hack. Proper solution requires dependency resolution and knowledge of the metadata. | ||||
|         if (forgeVersion == "recommended") { | ||||
|             if (forgemap.contains(mcVersion)) { | ||||
|                 forgeVersion = forgemap[mcVersion]; | ||||
|             } else { | ||||
|                 logWarning(tr("Could not map recommended Forge version for Minecraft %1").arg(mcVersion)); | ||||
|     if (!loaderType.isEmpty()) { | ||||
|         auto version = getVersionForLoader(loaderUid, loaderType, loaderVersion, mcVersion); | ||||
|         if (version.isEmpty()) | ||||
|             return false; | ||||
|         components->setComponentVersion(loaderUid, version); | ||||
|     } | ||||
|         } | ||||
|         components->setComponentVersion("net.minecraftforge", forgeVersion); | ||||
|     } | ||||
|     if (!fabricVersion.isEmpty()) | ||||
|         components->setComponentVersion("net.fabricmc.fabric-loader", fabricVersion); | ||||
|  | ||||
|     if (m_instIcon != "default") { | ||||
|         instance.setIconKey(m_instIcon); | ||||
| @@ -386,7 +432,7 @@ bool FlameCreationTask::createInstance() | ||||
|     }); | ||||
|     connect(m_mod_id_resolver.get(), &Flame::FileResolvingTask::progress, this, &FlameCreationTask::setProgress); | ||||
|     connect(m_mod_id_resolver.get(), &Flame::FileResolvingTask::status, this, &FlameCreationTask::setStatus); | ||||
|     connect(m_mod_id_resolver.get(), &Flame::FileResolvingTask::stepProgress, this, &FlameCreationTask::propogateStepProgress); | ||||
|     connect(m_mod_id_resolver.get(), &Flame::FileResolvingTask::stepProgress, this, &FlameCreationTask::propagateStepProgress); | ||||
|     connect(m_mod_id_resolver.get(), &Flame::FileResolvingTask::details, this, &FlameCreationTask::setDetails); | ||||
|     m_mod_id_resolver->start(); | ||||
|  | ||||
| @@ -506,7 +552,7 @@ void FlameCreationTask::setupDownloadJob(QEventLoop& loop) | ||||
|         setDetails(tr("%1 out of %2 complete").arg(current).arg(total)); | ||||
|         setProgress(current, total); | ||||
|     }); | ||||
|     connect(m_files_job.get(), &NetJob::stepProgress, this, &FlameCreationTask::propogateStepProgress); | ||||
|     connect(m_files_job.get(), &NetJob::stepProgress, this, &FlameCreationTask::propagateStepProgress); | ||||
|     connect(m_files_job.get(), &NetJob::finished, &loop, &QEventLoop::quit); | ||||
|  | ||||
|     setStatus(tr("Downloading mods...")); | ||||
| @@ -545,7 +591,6 @@ void FlameCreationTask::copyBlockedMods(QList<BlockedMod> const& blocked_mods) | ||||
|     setAbortable(true); | ||||
| } | ||||
|  | ||||
|  | ||||
| void FlameCreationTask::validateZIPResouces() | ||||
| { | ||||
|     qDebug() << "Validating whether resources stored as .zip are in the right place"; | ||||
|   | ||||
| @@ -57,10 +57,7 @@ class FlameCreationTask final : public InstanceCreationTask { | ||||
|                       QString id, | ||||
|                       QString version_id, | ||||
|                       QString original_instance_id = {}) | ||||
|         : InstanceCreationTask() | ||||
|         , m_parent(parent) | ||||
|         , m_managed_id(std::move(id)) | ||||
|         , m_managed_version_id(std::move(version_id)) | ||||
|         : InstanceCreationTask(), m_parent(parent), m_managed_id(std::move(id)), m_managed_version_id(std::move(version_id)) | ||||
|     { | ||||
|         setStagingPath(staging_path); | ||||
|         setParentSettings(global_settings); | ||||
| @@ -78,6 +75,7 @@ class FlameCreationTask final : public InstanceCreationTask { | ||||
|     void setupDownloadJob(QEventLoop&); | ||||
|     void copyBlockedMods(QList<BlockedMod> const& blocked_mods); | ||||
|     void validateZIPResouces(); | ||||
|     QString getVersionForLoader(QString uid, QString loaderType, QString version, QString mcVersion); | ||||
|  | ||||
|    private: | ||||
|     QWidget* m_parent = nullptr; | ||||
|   | ||||
| @@ -166,7 +166,7 @@ void FlamePackExportTask::collectHashes() | ||||
|         stepProgress(*progressStep); | ||||
|         emitFailed(reason); | ||||
|     }); | ||||
|     connect(hashingTask.get(), &Task::stepProgress, this, &FlamePackExportTask::propogateStepProgress); | ||||
|     connect(hashingTask.get(), &Task::stepProgress, this, &FlamePackExportTask::propagateStepProgress); | ||||
|  | ||||
|     connect(hashingTask.get(), &Task::progress, this, [this, progressStep](qint64 current, qint64 total) { | ||||
|         progressStep->update(current, total); | ||||
|   | ||||
| @@ -81,7 +81,7 @@ void PackInstallTask::downloadPack() | ||||
|  | ||||
|     connect(netJobContainer.get(), &NetJob::succeeded, this, &PackInstallTask::unzip); | ||||
|     connect(netJobContainer.get(), &NetJob::failed, this, &PackInstallTask::emitFailed); | ||||
|     connect(netJobContainer.get(), &NetJob::stepProgress, this, &PackInstallTask::propogateStepProgress); | ||||
|     connect(netJobContainer.get(), &NetJob::stepProgress, this, &PackInstallTask::propagateStepProgress); | ||||
|     connect(netJobContainer.get(), &NetJob::aborted, this, &PackInstallTask::emitAborted); | ||||
|  | ||||
|     netJobContainer->start(); | ||||
|   | ||||
| @@ -267,7 +267,7 @@ bool ModrinthCreationTask::createInstance() | ||||
|         setDetails(tr("%1 out of %2 complete").arg(current).arg(total)); | ||||
|         setProgress(current, total);  | ||||
|     }); | ||||
|     connect(m_files_job.get(), &NetJob::stepProgress, this, &ModrinthCreationTask::propogateStepProgress); | ||||
|     connect(m_files_job.get(), &NetJob::stepProgress, this, &ModrinthCreationTask::propagateStepProgress); | ||||
|  | ||||
|     setStatus(tr("Downloading mods...")); | ||||
|     m_files_job->start(); | ||||
|   | ||||
| @@ -50,7 +50,7 @@ void Technic::SingleZipPackInstallTask::executeTask() | ||||
|     auto job = m_filesNetJob.get(); | ||||
|     connect(job, &NetJob::succeeded, this, &Technic::SingleZipPackInstallTask::downloadSucceeded); | ||||
|     connect(job, &NetJob::progress, this, &Technic::SingleZipPackInstallTask::downloadProgressChanged); | ||||
|     connect(job, &NetJob::stepProgress, this, &Technic::SingleZipPackInstallTask::propogateStepProgress); | ||||
|     connect(job, &NetJob::stepProgress, this, &Technic::SingleZipPackInstallTask::propagateStepProgress); | ||||
|     connect(job, &NetJob::failed, this, &Technic::SingleZipPackInstallTask::downloadFailed); | ||||
|     m_filesNetJob->start(); | ||||
| } | ||||
|   | ||||
| @@ -126,7 +126,7 @@ void Technic::SolderPackInstallTask::fileListSucceeded() | ||||
|  | ||||
|     connect(m_filesNetJob.get(), &NetJob::succeeded, this, &Technic::SolderPackInstallTask::downloadSucceeded); | ||||
|     connect(m_filesNetJob.get(), &NetJob::progress, this, &Technic::SolderPackInstallTask::downloadProgressChanged); | ||||
|     connect(m_filesNetJob.get(), &NetJob::stepProgress, this, &Technic::SolderPackInstallTask::propogateStepProgress); | ||||
|     connect(m_filesNetJob.get(), &NetJob::stepProgress, this, &Technic::SolderPackInstallTask::propagateStepProgress); | ||||
|     connect(m_filesNetJob.get(), &NetJob::failed, this, &Technic::SolderPackInstallTask::downloadFailed); | ||||
|     connect(m_filesNetJob.get(), &NetJob::aborted, this, &Technic::SolderPackInstallTask::downloadAborted); | ||||
|     m_filesNetJob->start(); | ||||
|   | ||||
| @@ -161,7 +161,7 @@ void Task::emitSucceeded() | ||||
|     emit finished(); | ||||
| } | ||||
|  | ||||
| void Task::propogateStepProgress(TaskStepProgress const& task_progress) | ||||
| void Task::propagateStepProgress(TaskStepProgress const& task_progress) | ||||
| { | ||||
|     emit stepProgress(task_progress); | ||||
| } | ||||
|   | ||||
| @@ -167,7 +167,7 @@ class Task : public QObject, public QRunnable { | ||||
|     virtual void emitAborted(); | ||||
|     virtual void emitFailed(QString reason = ""); | ||||
|  | ||||
|     virtual void propogateStepProgress(TaskStepProgress const& task_progress); | ||||
|     virtual void propagateStepProgress(TaskStepProgress const& task_progress); | ||||
|  | ||||
|    public slots: | ||||
|     void setStatus(const QString& status); | ||||
|   | ||||
| @@ -48,7 +48,6 @@ | ||||
| #include <QAccessible> | ||||
|  | ||||
| #include "VisualGroup.h" | ||||
| #include "ui/themes/ThemeManager.h" | ||||
| #include <QDebug> | ||||
|  | ||||
| #include <Application.h> | ||||
| @@ -504,7 +503,7 @@ void InstanceView::setPaintCat(bool visible) | ||||
| { | ||||
|     m_catVisible = visible; | ||||
|     if (visible) | ||||
|         m_catPixmap.load(QString(":/backgrounds/%1").arg(ThemeManager::getCatImage())); | ||||
|         m_catPixmap.load(APPLICATION->getCatPack()); | ||||
|     else | ||||
|         m_catPixmap = QPixmap(); | ||||
| } | ||||
|   | ||||
| @@ -30,7 +30,7 @@ | ||||
|      </property> | ||||
|      <widget class="QWidget" name="tab"> | ||||
|       <attribute name="title"> | ||||
|        <string notr="true">Services</string> | ||||
|        <string>Services</string> | ||||
|       </attribute> | ||||
|       <layout class="QVBoxLayout" name="verticalLayout_2"> | ||||
|        <item> | ||||
|   | ||||
| @@ -58,7 +58,7 @@ | ||||
|           <item row="2" column="0"> | ||||
|            <widget class="QLabel" name="labelPermGen"> | ||||
|             <property name="text"> | ||||
|              <string notr="true">&PermGen:</string> | ||||
|              <string>&PermGen:</string> | ||||
|             </property> | ||||
|             <property name="buddy"> | ||||
|              <cstring>permGenSpinBox</cstring> | ||||
|   | ||||
| @@ -62,7 +62,7 @@ public: | ||||
|  | ||||
|     QString displayName() const override | ||||
|     { | ||||
|         return "Launcher"; | ||||
|         return tr("Launcher"); | ||||
|     } | ||||
|     QIcon icon() const override | ||||
|     { | ||||
|   | ||||
| @@ -39,7 +39,7 @@ | ||||
|      </property> | ||||
|      <widget class="QWidget" name="minecraftTab"> | ||||
|       <attribute name="title"> | ||||
|        <string notr="true">General</string> | ||||
|        <string>General</string> | ||||
|       </attribute> | ||||
|       <layout class="QVBoxLayout" name="verticalLayout_3"> | ||||
|        <item> | ||||
|   | ||||
| @@ -116,7 +116,7 @@ | ||||
|           <item row="2" column="0"> | ||||
|            <widget class="QLabel" name="labelPermGen"> | ||||
|             <property name="text"> | ||||
|              <string notr="true">PermGen:</string> | ||||
|              <string>PermGen:</string> | ||||
|             </property> | ||||
|            </widget> | ||||
|           </item> | ||||
|   | ||||
| @@ -61,7 +61,7 @@ void ThemeWizardPage::updateIcons() | ||||
| void ThemeWizardPage::updateCat() | ||||
| { | ||||
|     qDebug() << "Setting Cat"; | ||||
|     ui->catImagePreviewButton->setIcon(QIcon(QString(R"(:/backgrounds/%1)").arg(ThemeManager::getCatImage()))); | ||||
|     ui->catImagePreviewButton->setIcon(QIcon(QString(R"(%1)").arg(APPLICATION->getCatPack()))); | ||||
| } | ||||
|  | ||||
| void ThemeWizardPage::retranslate() | ||||
|   | ||||
							
								
								
									
										117
									
								
								launcher/ui/themes/CatPack.cpp
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										117
									
								
								launcher/ui/themes/CatPack.cpp
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,117 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com> | ||||
|  * | ||||
|  *  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, see <https://www.gnu.org/licenses/>. | ||||
|  * | ||||
|  * 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 "ui/themes/CatPack.h" | ||||
| #include <QDate> | ||||
| #include <QDir> | ||||
| #include <QFileInfo> | ||||
| #include "FileSystem.h" | ||||
| #include "Json.h" | ||||
|  | ||||
| QString BasicCatPack::path() | ||||
| { | ||||
|     const auto now = QDate::currentDate(); | ||||
|     const auto birthday = QDate(now.year(), 11, 30); | ||||
|     const auto xmas = QDate(now.year(), 12, 25); | ||||
|     const auto halloween = QDate(now.year(), 10, 31); | ||||
|  | ||||
|     QString cat = QString(":/backgrounds/%1").arg(m_id); | ||||
|     if (std::abs(now.daysTo(xmas)) <= 4) { | ||||
|         cat += "-xmas"; | ||||
|     } else if (std::abs(now.daysTo(halloween)) <= 4) { | ||||
|         cat += "-spooky"; | ||||
|     } else if (std::abs(now.daysTo(birthday)) <= 12) { | ||||
|         cat += "-bday"; | ||||
|     } | ||||
|     return cat; | ||||
| } | ||||
|  | ||||
| JsonCatPack::PartialDate partialDate(QJsonObject date) | ||||
| { | ||||
|     auto month = Json::ensureInteger(date, "month", 1); | ||||
|     if (month > 12) | ||||
|         month = 12; | ||||
|     else if (month <= 0) | ||||
|         month = 1; | ||||
|     auto day = Json::ensureInteger(date, "day", 1); | ||||
|     if (day > 31) | ||||
|         day = 31; | ||||
|     else if (day <= 0) | ||||
|         day = 1; | ||||
|     return { month, day }; | ||||
| }; | ||||
|  | ||||
| JsonCatPack::JsonCatPack(QFileInfo& manifestInfo) : BasicCatPack(manifestInfo.dir().dirName()) | ||||
| { | ||||
|     QString path = manifestInfo.path(); | ||||
|     auto doc = Json::requireDocument(manifestInfo.absoluteFilePath(), "CatPack JSON file"); | ||||
|     const auto root = doc.object(); | ||||
|     m_name = Json::requireString(root, "name", "Catpack name"); | ||||
|     m_defaultPath = FS::PathCombine(path, Json::requireString(root, "default", "Default Cat")); | ||||
|     auto variants = Json::ensureArray(root, "variants", QJsonArray(), "Catpack Variants"); | ||||
|     for (auto v : variants) { | ||||
|         auto variant = Json::ensureObject(v, QJsonObject(), "Cat variant"); | ||||
|         m_variants << Variant{ FS::PathCombine(path, Json::requireString(variant, "path", "Variant path")), | ||||
|                                partialDate(Json::requireObject(variant, "startTime", "Variant startTime")), | ||||
|                                partialDate(Json::requireObject(variant, "endTime", "Variant endTime")) }; | ||||
|     } | ||||
| } | ||||
|  | ||||
| QDate ensureDay(int year, int month, int day) | ||||
| { | ||||
|     QDate date(year, month, 1); | ||||
|     if (day > date.daysInMonth()) | ||||
|         day = date.daysInMonth(); | ||||
|     return QDate(year, month, day); | ||||
| } | ||||
|  | ||||
| QString JsonCatPack::path() | ||||
| { | ||||
|     const QDate now = QDate::currentDate(); | ||||
|     for (auto var : m_variants) { | ||||
|         QDate startDate = ensureDay(now.year(), var.startTime.month, var.startTime.day); | ||||
|         QDate endDate = ensureDay(now.year(), var.endTime.month, var.endTime.day); | ||||
|         if (startDate > endDate) {  // it's spans over multiple years | ||||
|             if (endDate <= now)     // end date is in the past so jump one year into the future for endDate | ||||
|                 endDate = endDate.addYears(1); | ||||
|             else  // end date is in the future so jump one year into the past for startDate | ||||
|                 startDate = startDate.addYears(-1); | ||||
|         } | ||||
|  | ||||
|         if (startDate >= now && now >= endDate) | ||||
|             return var.path; | ||||
|     } | ||||
|     return m_defaultPath; | ||||
| } | ||||
							
								
								
									
										91
									
								
								launcher/ui/themes/CatPack.h
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										91
									
								
								launcher/ui/themes/CatPack.h
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,91 @@ | ||||
| // SPDX-License-Identifier: GPL-3.0-only | ||||
| /* | ||||
|  *  Prism Launcher - Minecraft Launcher | ||||
|  *  Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com> | ||||
|  * | ||||
|  *  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, see <https://www.gnu.org/licenses/>. | ||||
|  * | ||||
|  * 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. | ||||
|  */ | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include <QDate> | ||||
| #include <QFileInfo> | ||||
| #include <QList> | ||||
| #include <QString> | ||||
|  | ||||
| class CatPack { | ||||
|    public: | ||||
|     virtual ~CatPack() {} | ||||
|     virtual QString id() = 0; | ||||
|     virtual QString name() = 0; | ||||
|     virtual QString path() = 0; | ||||
| }; | ||||
|  | ||||
| class BasicCatPack : public CatPack { | ||||
|    public: | ||||
|     BasicCatPack(QString id, QString name) : m_id(id), m_name(name) {} | ||||
|     BasicCatPack(QString id) : BasicCatPack(id, id) {} | ||||
|     virtual QString id() { return m_id; }; | ||||
|     virtual QString name() { return m_name; }; | ||||
|     virtual QString path(); | ||||
|  | ||||
|    protected: | ||||
|     QString m_id; | ||||
|     QString m_name; | ||||
| }; | ||||
|  | ||||
| class FileCatPack : public BasicCatPack { | ||||
|    public: | ||||
|     FileCatPack(QString id, QFileInfo& fileInfo) : BasicCatPack(id), m_path(fileInfo.absoluteFilePath()) {} | ||||
|     FileCatPack(QFileInfo& fileInfo) : FileCatPack(fileInfo.baseName(), fileInfo) {} | ||||
|     virtual QString path() { return m_path; } | ||||
|  | ||||
|    private: | ||||
|     QString m_path; | ||||
| }; | ||||
|  | ||||
| class JsonCatPack : public BasicCatPack { | ||||
|    public: | ||||
|     struct PartialDate { | ||||
|         int month; | ||||
|         int day; | ||||
|     }; | ||||
|     struct Variant { | ||||
|         QString path; | ||||
|         PartialDate startTime; | ||||
|         PartialDate endTime; | ||||
|     }; | ||||
|     JsonCatPack(QFileInfo& manifestInfo); | ||||
|     virtual QString path(); | ||||
|  | ||||
|    private: | ||||
|     QString m_defaultPath; | ||||
|     QList<Variant> m_variants; | ||||
| }; | ||||
| @@ -21,7 +21,10 @@ | ||||
| #include <QDir> | ||||
| #include <QDirIterator> | ||||
| #include <QIcon> | ||||
| #include <QImageReader> | ||||
| #include "Exception.h" | ||||
| #include "ui/themes/BrightTheme.h" | ||||
| #include "ui/themes/CatPack.h" | ||||
| #include "ui/themes/CustomTheme.h" | ||||
| #include "ui/themes/DarkTheme.h" | ||||
| #include "ui/themes/SystemTheme.h" | ||||
| @@ -32,6 +35,7 @@ ThemeManager::ThemeManager(MainWindow* mainWindow) | ||||
| { | ||||
|     m_mainWindow = mainWindow; | ||||
|     initializeThemes(); | ||||
|     initializeCatPacks(); | ||||
| } | ||||
|  | ||||
| /// @brief Adds the Theme to the list of themes | ||||
| @@ -40,7 +44,10 @@ ThemeManager::ThemeManager(MainWindow* mainWindow) | ||||
| QString ThemeManager::addTheme(std::unique_ptr<ITheme> theme) | ||||
| { | ||||
|     QString id = theme->id(); | ||||
|     if (m_themes.find(id) == m_themes.end()) | ||||
|         m_themes.emplace(id, std::move(theme)); | ||||
|     else | ||||
|         themeWarningLog() << "Theme(" << id << ") not added to prevent id duplication"; | ||||
|     return id; | ||||
| } | ||||
|  | ||||
| @@ -77,7 +84,7 @@ void ThemeManager::initializeThemes() | ||||
|         QString themeFolder = QDir("./themes/").absoluteFilePath(""); | ||||
|         themeDebugLog() << "Theme Folder Path: " << themeFolder; | ||||
|  | ||||
|         QDirIterator directoryIterator(themeFolder, QDir::Dirs | QDir::NoDotAndDotDot, QDirIterator::Subdirectories); | ||||
|         QDirIterator directoryIterator(themeFolder, QDir::Dirs | QDir::NoDotAndDotDot); | ||||
|         while (directoryIterator.hasNext()) { | ||||
|             QDir dir(directoryIterator.next()); | ||||
|             QFileInfo themeJson(dir.absoluteFilePath("theme.json")); | ||||
| @@ -111,6 +118,16 @@ QList<ITheme*> ThemeManager::getValidApplicationThemes() | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| QList<CatPack*> ThemeManager::getValidCatPacks() | ||||
| { | ||||
|     QList<CatPack*> ret; | ||||
|     ret.reserve(m_catPacks.size()); | ||||
|     for (auto&& [id, theme] : m_catPacks) { | ||||
|         ret.append(theme.get()); | ||||
|     } | ||||
|     return ret; | ||||
| } | ||||
|  | ||||
| void ThemeManager::setIconTheme(const QString& name) | ||||
| { | ||||
|     QIcon::setThemeName(name); | ||||
| @@ -137,19 +154,74 @@ void ThemeManager::setApplicationTheme(const QString& name, bool initial) | ||||
|     } | ||||
| } | ||||
|  | ||||
| QString ThemeManager::getCatImage(QString catName) | ||||
| QString ThemeManager::getCatPack(QString catName) | ||||
| { | ||||
|     QDateTime now = QDateTime::currentDateTime(); | ||||
|     QDateTime birthday(QDate(now.date().year(), 11, 30), QTime(0, 0)); | ||||
|     QDateTime xmas(QDate(now.date().year(), 12, 25), QTime(0, 0)); | ||||
|     QDateTime halloween(QDate(now.date().year(), 10, 31), QTime(0, 0)); | ||||
|     QString cat = !catName.isEmpty() ? catName : APPLICATION->settings()->get("BackgroundCat").toString(); | ||||
|     if (std::abs(now.daysTo(xmas)) <= 4) { | ||||
|         cat += "-xmas"; | ||||
|     } else if (std::abs(now.daysTo(halloween)) <= 4) { | ||||
|         cat += "-spooky"; | ||||
|     } else if (std::abs(now.daysTo(birthday)) <= 12) { | ||||
|         cat += "-bday"; | ||||
|     auto catIter = m_catPacks.find(!catName.isEmpty() ? catName : APPLICATION->settings()->get("BackgroundCat").toString()); | ||||
|     if (catIter != m_catPacks.end()) { | ||||
|         auto& catPack = catIter->second; | ||||
|         themeDebugLog() << "applying catpack" << catPack->id(); | ||||
|         return catPack->path(); | ||||
|     } else { | ||||
|         themeWarningLog() << "Tried to get invalid catPack:" << catName; | ||||
|     } | ||||
|  | ||||
|     return m_catPacks.begin()->second->path(); | ||||
| } | ||||
|  | ||||
| QString ThemeManager::addCatPack(std::unique_ptr<CatPack> catPack) | ||||
| { | ||||
|     QString id = catPack->id(); | ||||
|     if (m_catPacks.find(id) == m_catPacks.end()) | ||||
|         m_catPacks.emplace(id, std::move(catPack)); | ||||
|     else | ||||
|         themeWarningLog() << "CatPack(" << id << ") not added to prevent id duplication"; | ||||
|     return id; | ||||
| } | ||||
|  | ||||
| void ThemeManager::initializeCatPacks() | ||||
| { | ||||
|     QList<std::pair<QString, QString>> defaultCats{ { "kitteh", QObject::tr("Background Cat (from MultiMC)") }, | ||||
|                                                     { "rory", QObject::tr("Rory ID 11 (drawn by Ashtaka)") }, | ||||
|                                                     { "rory-flat", QObject::tr("Rory ID 11 (flat edition, drawn by Ashtaka)") }, | ||||
|                                                     { "teawie", QObject::tr("Teawie (drawn by SympathyTea)") } }; | ||||
|     for (auto [id, name] : defaultCats) { | ||||
|         addCatPack(std::unique_ptr<CatPack>(new BasicCatPack(id, name))); | ||||
|     } | ||||
|     QDir catpacksDir("catpacks"); | ||||
|     QString catpacksFolder = catpacksDir.absoluteFilePath(""); | ||||
|     themeDebugLog() << "CatPacks Folder Path:" << catpacksFolder; | ||||
|  | ||||
|     QStringList supportedImageFormats; | ||||
|     for (auto format : QImageReader::supportedImageFormats()) { | ||||
|         supportedImageFormats.append("*." + format); | ||||
|     } | ||||
|     auto loadFiles = [this, supportedImageFormats](QDir dir) { | ||||
|         // Load image files directly | ||||
|         QDirIterator ImageFileIterator(dir.absoluteFilePath(""), supportedImageFormats, QDir::Files); | ||||
|         while (ImageFileIterator.hasNext()) { | ||||
|             QFile customCatFile(ImageFileIterator.next()); | ||||
|             QFileInfo customCatFileInfo(customCatFile); | ||||
|             themeDebugLog() << "Loading CatPack from:" << customCatFileInfo.absoluteFilePath(); | ||||
|             addCatPack(std::unique_ptr<CatPack>(new FileCatPack(customCatFileInfo))); | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     loadFiles(catpacksDir); | ||||
|  | ||||
|     QDirIterator directoryIterator(catpacksFolder, QDir::Dirs | QDir::NoDotAndDotDot); | ||||
|     while (directoryIterator.hasNext()) { | ||||
|         QDir dir(directoryIterator.next()); | ||||
|         QFileInfo manifest(dir.absoluteFilePath("catpack.json")); | ||||
|         if (manifest.isFile()) { | ||||
|             try { | ||||
|                 // Load background manifest | ||||
|                 themeDebugLog() << "Loading background manifest from:" << manifest.absoluteFilePath(); | ||||
|                 addCatPack(std::unique_ptr<CatPack>(new JsonCatPack(manifest))); | ||||
|             } catch (const Exception& e) { | ||||
|                 themeWarningLog() << "Couldn't load catpack json:" << e.cause(); | ||||
|             } | ||||
|         } else { | ||||
|             loadFiles(dir); | ||||
|         } | ||||
|     } | ||||
|     return cat; | ||||
| } | ||||
|   | ||||
| @@ -20,6 +20,7 @@ | ||||
| #include <QString> | ||||
|  | ||||
| #include "ui/MainWindow.h" | ||||
| #include "ui/themes/CatPack.h" | ||||
| #include "ui/themes/ITheme.h" | ||||
|  | ||||
| inline auto themeDebugLog() | ||||
| @@ -40,18 +41,20 @@ class ThemeManager { | ||||
|     void applyCurrentlySelectedTheme(bool initial = false); | ||||
|     void setApplicationTheme(const QString& name, bool initial = false); | ||||
|  | ||||
|     /// <summary> | ||||
|     /// Returns the cat based on selected cat and with events (Birthday, XMas, etc.) | ||||
|     /// </summary> | ||||
|     /// <param name="catName">Optional, if you need a specific cat.</param> | ||||
|     /// <returns></returns> | ||||
|     static QString getCatImage(QString catName = ""); | ||||
|     /// @brief Returns the background based on selected and with events (Birthday, XMas, etc.) | ||||
|     /// @param catName Optional, if you need a specific background. | ||||
|     /// @return | ||||
|     QString getCatPack(QString catName = ""); | ||||
|     QList<CatPack*> getValidCatPacks(); | ||||
|  | ||||
|    private: | ||||
|     std::map<QString, std::unique_ptr<ITheme>> m_themes; | ||||
|     std::map<QString, std::unique_ptr<CatPack>> m_catPacks; | ||||
|     MainWindow* m_mainWindow; | ||||
|  | ||||
|     void initializeThemes(); | ||||
|     void initializeCatPacks(); | ||||
|     QString addTheme(std::unique_ptr<ITheme> theme); | ||||
|     ITheme* getTheme(QString themeId); | ||||
|     QString addCatPack(std::unique_ptr<CatPack> catPack); | ||||
| }; | ||||
|   | ||||
| @@ -95,9 +95,14 @@ void ThemeCustomizationWidget::applyWidgetTheme(int index) { | ||||
|     emit currentWidgetThemeChanged(index); | ||||
| } | ||||
|  | ||||
| void ThemeCustomizationWidget::applyCatTheme(int index) { | ||||
| void ThemeCustomizationWidget::applyCatTheme(int index) | ||||
| { | ||||
|     auto settings = APPLICATION->settings(); | ||||
|     settings->set("BackgroundCat", m_catOptions[index].first); | ||||
|     auto originalCat = settings->get("BackgroundCat").toString(); | ||||
|     auto newCat = ui->backgroundCatComboBox->currentData().toString(); | ||||
|     if (originalCat != newCat) { | ||||
|         settings->set("BackgroundCat", newCat); | ||||
|     } | ||||
|  | ||||
|     emit currentCatChanged(index); | ||||
| } | ||||
| @@ -135,10 +140,10 @@ void ThemeCustomizationWidget::loadSettings() | ||||
|     } | ||||
|  | ||||
|     auto cat = settings->get("BackgroundCat").toString(); | ||||
|     for (auto& catFromList : m_catOptions) { | ||||
|         QIcon catIcon = QIcon(QString(":/backgrounds/%1").arg(ThemeManager::getCatImage(catFromList.first))); | ||||
|         ui->backgroundCatComboBox->addItem(catIcon, catFromList.second); | ||||
|         if (cat == catFromList.first) { | ||||
|     for (auto& catFromList : APPLICATION->getValidCatPacks()) { | ||||
|         QIcon catIcon = QIcon(QString("%1").arg(catFromList->path())); | ||||
|         ui->backgroundCatComboBox->addItem(catIcon, catFromList->name(), catFromList->id()); | ||||
|         if (cat == catFromList->id()) { | ||||
|             ui->backgroundCatComboBox->setCurrentIndex(ui->backgroundCatComboBox->count() - 1); | ||||
|         } | ||||
|     } | ||||
|   | ||||
| @@ -54,8 +54,7 @@ class ThemeCustomizationWidget : public QWidget { | ||||
|     Ui::ThemeCustomizationWidget* ui; | ||||
|  | ||||
|     // TODO finish implementing | ||||
|     QList<std::pair<QString, QString>> m_iconThemeOptions{  | ||||
|         { "pe_colored",     QObject::tr("Simple (Colored Icons)") },  | ||||
|     QList<std::pair<QString, QString>> m_iconThemeOptions{ { "pe_colored", QObject::tr("Simple (Colored Icons)") }, | ||||
|                                                            { "pe_light", QObject::tr("Simple (Light Icons)") }, | ||||
|                                                            { "pe_dark", QObject::tr("Simple (Dark Icons)") }, | ||||
|                                                            { "pe_blue", QObject::tr("Simple (Blue Icons)") }, | ||||
| @@ -66,12 +65,5 @@ class ThemeCustomizationWidget : public QWidget { | ||||
|                                                            { "flat", QObject::tr("Flat") }, | ||||
|                                                            { "flat_white", QObject::tr("Flat (White)") }, | ||||
|                                                            { "multimc", QObject::tr("Legacy") }, | ||||
|         { "custom",         QObject::tr("Custom") }  | ||||
|     }; | ||||
|     QList<std::pair<QString, QString>> m_catOptions{  | ||||
|         { "kitteh",     QObject::tr("Background Cat (from MultiMC)") },  | ||||
|         { "rory",       QObject::tr("Rory ID 11 (drawn by Ashtaka)") },  | ||||
|         { "rory-flat",  QObject::tr("Rory ID 11 (flat edition, drawn by Ashtaka)") }, | ||||
|         { "teawie",     QObject::tr("Teawie (drawn by SympathyTea)") } | ||||
|     }; | ||||
|                                                            { "custom", QObject::tr("Custom") } }; | ||||
| }; | ||||
|   | ||||
 Submodule libraries/libnbtplusplus updated: 2203af7eeb...a5e8fd52b8
									
								
							
		Reference in New Issue
	
	Block a user
	 Trial97
					Trial97