Merge branch 'develop' of https://github.com/PrismLauncher/PrismLauncher into settings
Signed-off-by: Trial97 <alexandru.tripon97@gmail.com>
This commit is contained in:
commit
bc2940e16a
26
.github/workflows/update-flake.yml
vendored
Normal file
26
.github/workflows/update-flake.yml
vendored
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
name: Update Flake Lockfile
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
# run weekly on sunday
|
||||||
|
- cron: "0 0 * * 0"
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
pull-requests: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
update-flake:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- uses: actions/checkout@v3
|
||||||
|
- uses: cachix/install-nix-action@v22
|
||||||
|
|
||||||
|
- uses: DeterminateSystems/update-flake-lock@v19
|
||||||
|
with:
|
||||||
|
commit-msg: "chore(nix): update lockfile"
|
||||||
|
pr-title: "chore(nix): update lockfile"
|
||||||
|
pr-labels: |
|
||||||
|
Linux
|
||||||
|
simple change
|
48
flake.lock
generated
48
flake.lock
generated
@ -21,11 +21,11 @@
|
|||||||
"nixpkgs-lib": "nixpkgs-lib"
|
"nixpkgs-lib": "nixpkgs-lib"
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1683560683,
|
"lastModified": 1688254665,
|
||||||
"narHash": "sha256-XAygPMN5Xnk/W2c1aW0jyEa6lfMDZWlQgiNtmHXytPc=",
|
"narHash": "sha256-8FHEgBrr7gYNiS/NzCxIO3m4hvtLRW9YY1nYo1ivm3o=",
|
||||||
"owner": "hercules-ci",
|
"owner": "hercules-ci",
|
||||||
"repo": "flake-parts",
|
"repo": "flake-parts",
|
||||||
"rev": "006c75898cf814ef9497252b022e91c946ba8e17",
|
"rev": "267149c58a14d15f7f81b4d737308421de9d7152",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@ -35,12 +35,15 @@
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
"flake-utils": {
|
"flake-utils": {
|
||||||
|
"inputs": {
|
||||||
|
"systems": "systems"
|
||||||
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1667395993,
|
"lastModified": 1685518550,
|
||||||
"narHash": "sha256-nuEHfE/LcWyuSWnS8t12N1wc105Qtau+/OdUAjtQ0rA=",
|
"narHash": "sha256-o2d0KcvaXzTrPRIo0kOLV0/QXHhDQ5DTi+OxcjO8xqY=",
|
||||||
"owner": "numtide",
|
"owner": "numtide",
|
||||||
"repo": "flake-utils",
|
"repo": "flake-utils",
|
||||||
"rev": "5aed5285a952e0b949eb3ba02c12fa4fcfef535f",
|
"rev": "a1720a10a6cfe8234c0e93907ffe81be440f4cef",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@ -88,11 +91,11 @@
|
|||||||
},
|
},
|
||||||
"nixpkgs": {
|
"nixpkgs": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1685012353,
|
"lastModified": 1688221086,
|
||||||
"narHash": "sha256-U3oOge4cHnav8OLGdRVhL45xoRj4Ppd+It6nPC9nNIU=",
|
"narHash": "sha256-cdW6qUL71cNWhHCpMPOJjlw0wzSRP0pVlRn2vqX/VVg=",
|
||||||
"owner": "nixos",
|
"owner": "nixos",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "aeb75dba965e790de427b73315d5addf91a54955",
|
"rev": "cd99c2b3c9f160cd004318e0697f90bbd5960825",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@ -105,11 +108,11 @@
|
|||||||
"nixpkgs-lib": {
|
"nixpkgs-lib": {
|
||||||
"locked": {
|
"locked": {
|
||||||
"dir": "lib",
|
"dir": "lib",
|
||||||
"lastModified": 1682879489,
|
"lastModified": 1688049487,
|
||||||
"narHash": "sha256-sASwo8gBt7JDnOOstnps90K1wxmVfyhsTPPNTGBPjjg=",
|
"narHash": "sha256-100g4iaKC9MalDjUW9iN6Jl/OocTDtXdeAj7pEGIRh4=",
|
||||||
"owner": "NixOS",
|
"owner": "NixOS",
|
||||||
"repo": "nixpkgs",
|
"repo": "nixpkgs",
|
||||||
"rev": "da45bf6ec7bbcc5d1e14d3795c025199f28e0de0",
|
"rev": "4bc72cae107788bf3f24f30db2e2f685c9298dc9",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@ -135,11 +138,11 @@
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"locked": {
|
"locked": {
|
||||||
"lastModified": 1684842236,
|
"lastModified": 1688386108,
|
||||||
"narHash": "sha256-rYWsIXHvNhVQ15RQlBUv67W3YnM+Pd+DuXGMvCBq2IE=",
|
"narHash": "sha256-Vffto9QaVonzYAcPlAzd0soqWYpPpKk60dfNLSIXcFA=",
|
||||||
"owner": "cachix",
|
"owner": "cachix",
|
||||||
"repo": "pre-commit-hooks.nix",
|
"repo": "pre-commit-hooks.nix",
|
||||||
"rev": "61e567d6497bc9556f391faebe5e410e6623217f",
|
"rev": "42587d3414d1747999a5f71e92a83cf6547b62da",
|
||||||
"type": "github"
|
"type": "github"
|
||||||
},
|
},
|
||||||
"original": {
|
"original": {
|
||||||
@ -156,6 +159,21 @@
|
|||||||
"nixpkgs": "nixpkgs",
|
"nixpkgs": "nixpkgs",
|
||||||
"pre-commit-hooks": "pre-commit-hooks"
|
"pre-commit-hooks": "pre-commit-hooks"
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
"systems": {
|
||||||
|
"locked": {
|
||||||
|
"lastModified": 1681028828,
|
||||||
|
"narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=",
|
||||||
|
"owner": "nix-systems",
|
||||||
|
"repo": "default",
|
||||||
|
"rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e",
|
||||||
|
"type": "github"
|
||||||
|
},
|
||||||
|
"original": {
|
||||||
|
"owner": "nix-systems",
|
||||||
|
"repo": "default",
|
||||||
|
"type": "github"
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"root": "root",
|
"root": "root",
|
||||||
|
@ -687,9 +687,17 @@ Application::Application(int &argc, char **argv) : QApplication(argc, argv)
|
|||||||
m_settings->reset("PastebinCustomAPIBase");
|
m_settings->reset("PastebinCustomAPIBase");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// meta URL
|
{
|
||||||
|
// Meta URL
|
||||||
m_settings->registerSetting("MetaURLOverride", "");
|
m_settings->registerSetting("MetaURLOverride", "");
|
||||||
|
|
||||||
|
QUrl metaUrl(m_settings->get("MetaURLOverride").toString());
|
||||||
|
|
||||||
|
// get rid of invalid meta urls
|
||||||
|
if (!metaUrl.isValid() || metaUrl.scheme() != "http" || metaUrl.scheme() != "https")
|
||||||
|
m_settings->reset("MetaURLOverride");
|
||||||
|
}
|
||||||
|
|
||||||
m_settings->registerSetting("CloseAfterLaunch", false);
|
m_settings->registerSetting("CloseAfterLaunch", false);
|
||||||
m_settings->registerSetting("QuitAfterGameStop", false);
|
m_settings->registerSetting("QuitAfterGameStop", false);
|
||||||
|
|
||||||
|
@ -36,6 +36,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "FileSystem.h"
|
#include "FileSystem.h"
|
||||||
|
#include <QPair>
|
||||||
|
|
||||||
#include "BuildConfig.h"
|
#include "BuildConfig.h"
|
||||||
|
|
||||||
@ -246,6 +247,7 @@ bool copy::operator()(const QString& offset, bool dryRun)
|
|||||||
{
|
{
|
||||||
using copy_opts = fs::copy_options;
|
using copy_opts = fs::copy_options;
|
||||||
m_copied = 0; // reset counter
|
m_copied = 0; // reset counter
|
||||||
|
m_failedPaths.clear();
|
||||||
|
|
||||||
// NOTE always deep copy on windows. the alternatives are too messy.
|
// NOTE always deep copy on windows. the alternatives are too messy.
|
||||||
#if defined Q_OS_WIN32
|
#if defined Q_OS_WIN32
|
||||||
@ -277,6 +279,9 @@ bool copy::operator()(const QString& offset, bool dryRun)
|
|||||||
qWarning() << "Failed to copy files:" << QString::fromStdString(err.message());
|
qWarning() << "Failed to copy files:" << QString::fromStdString(err.message());
|
||||||
qDebug() << "Source file:" << src_path;
|
qDebug() << "Source file:" << src_path;
|
||||||
qDebug() << "Destination file:" << dst_path;
|
qDebug() << "Destination file:" << dst_path;
|
||||||
|
m_failedPaths.append(dst_path);
|
||||||
|
emit copyFailed(relative_dst_path);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
m_copied++;
|
m_copied++;
|
||||||
emit fileCopied(relative_dst_path);
|
emit fileCopied(relative_dst_path);
|
||||||
@ -1077,6 +1082,7 @@ bool clone::operator()(const QString& offset, bool dryRun)
|
|||||||
}
|
}
|
||||||
|
|
||||||
m_cloned = 0; // reset counter
|
m_cloned = 0; // reset counter
|
||||||
|
m_failedClones.clear();
|
||||||
|
|
||||||
auto src = PathCombine(m_src.absolutePath(), offset);
|
auto src = PathCombine(m_src.absolutePath(), offset);
|
||||||
auto dst = PathCombine(m_dst.absolutePath(), offset);
|
auto dst = PathCombine(m_dst.absolutePath(), offset);
|
||||||
@ -1097,6 +1103,9 @@ bool clone::operator()(const QString& offset, bool dryRun)
|
|||||||
qDebug() << "Failed to clone files: error" << err.value() << "message" << QString::fromStdString(err.message());
|
qDebug() << "Failed to clone files: error" << err.value() << "message" << QString::fromStdString(err.message());
|
||||||
qDebug() << "Source file:" << src_path;
|
qDebug() << "Source file:" << src_path;
|
||||||
qDebug() << "Destination file:" << dst_path;
|
qDebug() << "Destination file:" << dst_path;
|
||||||
|
m_failedClones.append(qMakePair(src_path, dst_path));
|
||||||
|
emit cloneFailed(src_path, dst_path);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
m_cloned++;
|
m_cloned++;
|
||||||
emit fileCloned(src_path, dst_path);
|
emit fileCloned(src_path, dst_path);
|
||||||
|
@ -43,6 +43,7 @@
|
|||||||
#include <system_error>
|
#include <system_error>
|
||||||
|
|
||||||
#include <QDir>
|
#include <QDir>
|
||||||
|
#include <QPair>
|
||||||
#include <QFlags>
|
#include <QFlags>
|
||||||
#include <QLocalServer>
|
#include <QLocalServer>
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
@ -112,9 +113,12 @@ class copy : public QObject {
|
|||||||
bool operator()(bool dryRun = false) { return operator()(QString(), dryRun); }
|
bool operator()(bool dryRun = false) { return operator()(QString(), dryRun); }
|
||||||
|
|
||||||
int totalCopied() { return m_copied; }
|
int totalCopied() { return m_copied; }
|
||||||
|
int totalFailed() { return m_failedPaths.length(); }
|
||||||
|
QStringList failed() { return m_failedPaths; }
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void fileCopied(const QString& relativeName);
|
void fileCopied(const QString& relativeName);
|
||||||
|
void copyFailed(const QString& relativeName);
|
||||||
// TODO: maybe add a "shouldCopy" signal in the future?
|
// TODO: maybe add a "shouldCopy" signal in the future?
|
||||||
|
|
||||||
private:
|
private:
|
||||||
@ -127,6 +131,7 @@ class copy : public QObject {
|
|||||||
QDir m_src;
|
QDir m_src;
|
||||||
QDir m_dst;
|
QDir m_dst;
|
||||||
int m_copied;
|
int m_copied;
|
||||||
|
QStringList m_failedPaths;
|
||||||
};
|
};
|
||||||
|
|
||||||
struct LinkPair {
|
struct LinkPair {
|
||||||
@ -471,6 +476,9 @@ class clone : public QObject {
|
|||||||
bool operator()(bool dryRun = false) { return operator()(QString(), dryRun); }
|
bool operator()(bool dryRun = false) { return operator()(QString(), dryRun); }
|
||||||
|
|
||||||
int totalCloned() { return m_cloned; }
|
int totalCloned() { return m_cloned; }
|
||||||
|
int totalFailed() { return m_failedClones.length(); }
|
||||||
|
|
||||||
|
QList<QPair<QString, QString>> failed() { return m_failedClones; }
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void fileCloned(const QString& src, const QString& dst);
|
void fileCloned(const QString& src, const QString& dst);
|
||||||
@ -485,6 +493,7 @@ class clone : public QObject {
|
|||||||
QDir m_src;
|
QDir m_src;
|
||||||
QDir m_dst;
|
QDir m_dst;
|
||||||
int m_cloned;
|
int m_cloned;
|
||||||
|
QList<QPair<QString, QString>> m_failedClones;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -187,8 +187,8 @@ void LaunchController::login() {
|
|||||||
switch(m_accountToUse->accountState()) {
|
switch(m_accountToUse->accountState()) {
|
||||||
case AccountState::Offline: {
|
case AccountState::Offline: {
|
||||||
m_session->wants_online = false;
|
m_session->wants_online = false;
|
||||||
// NOTE: fallthrough is intentional
|
|
||||||
}
|
}
|
||||||
|
/* fallthrough */
|
||||||
case AccountState::Online: {
|
case AccountState::Online: {
|
||||||
if(!m_session->wants_online) {
|
if(!m_session->wants_online) {
|
||||||
// we ask the user for a player name
|
// we ask the user for a player name
|
||||||
@ -267,8 +267,8 @@ void LaunchController::login() {
|
|||||||
// This means some sort of soft error that we can fix with a refresh ... so let's refresh.
|
// This means some sort of soft error that we can fix with a refresh ... so let's refresh.
|
||||||
case AccountState::Unchecked: {
|
case AccountState::Unchecked: {
|
||||||
m_accountToUse->refresh();
|
m_accountToUse->refresh();
|
||||||
// NOTE: fallthrough intentional
|
|
||||||
}
|
}
|
||||||
|
/* fallthrough */
|
||||||
case AccountState::Working: {
|
case AccountState::Working: {
|
||||||
// refresh is in progress, we need to wait for it to finish to proceed.
|
// refresh is in progress, we need to wait for it to finish to proceed.
|
||||||
ProgressDialog progDialog(m_parentWidget);
|
ProgressDialog progDialog(m_parentWidget);
|
||||||
|
@ -193,33 +193,23 @@ QVariant VersionProxyModel::data(const QModelIndex &index, int role) const
|
|||||||
}
|
}
|
||||||
case Qt::ToolTipRole:
|
case Qt::ToolTipRole:
|
||||||
{
|
{
|
||||||
switch(column)
|
if(column == Name && hasRecommended)
|
||||||
{
|
|
||||||
case Name:
|
|
||||||
{
|
|
||||||
if(hasRecommended)
|
|
||||||
{
|
{
|
||||||
auto value = sourceModel()->data(parentIndex, BaseVersionList::RecommendedRole);
|
auto value = sourceModel()->data(parentIndex, BaseVersionList::RecommendedRole);
|
||||||
if(value.toBool())
|
if(value.toBool())
|
||||||
{
|
{
|
||||||
return tr("Recommended");
|
return tr("Recommended");
|
||||||
}
|
} else if(hasLatest) {
|
||||||
else if(hasLatest)
|
|
||||||
{
|
|
||||||
auto value = sourceModel()->data(parentIndex, BaseVersionList::LatestRole);
|
auto value = sourceModel()->data(parentIndex, BaseVersionList::LatestRole);
|
||||||
if(value.toBool())
|
if(value.toBool())
|
||||||
{
|
{
|
||||||
return tr("Latest");
|
return tr("Latest");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
} else {
|
||||||
}
|
|
||||||
default:
|
|
||||||
{
|
|
||||||
return sourceModel()->data(parentIndex, BaseVersionList::VersionIdRole);
|
return sourceModel()->data(parentIndex, BaseVersionList::VersionIdRole);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
case Qt::DecorationRole:
|
case Qt::DecorationRole:
|
||||||
{
|
{
|
||||||
switch(column)
|
switch(column)
|
||||||
|
@ -45,10 +45,10 @@ QVariant Index::data(const QModelIndex &index, int role) const
|
|||||||
switch (role)
|
switch (role)
|
||||||
{
|
{
|
||||||
case Qt::DisplayRole:
|
case Qt::DisplayRole:
|
||||||
switch (index.column())
|
if (index.column() == 0) {
|
||||||
{
|
return list->humanReadable();
|
||||||
case 0: return list->humanReadable();
|
} else {
|
||||||
default: break;
|
break;
|
||||||
}
|
}
|
||||||
case UidRole: return list->uid();
|
case UidRole: return list->uid();
|
||||||
case NameRole: return list->name();
|
case NameRole: return list->name();
|
||||||
|
@ -328,6 +328,9 @@ QVariant AccountList::data(const QModelIndex &index, int role) const
|
|||||||
case AccountState::Gone: {
|
case AccountState::Gone: {
|
||||||
return tr("Gone", "Account status");
|
return tr("Gone", "Account status");
|
||||||
}
|
}
|
||||||
|
default: {
|
||||||
|
return tr("Unknown", "Account status");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -354,12 +357,13 @@ QVariant AccountList::data(const QModelIndex &index, int role) const
|
|||||||
return QVariant::fromValue(account);
|
return QVariant::fromValue(account);
|
||||||
|
|
||||||
case Qt::CheckStateRole:
|
case Qt::CheckStateRole:
|
||||||
switch (index.column())
|
if (index.column() == ProfileNameColumn) {
|
||||||
{
|
|
||||||
case ProfileNameColumn:
|
|
||||||
return account == m_defaultAccount ? Qt::Checked : Qt::Unchecked;
|
return account == m_defaultAccount ? Qt::Checked : Qt::Unchecked;
|
||||||
|
} else {
|
||||||
|
return QVariant();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return QVariant();
|
return QVariant();
|
||||||
}
|
}
|
||||||
|
@ -273,6 +273,7 @@ void Yggdrasil::processReply() {
|
|||||||
AccountTaskState::STATE_FAILED_GONE,
|
AccountTaskState::STATE_FAILED_GONE,
|
||||||
tr("The Mojang account no longer exists. It may have been migrated to a Microsoft account.")
|
tr("The Mojang account no longer exists. It may have been migrated to a Microsoft account.")
|
||||||
);
|
);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
default:
|
default:
|
||||||
changeState(
|
changeState(
|
||||||
|
@ -74,6 +74,7 @@ std::pair<int, bool> DataPack::compare(const Resource& other, SortType type) con
|
|||||||
auto res = Resource::compare(other, type);
|
auto res = Resource::compare(other, type);
|
||||||
if (res.first != 0)
|
if (res.first != 0)
|
||||||
return res;
|
return res;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
case SortType::PACK_FORMAT: {
|
case SortType::PACK_FORMAT: {
|
||||||
auto this_ver = packFormat();
|
auto this_ver = packFormat();
|
||||||
@ -83,6 +84,7 @@ std::pair<int, bool> DataPack::compare(const Resource& other, SortType type) con
|
|||||||
return { 1, type == SortType::PACK_FORMAT };
|
return { 1, type == SortType::PACK_FORMAT };
|
||||||
if (this_ver < other_ver)
|
if (this_ver < other_ver)
|
||||||
return { -1, type == SortType::PACK_FORMAT };
|
return { -1, type == SortType::PACK_FORMAT };
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return { 0, false };
|
return { 0, false };
|
||||||
|
@ -91,6 +91,7 @@ std::pair<int, bool> Mod::compare(const Resource& other, SortType type) const
|
|||||||
auto res = Resource::compare(other, type);
|
auto res = Resource::compare(other, type);
|
||||||
if (res.first != 0)
|
if (res.first != 0)
|
||||||
return res;
|
return res;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
case SortType::VERSION: {
|
case SortType::VERSION: {
|
||||||
auto this_ver = Version(version());
|
auto this_ver = Version(version());
|
||||||
@ -99,11 +100,13 @@ std::pair<int, bool> Mod::compare(const Resource& other, SortType type) const
|
|||||||
return { 1, type == SortType::VERSION };
|
return { 1, type == SortType::VERSION };
|
||||||
if (this_ver < other_ver)
|
if (this_ver < other_ver)
|
||||||
return { -1, type == SortType::VERSION };
|
return { -1, type == SortType::VERSION };
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
case SortType::PROVIDER: {
|
case SortType::PROVIDER: {
|
||||||
auto compare_result = QString::compare(provider().value_or("Unknown"), cast_other->provider().value_or("Unknown"), Qt::CaseInsensitive);
|
auto compare_result = QString::compare(provider().value_or("Unknown"), cast_other->provider().value_or("Unknown"), Qt::CaseInsensitive);
|
||||||
if (compare_result != 0)
|
if (compare_result != 0)
|
||||||
return { compare_result, type == SortType::PROVIDER };
|
return { compare_result, type == SortType::PROVIDER };
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return { 0, false };
|
return { 0, false };
|
||||||
@ -166,6 +169,13 @@ auto Mod::homeurl() const -> QString
|
|||||||
return details().homeurl;
|
return details().homeurl;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
auto Mod::metaurl() const -> QString
|
||||||
|
{
|
||||||
|
if (metadata() == nullptr)
|
||||||
|
return homeurl();
|
||||||
|
return ModPlatform::getMetaURL(metadata()->provider, metadata()->project_id);
|
||||||
|
}
|
||||||
|
|
||||||
auto Mod::description() const -> QString
|
auto Mod::description() const -> QString
|
||||||
{
|
{
|
||||||
return details().description;
|
return details().description;
|
||||||
|
@ -70,6 +70,7 @@ public:
|
|||||||
auto provider() const -> std::optional<QString>;
|
auto provider() const -> std::optional<QString>;
|
||||||
auto licenses() const -> const QList<ModLicense>&;
|
auto licenses() const -> const QList<ModLicense>&;
|
||||||
auto issueTracker() const -> QString;
|
auto issueTracker() const -> QString;
|
||||||
|
auto metaurl() const -> QString;
|
||||||
|
|
||||||
/** Get the intneral path to the mod's icon file*/
|
/** Get the intneral path to the mod's icon file*/
|
||||||
QString iconPath() const { return m_local_details.icon_file; };
|
QString iconPath() const { return m_local_details.icon_file; };
|
||||||
|
@ -71,6 +71,7 @@ std::pair<int, bool> Resource::compare(const Resource& other, SortType type) con
|
|||||||
return { 1, type == SortType::ENABLED };
|
return { 1, type == SortType::ENABLED };
|
||||||
if (!enabled() && other.enabled())
|
if (!enabled() && other.enabled())
|
||||||
return { -1, type == SortType::ENABLED };
|
return { -1, type == SortType::ENABLED };
|
||||||
|
break;
|
||||||
case SortType::NAME: {
|
case SortType::NAME: {
|
||||||
QString this_name{ name() };
|
QString this_name{ name() };
|
||||||
QString other_name{ other.name() };
|
QString other_name{ other.name() };
|
||||||
@ -81,12 +82,14 @@ std::pair<int, bool> Resource::compare(const Resource& other, SortType type) con
|
|||||||
auto compare_result = QString::compare(this_name, other_name, Qt::CaseInsensitive);
|
auto compare_result = QString::compare(this_name, other_name, Qt::CaseInsensitive);
|
||||||
if (compare_result != 0)
|
if (compare_result != 0)
|
||||||
return { compare_result, type == SortType::NAME };
|
return { compare_result, type == SortType::NAME };
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
case SortType::DATE:
|
case SortType::DATE:
|
||||||
if (dateTimeChanged() > other.dateTimeChanged())
|
if (dateTimeChanged() > other.dateTimeChanged())
|
||||||
return { 1, type == SortType::DATE };
|
return { 1, type == SortType::DATE };
|
||||||
if (dateTimeChanged() < other.dateTimeChanged())
|
if (dateTimeChanged() < other.dateTimeChanged())
|
||||||
return { -1, type == SortType::DATE };
|
return { -1, type == SortType::DATE };
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
return { 0, false };
|
return { 0, false };
|
||||||
|
@ -102,6 +102,7 @@ std::pair<int, bool> ResourcePack::compare(const Resource& other, SortType type)
|
|||||||
auto res = Resource::compare(other, type);
|
auto res = Resource::compare(other, type);
|
||||||
if (res.first != 0)
|
if (res.first != 0)
|
||||||
return res;
|
return res;
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
case SortType::PACK_FORMAT: {
|
case SortType::PACK_FORMAT: {
|
||||||
auto this_ver = packFormat();
|
auto this_ver = packFormat();
|
||||||
@ -111,6 +112,7 @@ std::pair<int, bool> ResourcePack::compare(const Resource& other, SortType type)
|
|||||||
return { 1, type == SortType::PACK_FORMAT };
|
return { 1, type == SortType::PACK_FORMAT };
|
||||||
if (this_ver < other_ver)
|
if (this_ver < other_ver)
|
||||||
return { -1, type == SortType::PACK_FORMAT };
|
return { -1, type == SortType::PACK_FORMAT };
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return { 0, false };
|
return { 0, false };
|
||||||
|
@ -77,4 +77,10 @@ auto ProviderCapabilities::hash(ResourceProvider p, QIODevice* device, QString t
|
|||||||
return { hash.result().toHex() };
|
return { hash.result().toHex() };
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QString getMetaURL(ResourceProvider provider, QVariant projectID)
|
||||||
|
{
|
||||||
|
return ((provider == ModPlatform::ResourceProvider::FLAME) ? "https://www.curseforge.com/projects/" : "https://modrinth.com/mod/") +
|
||||||
|
projectID.toString();
|
||||||
|
}
|
||||||
|
|
||||||
} // namespace ModPlatform
|
} // namespace ModPlatform
|
||||||
|
@ -144,6 +144,7 @@ inline auto getOverrideDeps() -> QList<OverrideDep>
|
|||||||
{ "qvIfYCYJ", "P7dR8mSH", "API", ModPlatform::ResourceProvider::MODRINTH },
|
{ "qvIfYCYJ", "P7dR8mSH", "API", ModPlatform::ResourceProvider::MODRINTH },
|
||||||
{ "lwVhp9o5", "Ha28R6CL", "KotlinLibraries", ModPlatform::ResourceProvider::MODRINTH } };
|
{ "lwVhp9o5", "Ha28R6CL", "KotlinLibraries", ModPlatform::ResourceProvider::MODRINTH } };
|
||||||
};
|
};
|
||||||
|
QString getMetaURL(ResourceProvider provider, QVariant projectID);
|
||||||
|
|
||||||
} // namespace ModPlatform
|
} // namespace ModPlatform
|
||||||
|
|
||||||
|
@ -470,8 +470,9 @@ void FlameCreationTask::setupDownloadJob(QEventLoop& loop)
|
|||||||
switch (result.type) {
|
switch (result.type) {
|
||||||
case Flame::File::Type::Folder: {
|
case Flame::File::Type::Folder: {
|
||||||
logWarning(tr("This 'Folder' may need extracting: %1").arg(relpath));
|
logWarning(tr("This 'Folder' may need extracting: %1").arg(relpath));
|
||||||
// fall-through intentional, we treat these as plain old mods and dump them wherever.
|
// fallthrough intentional, we treat these as plain old mods and dump them wherever.
|
||||||
}
|
}
|
||||||
|
/* fallthrough */
|
||||||
case Flame::File::Type::SingleFile:
|
case Flame::File::Type::SingleFile:
|
||||||
case Flame::File::Type::Mod: {
|
case Flame::File::Type::Mod: {
|
||||||
if (!result.url.isEmpty()) {
|
if (!result.url.isEmpty()) {
|
||||||
|
@ -454,6 +454,7 @@ QVariant TranslationsModel::data(const QModelIndex& index, int role) const
|
|||||||
return QString("%1%").arg(lang.percentTranslated(), 3, 'f', 1);
|
return QString("%1%").arg(lang.percentTranslated(), 3, 'f', 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
qWarning("TranslationModel::data not implemented when role is DisplayRole");
|
||||||
}
|
}
|
||||||
case Qt::ToolTipRole:
|
case Qt::ToolTipRole:
|
||||||
{
|
{
|
||||||
|
@ -927,21 +927,8 @@ void MainWindow::onCatToggled(bool state)
|
|||||||
|
|
||||||
void MainWindow::setCatBackground(bool enabled)
|
void MainWindow::setCatBackground(bool enabled)
|
||||||
{
|
{
|
||||||
if (enabled) {
|
view->setPaintCat(enabled);
|
||||||
view->setStyleSheet(QString(R"(
|
view->viewport()->repaint();
|
||||||
InstanceView
|
|
||||||
{
|
|
||||||
background-image: url(:/backgrounds/%1);
|
|
||||||
background-attachment: fixed;
|
|
||||||
background-clip: padding;
|
|
||||||
background-position: bottom right;
|
|
||||||
background-repeat: none;
|
|
||||||
background-color:palette(base);
|
|
||||||
})")
|
|
||||||
.arg(ThemeManager::getCatImage()));
|
|
||||||
} else {
|
|
||||||
view->setStyleSheet(QString());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::runModalTask(Task *task)
|
void MainWindow::runModalTask(Task *task)
|
||||||
|
@ -248,8 +248,8 @@ bool AccessibleInstanceView::selectColumn(int column)
|
|||||||
if (view()->selectionBehavior() != QAbstractItemView::SelectColumns && rowCount() > 1) {
|
if (view()->selectionBehavior() != QAbstractItemView::SelectColumns && rowCount() > 1) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
// fallthrough intentional
|
|
||||||
}
|
}
|
||||||
|
/* fallthrough */
|
||||||
case QAbstractItemView::ContiguousSelection: {
|
case QAbstractItemView::ContiguousSelection: {
|
||||||
if ((!column || !view()->selectionModel()->isColumnSelected(column - 1, view()->rootIndex())) && !view()->selectionModel()->isColumnSelected(column + 1, view()->rootIndex())) {
|
if ((!column || !view()->selectionModel()->isColumnSelected(column - 1, view()->rootIndex())) && !view()->selectionModel()->isColumnSelected(column + 1, view()->rootIndex())) {
|
||||||
view()->clearSelection();
|
view()->clearSelection();
|
||||||
|
@ -48,6 +48,7 @@
|
|||||||
#include <QAccessible>
|
#include <QAccessible>
|
||||||
|
|
||||||
#include "VisualGroup.h"
|
#include "VisualGroup.h"
|
||||||
|
#include "ui/themes/ThemeManager.h"
|
||||||
#include <QDebug>
|
#include <QDebug>
|
||||||
|
|
||||||
#include <Application.h>
|
#include <Application.h>
|
||||||
@ -73,6 +74,7 @@ InstanceView::InstanceView(QWidget *parent)
|
|||||||
setVerticalScrollBarPolicy(Qt::ScrollBarAsNeeded);
|
setVerticalScrollBarPolicy(Qt::ScrollBarAsNeeded);
|
||||||
setAcceptDrops(true);
|
setAcceptDrops(true);
|
||||||
setAutoScroll(true);
|
setAutoScroll(true);
|
||||||
|
setPaintCat(APPLICATION->settings()->get("TheCat").toBool());
|
||||||
}
|
}
|
||||||
|
|
||||||
InstanceView::~InstanceView()
|
InstanceView::~InstanceView()
|
||||||
@ -498,12 +500,34 @@ void InstanceView::mouseDoubleClickEvent(QMouseEvent *event)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void InstanceView::setPaintCat(bool visible)
|
||||||
|
{
|
||||||
|
m_catVisible = visible;
|
||||||
|
if (visible)
|
||||||
|
m_catPixmap.load(QString(":/backgrounds/%1").arg(ThemeManager::getCatImage()));
|
||||||
|
else
|
||||||
|
m_catPixmap = QPixmap();
|
||||||
|
}
|
||||||
|
|
||||||
void InstanceView::paintEvent(QPaintEvent* event)
|
void InstanceView::paintEvent(QPaintEvent* event)
|
||||||
{
|
{
|
||||||
executeDelayedItemsLayout();
|
executeDelayedItemsLayout();
|
||||||
|
|
||||||
QPainter painter(this->viewport());
|
QPainter painter(this->viewport());
|
||||||
|
|
||||||
|
if (m_catVisible) {
|
||||||
|
int widWidth = this->viewport()->width();
|
||||||
|
int widHeight = this->viewport()->height();
|
||||||
|
if (m_catPixmap.width() < widWidth)
|
||||||
|
widWidth = m_catPixmap.width();
|
||||||
|
if (m_catPixmap.height() < widHeight)
|
||||||
|
widHeight = m_catPixmap.height();
|
||||||
|
auto pixmap = m_catPixmap.scaled(widWidth, widHeight, Qt::KeepAspectRatio);
|
||||||
|
QRect rectOfPixmap = pixmap.rect();
|
||||||
|
rectOfPixmap.moveBottomRight(this->viewport()->rect().bottomRight());
|
||||||
|
painter.drawPixmap(rectOfPixmap.topLeft(), pixmap);
|
||||||
|
}
|
||||||
|
|
||||||
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
#if QT_VERSION >= QT_VERSION_CHECK(6, 0, 0)
|
||||||
QStyleOptionViewItem option;
|
QStyleOptionViewItem option;
|
||||||
initViewItemOption(&option);
|
initViewItemOption(&option);
|
||||||
|
@ -85,10 +85,8 @@ public:
|
|||||||
|
|
||||||
virtual QRegion visualRegionForSelection(const QItemSelection &selection) const override;
|
virtual QRegion visualRegionForSelection(const QItemSelection &selection) const override;
|
||||||
|
|
||||||
int spacing() const
|
int spacing() const { return m_spacing; };
|
||||||
{
|
void setPaintCat(bool visible);
|
||||||
return m_spacing;
|
|
||||||
};
|
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
virtual void updateGeometries() override;
|
virtual void updateGeometries() override;
|
||||||
@ -139,6 +137,8 @@ private:
|
|||||||
int m_currentItemsPerRow = -1;
|
int m_currentItemsPerRow = -1;
|
||||||
int m_currentCursorColumn= -1;
|
int m_currentCursorColumn= -1;
|
||||||
mutable QCache<int, QRect> geometryCache;
|
mutable QCache<int, QRect> geometryCache;
|
||||||
|
bool m_catVisible = false;
|
||||||
|
QPixmap m_catPixmap;
|
||||||
|
|
||||||
// point where the currently active mouse action started in geometry coordinates
|
// point where the currently active mouse action started in geometry coordinates
|
||||||
QPoint m_pressedPosition;
|
QPoint m_pressedPosition;
|
||||||
|
@ -81,6 +81,8 @@ APIPage::APIPage(QWidget *parent) :
|
|||||||
connect(ui->pasteTypeComboBox, currentIndexChangedSignal, this, &APIPage::updateBaseURLPlaceholder);
|
connect(ui->pasteTypeComboBox, currentIndexChangedSignal, this, &APIPage::updateBaseURLPlaceholder);
|
||||||
// This function needs to be called even when the ComboBox's index is still in its default state.
|
// This function needs to be called even when the ComboBox's index is still in its default state.
|
||||||
updateBaseURLPlaceholder(ui->pasteTypeComboBox->currentIndex());
|
updateBaseURLPlaceholder(ui->pasteTypeComboBox->currentIndex());
|
||||||
|
// NOTE: this allows http://, but we replace that with https later anyway
|
||||||
|
ui->metaURL->setValidator(new QRegularExpressionValidator(validUrlRegExp, ui->metaURL));
|
||||||
ui->baseURLEntry->setValidator(new QRegularExpressionValidator(validUrlRegExp, ui->baseURLEntry));
|
ui->baseURLEntry->setValidator(new QRegularExpressionValidator(validUrlRegExp, ui->baseURLEntry));
|
||||||
ui->msaClientID->setValidator(new QRegularExpressionValidator(validMSAClientID, ui->msaClientID));
|
ui->msaClientID->setValidator(new QRegularExpressionValidator(validMSAClientID, ui->msaClientID));
|
||||||
ui->flameKey->setValidator(new QRegularExpressionValidator(validFlameKey, ui->flameKey));
|
ui->flameKey->setValidator(new QRegularExpressionValidator(validFlameKey, ui->flameKey));
|
||||||
@ -163,7 +165,7 @@ void APIPage::applySettings()
|
|||||||
|
|
||||||
QString msaClientID = ui->msaClientID->text();
|
QString msaClientID = ui->msaClientID->text();
|
||||||
s->set("MSAClientIDOverride", msaClientID);
|
s->set("MSAClientIDOverride", msaClientID);
|
||||||
QUrl metaURL = ui->metaURL->text();
|
QUrl metaURL(ui->metaURL->text());
|
||||||
// Add required trailing slash
|
// Add required trailing slash
|
||||||
if (!metaURL.isEmpty() && !metaURL.path().endsWith('/'))
|
if (!metaURL.isEmpty() && !metaURL.path().endsWith('/'))
|
||||||
{
|
{
|
||||||
|
@ -157,6 +157,17 @@
|
|||||||
<string>Try to check or update all selected resources (all resources if none are selected)</string>
|
<string>Try to check or update all selected resources (all resources if none are selected)</string>
|
||||||
</property>
|
</property>
|
||||||
</action>
|
</action>
|
||||||
|
<action name="actionVisitItemPage">
|
||||||
|
<property name="enabled">
|
||||||
|
<bool>false</bool>
|
||||||
|
</property>
|
||||||
|
<property name="text">
|
||||||
|
<string>Visit mod's page</string>
|
||||||
|
</property>
|
||||||
|
<property name="toolTip">
|
||||||
|
<string>Go to mods home page</string>
|
||||||
|
</property>
|
||||||
|
</action>
|
||||||
</widget>
|
</widget>
|
||||||
<customwidgets>
|
<customwidgets>
|
||||||
<customwidget>
|
<customwidget>
|
||||||
|
@ -45,6 +45,7 @@
|
|||||||
#include <QMenu>
|
#include <QMenu>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
#include <QSortFilterProxyModel>
|
#include <QSortFilterProxyModel>
|
||||||
|
#include <algorithm>
|
||||||
|
|
||||||
#include "Application.h"
|
#include "Application.h"
|
||||||
|
|
||||||
@ -60,6 +61,7 @@
|
|||||||
#include "minecraft/mod/Mod.h"
|
#include "minecraft/mod/Mod.h"
|
||||||
#include "minecraft/mod/ModFolderModel.h"
|
#include "minecraft/mod/ModFolderModel.h"
|
||||||
|
|
||||||
|
#include "modplatform/ModIndex.h"
|
||||||
#include "modplatform/ResourceAPI.h"
|
#include "modplatform/ResourceAPI.h"
|
||||||
|
|
||||||
#include "Version.h"
|
#include "Version.h"
|
||||||
@ -86,10 +88,28 @@ ModFolderPage::ModFolderPage(BaseInstance* inst, std::shared_ptr<ModFolderModel>
|
|||||||
ui->actionsToolbar->insertActionAfter(ui->actionAddItem, ui->actionUpdateItem);
|
ui->actionsToolbar->insertActionAfter(ui->actionAddItem, ui->actionUpdateItem);
|
||||||
connect(ui->actionUpdateItem, &QAction::triggered, this, &ModFolderPage::updateMods);
|
connect(ui->actionUpdateItem, &QAction::triggered, this, &ModFolderPage::updateMods);
|
||||||
|
|
||||||
|
ui->actionVisitItemPage->setToolTip(tr("Go to mod's home page"));
|
||||||
|
ui->actionsToolbar->insertActionAfter(ui->actionViewFolder, ui->actionVisitItemPage);
|
||||||
|
connect(ui->actionVisitItemPage, &QAction::triggered, this, &ModFolderPage::visitModPages);
|
||||||
|
|
||||||
auto check_allow_update = [this] { return ui->treeView->selectionModel()->hasSelection() || !m_model->empty(); };
|
auto check_allow_update = [this] { return ui->treeView->selectionModel()->hasSelection() || !m_model->empty(); };
|
||||||
|
|
||||||
connect(ui->treeView->selectionModel(), &QItemSelectionModel::selectionChanged, this,
|
connect(ui->treeView->selectionModel(), &QItemSelectionModel::selectionChanged, this, [this, check_allow_update] {
|
||||||
[this, check_allow_update] { ui->actionUpdateItem->setEnabled(check_allow_update()); });
|
ui->actionUpdateItem->setEnabled(check_allow_update());
|
||||||
|
|
||||||
|
auto selection = m_filterModel->mapSelectionToSource(ui->treeView->selectionModel()->selection()).indexes();
|
||||||
|
auto mods_list = m_model->selectedMods(selection);
|
||||||
|
auto selected = std::count_if(mods_list.cbegin(), mods_list.cend(),
|
||||||
|
[](Mod* v) { return v->metadata() != nullptr || v->homeurl().size() != 0; });
|
||||||
|
if (selected <= 1) {
|
||||||
|
ui->actionVisitItemPage->setText(tr("Visit mod's page"));
|
||||||
|
ui->actionVisitItemPage->setToolTip(tr("Go to mod's home page"));
|
||||||
|
} else {
|
||||||
|
ui->actionVisitItemPage->setText(tr("Visit mods' pages"));
|
||||||
|
ui->actionVisitItemPage->setToolTip(tr("Go to the pages of the selected mods"));
|
||||||
|
}
|
||||||
|
ui->actionVisitItemPage->setEnabled(selected != 0);
|
||||||
|
});
|
||||||
|
|
||||||
connect(mods.get(), &ModFolderModel::rowsInserted, this,
|
connect(mods.get(), &ModFolderModel::rowsInserted, this,
|
||||||
[this, check_allow_update] { ui->actionUpdateItem->setEnabled(check_allow_update()); });
|
[this, check_allow_update] { ui->actionUpdateItem->setEnabled(check_allow_update()); });
|
||||||
@ -267,3 +287,13 @@ bool NilModFolderPage::shouldDisplay() const
|
|||||||
{
|
{
|
||||||
return m_model->dir().exists();
|
return m_model->dir().exists();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ModFolderPage::visitModPages()
|
||||||
|
{
|
||||||
|
auto selection = m_filterModel->mapSelectionToSource(ui->treeView->selectionModel()->selection()).indexes();
|
||||||
|
for (auto mod : m_model->selectedMods(selection)) {
|
||||||
|
auto url = mod->metaurl();
|
||||||
|
if (!url.isEmpty())
|
||||||
|
DesktopServices::openUrl(url);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
* Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org>
|
* Copyright (c) 2022 Jamie Mansfield <jmansfield@cadixdev.org>
|
||||||
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
* Copyright (C) 2022 Sefa Eyeoglu <contact@scrumplex.net>
|
||||||
* Copyright (C) 2022 TheKodeToad <TheKodeToad@proton.me>
|
* Copyright (C) 2022 TheKodeToad <TheKodeToad@proton.me>
|
||||||
|
* Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -63,6 +64,7 @@ class ModFolderPage : public ExternalResourcesPage {
|
|||||||
|
|
||||||
void installMods();
|
void installMods();
|
||||||
void updateMods();
|
void updateMods();
|
||||||
|
void visitModPages();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
std::shared_ptr<ModFolderModel> m_model;
|
std::shared_ptr<ModFolderModel> m_model;
|
||||||
|
@ -97,17 +97,13 @@ public:
|
|||||||
return;
|
return;
|
||||||
if ((info.suffix().compare("png", Qt::CaseInsensitive) != 0))
|
if ((info.suffix().compare("png", Qt::CaseInsensitive) != 0))
|
||||||
return;
|
return;
|
||||||
int tries = 5;
|
|
||||||
while (tries)
|
|
||||||
{
|
|
||||||
if (!m_cache->stale(m_path))
|
if (!m_cache->stale(m_path))
|
||||||
return;
|
return;
|
||||||
QImage image(m_path);
|
QImage image(m_path);
|
||||||
if (image.isNull())
|
if (image.isNull()) {
|
||||||
{
|
m_resultEmitter.emitResultsFailed(m_path);
|
||||||
QThread::msleep(500);
|
qDebug() << "Error loading screenshot: " + m_path + ". Perhaps too large?";
|
||||||
tries--;
|
return;
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
QImage small;
|
QImage small;
|
||||||
if (image.width() > image.height())
|
if (image.width() > image.height())
|
||||||
@ -125,9 +121,6 @@ public:
|
|||||||
QIcon icon(QPixmap::fromImage(square));
|
QIcon icon(QPixmap::fromImage(square));
|
||||||
m_cache->add(m_path, icon);
|
m_cache->add(m_path, icon);
|
||||||
m_resultEmitter.emitResultsReady(m_path);
|
m_resultEmitter.emitResultsReady(m_path);
|
||||||
return;
|
|
||||||
}
|
|
||||||
m_resultEmitter.emitResultsFailed(m_path);
|
|
||||||
}
|
}
|
||||||
QString m_path;
|
QString m_path;
|
||||||
SharedIconCachePtr m_cache;
|
SharedIconCachePtr m_cache;
|
||||||
@ -146,9 +139,12 @@ public:
|
|||||||
m_thumbnailCache = std::make_shared<SharedIconCache>();
|
m_thumbnailCache = std::make_shared<SharedIconCache>();
|
||||||
m_thumbnailCache->add("placeholder", APPLICATION->getThemedIcon("screenshot-placeholder"));
|
m_thumbnailCache->add("placeholder", APPLICATION->getThemedIcon("screenshot-placeholder"));
|
||||||
connect(&watcher, SIGNAL(fileChanged(QString)), SLOT(fileChanged(QString)));
|
connect(&watcher, SIGNAL(fileChanged(QString)), SLOT(fileChanged(QString)));
|
||||||
// FIXME: the watched file set is not updated when files are removed
|
|
||||||
}
|
}
|
||||||
virtual ~FilterModel() { m_thumbnailingPool.waitForDone(500); }
|
virtual ~FilterModel() {
|
||||||
|
m_thumbnailingPool.clear();
|
||||||
|
if (!m_thumbnailingPool.waitForDone(500))
|
||||||
|
qDebug() << "Thumbnail pool took longer than 500ms to finish";
|
||||||
|
}
|
||||||
virtual QVariant data(const QModelIndex &proxyIndex, int role = Qt::DisplayRole) const
|
virtual QVariant data(const QModelIndex &proxyIndex, int role = Qt::DisplayRole) const
|
||||||
{
|
{
|
||||||
auto model = sourceModel();
|
auto model = sourceModel();
|
||||||
@ -215,10 +211,12 @@ private slots:
|
|||||||
void fileChanged(QString filepath)
|
void fileChanged(QString filepath)
|
||||||
{
|
{
|
||||||
m_thumbnailCache->setStale(filepath);
|
m_thumbnailCache->setStale(filepath);
|
||||||
thumbnailImage(filepath);
|
|
||||||
// reinsert the path...
|
// reinsert the path...
|
||||||
watcher.removePath(filepath);
|
watcher.removePath(filepath);
|
||||||
|
if (QFile::exists(filepath)) {
|
||||||
watcher.addPath(filepath);
|
watcher.addPath(filepath);
|
||||||
|
thumbnailImage(filepath);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -339,6 +339,7 @@ void WorldListPage::mceditState(LoggedProcess::State state)
|
|||||||
{
|
{
|
||||||
failed = true;
|
failed = true;
|
||||||
}
|
}
|
||||||
|
/* fallthrough */
|
||||||
case LoggedProcess::Running:
|
case LoggedProcess::Running:
|
||||||
case LoggedProcess::Finished:
|
case LoggedProcess::Finished:
|
||||||
{
|
{
|
||||||
|
@ -69,6 +69,7 @@ bool JavaWizardPage::validatePage()
|
|||||||
case JavaSettingsWidget::ValidationStatus::AllOK:
|
case JavaSettingsWidget::ValidationStatus::AllOK:
|
||||||
{
|
{
|
||||||
settings->set("JavaPath", m_java_widget->javaPath());
|
settings->set("JavaPath", m_java_widget->javaPath());
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
case JavaSettingsWidget::ValidationStatus::JavaBad:
|
case JavaSettingsWidget::ValidationStatus::JavaBad:
|
||||||
{
|
{
|
||||||
|
@ -1,7 +1,8 @@
|
|||||||
// SPDX-License-Identifier: GPL-3.0-only
|
// SPDX-License-Identifier: GPL-3.0-only
|
||||||
/*
|
/*
|
||||||
* PolyMC - Minecraft Launcher
|
* Prism Launcher - Minecraft Launcher
|
||||||
* Copyright (c) 2022 flowln <flowlnlnln@gmail.com>
|
* Copyright (c) 2022 flowln <flowlnlnln@gmail.com>
|
||||||
|
* Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com>
|
||||||
*
|
*
|
||||||
* This program is free software: you can redistribute it and/or modify
|
* 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
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -33,22 +34,37 @@
|
|||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#include <QLabel>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
|
#include <QToolTip>
|
||||||
|
|
||||||
#include "InfoFrame.h"
|
#include "InfoFrame.h"
|
||||||
#include "ui_InfoFrame.h"
|
#include "ui_InfoFrame.h"
|
||||||
|
|
||||||
#include "ui/dialogs/CustomMessageBox.h"
|
#include "ui/dialogs/CustomMessageBox.h"
|
||||||
|
|
||||||
InfoFrame::InfoFrame(QWidget *parent) :
|
void setupLinkToolTip(QLabel* label)
|
||||||
QFrame(parent),
|
{
|
||||||
ui(new Ui::InfoFrame)
|
QObject::connect(label, &QLabel::linkHovered, [label](const QString& link) {
|
||||||
|
if (auto url = QUrl(link); !url.isValid() || (url.scheme() != "http" && url.scheme() != "https"))
|
||||||
|
return;
|
||||||
|
label->setToolTip(link);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
InfoFrame::InfoFrame(QWidget* parent) : QFrame(parent), ui(new Ui::InfoFrame)
|
||||||
{
|
{
|
||||||
ui->setupUi(this);
|
ui->setupUi(this);
|
||||||
ui->descriptionLabel->setHidden(true);
|
ui->descriptionLabel->setHidden(true);
|
||||||
ui->nameLabel->setHidden(true);
|
ui->nameLabel->setHidden(true);
|
||||||
ui->licenseLabel->setHidden(true);
|
ui->licenseLabel->setHidden(true);
|
||||||
ui->issueTrackerLabel->setHidden(true);
|
ui->issueTrackerLabel->setHidden(true);
|
||||||
|
|
||||||
|
setupLinkToolTip(ui->iconLabel);
|
||||||
|
setupLinkToolTip(ui->descriptionLabel);
|
||||||
|
setupLinkToolTip(ui->nameLabel);
|
||||||
|
setupLinkToolTip(ui->licenseLabel);
|
||||||
|
setupLinkToolTip(ui->issueTrackerLabel);
|
||||||
updateHiddenState();
|
updateHiddenState();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -59,34 +75,32 @@ InfoFrame::~InfoFrame()
|
|||||||
|
|
||||||
void InfoFrame::updateWithMod(Mod const& m)
|
void InfoFrame::updateWithMod(Mod const& m)
|
||||||
{
|
{
|
||||||
if (m.type() == ResourceType::FOLDER)
|
if (m.type() == ResourceType::FOLDER) {
|
||||||
{
|
|
||||||
clear();
|
clear();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
QString text = "";
|
QString text = "";
|
||||||
QString name = "";
|
QString name = "";
|
||||||
|
QString link = m.metaurl();
|
||||||
if (m.name().isEmpty())
|
if (m.name().isEmpty())
|
||||||
name = m.internal_id();
|
name = m.internal_id();
|
||||||
else
|
else
|
||||||
name = m.name();
|
name = m.name();
|
||||||
|
|
||||||
if (m.homeurl().isEmpty())
|
if (link.isEmpty())
|
||||||
text = name;
|
text = name;
|
||||||
else
|
else {
|
||||||
text = "<a href=\"" + m.homeurl() + "\">" + name + "</a>";
|
text = "<a href=\"" + link + "\">" + name + "</a>";
|
||||||
|
}
|
||||||
if (!m.authors().isEmpty())
|
if (!m.authors().isEmpty())
|
||||||
text += " by " + m.authors().join(", ");
|
text += " by " + m.authors().join(", ");
|
||||||
|
|
||||||
setName(text);
|
setName(text);
|
||||||
|
|
||||||
if (m.description().isEmpty())
|
if (m.description().isEmpty()) {
|
||||||
{
|
|
||||||
setDescription(QString());
|
setDescription(QString());
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
setDescription(m.description());
|
setDescription(m.description());
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -133,7 +147,8 @@ void InfoFrame::updateWithResource(const Resource& resource)
|
|||||||
setImage();
|
setImage();
|
||||||
}
|
}
|
||||||
|
|
||||||
QString InfoFrame::renderColorCodes(QString input) {
|
QString InfoFrame::renderColorCodes(QString input)
|
||||||
|
{
|
||||||
// We have to manually set the colors for use.
|
// We have to manually set the colors for use.
|
||||||
//
|
//
|
||||||
// A color is set using §x, with x = a hex number from 0 to f.
|
// A color is set using §x, with x = a hex number from 0 to f.
|
||||||
@ -144,16 +159,12 @@ QString InfoFrame::renderColorCodes(QString input) {
|
|||||||
// TODO: Wrap links inside <a> tags
|
// TODO: Wrap links inside <a> tags
|
||||||
|
|
||||||
// https://minecraft.fandom.com/wiki/Formatting_codes#Color_codes
|
// https://minecraft.fandom.com/wiki/Formatting_codes#Color_codes
|
||||||
const QMap<QChar, QString> color_codes_map = {
|
const QMap<QChar, QString> color_codes_map = { { '0', "#000000" }, { '1', "#0000AA" }, { '2', "#00AA00" }, { '3', "#00AAAA" },
|
||||||
{'0', "#000000"}, {'1', "#0000AA"}, {'2', "#00AA00"}, {'3', "#00AAAA"}, {'4', "#AA0000"},
|
{ '4', "#AA0000" }, { '5', "#AA00AA" }, { '6', "#FFAA00" }, { '7', "#AAAAAA" },
|
||||||
{'5', "#AA00AA"}, {'6', "#FFAA00"}, {'7', "#AAAAAA"}, {'8', "#555555"}, {'9', "#5555FF"},
|
{ '8', "#555555" }, { '9', "#5555FF" }, { 'a', "#55FF55" }, { 'b', "#55FFFF" },
|
||||||
{'a', "#55FF55"}, {'b', "#55FFFF"}, {'c', "#FF5555"}, {'d', "#FF55FF"}, {'e', "#FFFF55"},
|
{ 'c', "#FF5555" }, { 'd', "#FF55FF" }, { 'e', "#FFFF55" }, { 'f', "#FFFFFF" } };
|
||||||
{'f', "#FFFFFF"}
|
|
||||||
};
|
|
||||||
// https://minecraft.fandom.com/wiki/Formatting_codes#Formatting_codes
|
// https://minecraft.fandom.com/wiki/Formatting_codes#Formatting_codes
|
||||||
const QMap<QChar, QString> formatting_codes_map = {
|
const QMap<QChar, QString> formatting_codes_map = { { 'l', "b" }, { 'm', "s" }, { 'n', "u" }, { 'o', "i" } };
|
||||||
{'l', "b"}, {'m', "s"}, {'n', "u"}, {'o', "i"}
|
|
||||||
};
|
|
||||||
|
|
||||||
QString html("<html>");
|
QString html("<html>");
|
||||||
QList<QString> tags{};
|
QList<QString> tags{};
|
||||||
@ -229,12 +240,9 @@ void InfoFrame::updateHiddenState()
|
|||||||
|
|
||||||
void InfoFrame::setName(QString text)
|
void InfoFrame::setName(QString text)
|
||||||
{
|
{
|
||||||
if(text.isEmpty())
|
if (text.isEmpty()) {
|
||||||
{
|
|
||||||
ui->nameLabel->setHidden(true);
|
ui->nameLabel->setHidden(true);
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
ui->nameLabel->setText(text);
|
ui->nameLabel->setText(text);
|
||||||
ui->nameLabel->setHidden(false);
|
ui->nameLabel->setHidden(false);
|
||||||
}
|
}
|
||||||
@ -243,14 +251,11 @@ void InfoFrame::setName(QString text)
|
|||||||
|
|
||||||
void InfoFrame::setDescription(QString text)
|
void InfoFrame::setDescription(QString text)
|
||||||
{
|
{
|
||||||
if(text.isEmpty())
|
if (text.isEmpty()) {
|
||||||
{
|
|
||||||
ui->descriptionLabel->setHidden(true);
|
ui->descriptionLabel->setHidden(true);
|
||||||
updateHiddenState();
|
updateHiddenState();
|
||||||
return;
|
return;
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
ui->descriptionLabel->setHidden(false);
|
ui->descriptionLabel->setHidden(false);
|
||||||
updateHiddenState();
|
updateHiddenState();
|
||||||
}
|
}
|
||||||
@ -260,8 +265,7 @@ void InfoFrame::setDescription(QString text)
|
|||||||
QChar rem('\n');
|
QChar rem('\n');
|
||||||
QString finaltext;
|
QString finaltext;
|
||||||
finaltext.reserve(intermediatetext.size());
|
finaltext.reserve(intermediatetext.size());
|
||||||
foreach(const QChar& c, intermediatetext)
|
foreach (const QChar& c, intermediatetext) {
|
||||||
{
|
|
||||||
if (c == rem && prev) {
|
if (c == rem && prev) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -270,17 +274,14 @@ void InfoFrame::setDescription(QString text)
|
|||||||
}
|
}
|
||||||
QString labeltext;
|
QString labeltext;
|
||||||
labeltext.reserve(300);
|
labeltext.reserve(300);
|
||||||
if(finaltext.length() > 290)
|
if (finaltext.length() > 290) {
|
||||||
{
|
|
||||||
ui->descriptionLabel->setOpenExternalLinks(false);
|
ui->descriptionLabel->setOpenExternalLinks(false);
|
||||||
ui->descriptionLabel->setTextFormat(Qt::TextFormat::RichText);
|
ui->descriptionLabel->setTextFormat(Qt::TextFormat::RichText);
|
||||||
m_description = text;
|
m_description = text;
|
||||||
// This allows injecting HTML here.
|
// This allows injecting HTML here.
|
||||||
labeltext.append("<html><body>" + finaltext.left(287) + "<a href=\"#mod_desc\">...</a></body></html>");
|
labeltext.append("<html><body>" + finaltext.left(287) + "<a href=\"#mod_desc\">...</a></body></html>");
|
||||||
QObject::connect(ui->descriptionLabel, &QLabel::linkActivated, this, &InfoFrame::descriptionEllipsisHandler);
|
QObject::connect(ui->descriptionLabel, &QLabel::linkActivated, this, &InfoFrame::descriptionEllipsisHandler);
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
ui->descriptionLabel->setTextFormat(Qt::TextFormat::AutoText);
|
ui->descriptionLabel->setTextFormat(Qt::TextFormat::AutoText);
|
||||||
labeltext.append(finaltext);
|
labeltext.append(finaltext);
|
||||||
}
|
}
|
||||||
@ -289,14 +290,11 @@ void InfoFrame::setDescription(QString text)
|
|||||||
|
|
||||||
void InfoFrame::setLicense(QString text)
|
void InfoFrame::setLicense(QString text)
|
||||||
{
|
{
|
||||||
if(text.isEmpty())
|
if (text.isEmpty()) {
|
||||||
{
|
|
||||||
ui->licenseLabel->setHidden(true);
|
ui->licenseLabel->setHidden(true);
|
||||||
updateHiddenState();
|
updateHiddenState();
|
||||||
return;
|
return;
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
ui->licenseLabel->setHidden(false);
|
ui->licenseLabel->setHidden(false);
|
||||||
updateHiddenState();
|
updateHiddenState();
|
||||||
}
|
}
|
||||||
@ -306,8 +304,7 @@ void InfoFrame::setLicense(QString text)
|
|||||||
QChar rem('\n');
|
QChar rem('\n');
|
||||||
QString finaltext;
|
QString finaltext;
|
||||||
finaltext.reserve(intermediatetext.size());
|
finaltext.reserve(intermediatetext.size());
|
||||||
foreach(const QChar& c, intermediatetext)
|
foreach (const QChar& c, intermediatetext) {
|
||||||
{
|
|
||||||
if (c == rem && prev) {
|
if (c == rem && prev) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@ -316,17 +313,14 @@ void InfoFrame::setLicense(QString text)
|
|||||||
}
|
}
|
||||||
QString labeltext;
|
QString labeltext;
|
||||||
labeltext.reserve(300);
|
labeltext.reserve(300);
|
||||||
if(finaltext.length() > 290)
|
if (finaltext.length() > 290) {
|
||||||
{
|
|
||||||
ui->licenseLabel->setOpenExternalLinks(false);
|
ui->licenseLabel->setOpenExternalLinks(false);
|
||||||
ui->licenseLabel->setTextFormat(Qt::TextFormat::RichText);
|
ui->licenseLabel->setTextFormat(Qt::TextFormat::RichText);
|
||||||
m_description = text;
|
m_description = text;
|
||||||
// This allows injecting HTML here.
|
// This allows injecting HTML here.
|
||||||
labeltext.append("<html><body>" + finaltext.left(287) + "<a href=\"#mod_desc\">...</a></body></html>");
|
labeltext.append("<html><body>" + finaltext.left(287) + "<a href=\"#mod_desc\">...</a></body></html>");
|
||||||
QObject::connect(ui->licenseLabel, &QLabel::linkActivated, this, &InfoFrame::licenseEllipsisHandler);
|
QObject::connect(ui->licenseLabel, &QLabel::linkActivated, this, &InfoFrame::licenseEllipsisHandler);
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
ui->licenseLabel->setTextFormat(Qt::TextFormat::AutoText);
|
ui->licenseLabel->setTextFormat(Qt::TextFormat::AutoText);
|
||||||
labeltext.append(finaltext);
|
labeltext.append(finaltext);
|
||||||
}
|
}
|
||||||
@ -335,12 +329,9 @@ void InfoFrame::setLicense(QString text)
|
|||||||
|
|
||||||
void InfoFrame::setIssueTracker(QString text)
|
void InfoFrame::setIssueTracker(QString text)
|
||||||
{
|
{
|
||||||
if(text.isEmpty())
|
if (text.isEmpty()) {
|
||||||
{
|
|
||||||
ui->issueTrackerLabel->setHidden(true);
|
ui->issueTrackerLabel->setHidden(true);
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
ui->issueTrackerLabel->setText(text);
|
ui->issueTrackerLabel->setText(text);
|
||||||
ui->issueTrackerLabel->setHidden(false);
|
ui->issueTrackerLabel->setHidden(false);
|
||||||
}
|
}
|
||||||
@ -359,28 +350,22 @@ void InfoFrame::setImage(QPixmap img)
|
|||||||
|
|
||||||
void InfoFrame::descriptionEllipsisHandler(QString link)
|
void InfoFrame::descriptionEllipsisHandler(QString link)
|
||||||
{
|
{
|
||||||
if(!m_current_box)
|
if (!m_current_box) {
|
||||||
{
|
|
||||||
m_current_box = CustomMessageBox::selectable(this, "", m_description);
|
m_current_box = CustomMessageBox::selectable(this, "", m_description);
|
||||||
connect(m_current_box, &QMessageBox::finished, this, &InfoFrame::boxClosed);
|
connect(m_current_box, &QMessageBox::finished, this, &InfoFrame::boxClosed);
|
||||||
m_current_box->show();
|
m_current_box->show();
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
m_current_box->setText(m_description);
|
m_current_box->setText(m_description);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void InfoFrame::licenseEllipsisHandler(QString link)
|
void InfoFrame::licenseEllipsisHandler(QString link)
|
||||||
{
|
{
|
||||||
if(!m_current_box)
|
if (!m_current_box) {
|
||||||
{
|
|
||||||
m_current_box = CustomMessageBox::selectable(this, "", m_license);
|
m_current_box = CustomMessageBox::selectable(this, "", m_license);
|
||||||
connect(m_current_box, &QMessageBox::finished, this, &InfoFrame::boxClosed);
|
connect(m_current_box, &QMessageBox::finished, this, &InfoFrame::boxClosed);
|
||||||
m_current_box->show();
|
m_current_box->show();
|
||||||
}
|
} else {
|
||||||
else
|
|
||||||
{
|
|
||||||
m_current_box->setText(m_license);
|
m_current_box->setText(m_license);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,4 +1,24 @@
|
|||||||
/* Copyright 2013-2021 MultiMC Contributors
|
// 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");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
@ -21,8 +41,7 @@
|
|||||||
#include "minecraft/mod/ResourcePack.h"
|
#include "minecraft/mod/ResourcePack.h"
|
||||||
#include "minecraft/mod/TexturePack.h"
|
#include "minecraft/mod/TexturePack.h"
|
||||||
|
|
||||||
namespace Ui
|
namespace Ui {
|
||||||
{
|
|
||||||
class InfoFrame;
|
class InfoFrame;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -89,8 +89,10 @@ void FourBytes_MurmurHash2(const unsigned char* data, IncrementalHashInfo& prev)
|
|||||||
switch (prev.len) {
|
switch (prev.len) {
|
||||||
case 3:
|
case 3:
|
||||||
prev.h ^= data[2] << 16;
|
prev.h ^= data[2] << 16;
|
||||||
|
/* fall through */
|
||||||
case 2:
|
case 2:
|
||||||
prev.h ^= data[1] << 8;
|
prev.h ^= data[1] << 8;
|
||||||
|
/* fall through */
|
||||||
case 1:
|
case 1:
|
||||||
prev.h ^= data[0];
|
prev.h ^= data[0];
|
||||||
prev.h *= m;
|
prev.h *= m;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user