2022-10-25 13:49:19 +05:30
|
|
|
#include "Application.h"
|
2022-07-17 04:44:54 +05:30
|
|
|
#include "BlockedModsDialog.h"
|
|
|
|
#include "ui_BlockedModsDialog.h"
|
2022-07-18 23:33:06 +05:30
|
|
|
#include <QPushButton>
|
2022-07-17 04:44:54 +05:30
|
|
|
#include <QDialogButtonBox>
|
|
|
|
#include <QDesktopServices>
|
|
|
|
|
2022-10-24 16:38:38 +05:30
|
|
|
#include <QDebug>
|
2022-10-25 13:49:19 +05:30
|
|
|
#include <QStandardPaths>
|
2022-07-17 04:44:54 +05:30
|
|
|
|
2022-10-24 16:38:38 +05:30
|
|
|
|
2022-10-25 13:49:19 +05:30
|
|
|
|
|
|
|
|
|
|
|
BlockedModsDialog::BlockedModsDialog(QWidget *parent, const QString &title, const QString &text, QList<BlockedMod> &mods) :
|
2022-10-24 16:38:38 +05:30
|
|
|
QDialog(parent), ui(new Ui::BlockedModsDialog), mods(mods) {
|
2022-07-17 04:44:54 +05:30
|
|
|
ui->setupUi(this);
|
|
|
|
|
|
|
|
auto openAllButton = ui->buttonBox->addButton(tr("Open All"), QDialogButtonBox::ActionRole);
|
|
|
|
connect(openAllButton, &QPushButton::clicked, this, &BlockedModsDialog::openAll);
|
|
|
|
|
2022-10-25 13:49:19 +05:30
|
|
|
connect(&watcher, &QFileSystemWatcher::directoryChanged, this, &BlockedModsDialog::directoryChanged);
|
|
|
|
|
|
|
|
hashing_task = shared_qobject_ptr<ConcurrentTask>(new ConcurrentTask(this, "MakeHashesTask", 10));
|
|
|
|
|
2022-10-24 16:38:38 +05:30
|
|
|
qDebug() << "Mods List: " << mods;
|
|
|
|
|
2022-10-25 13:49:19 +05:30
|
|
|
setupWatch();
|
2022-10-26 08:48:14 +05:30
|
|
|
scanPaths();
|
2022-10-25 13:49:19 +05:30
|
|
|
|
2022-07-17 04:44:54 +05:30
|
|
|
this->setWindowTitle(title);
|
|
|
|
ui->label->setText(text);
|
2022-10-25 23:29:37 +05:30
|
|
|
ui->labelModsFound->setText("Please download the missing mods.");
|
2022-10-24 16:38:38 +05:30
|
|
|
update();
|
2022-07-17 04:44:54 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
BlockedModsDialog::~BlockedModsDialog() {
|
|
|
|
delete ui;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BlockedModsDialog::openAll() {
|
2022-10-24 16:38:38 +05:30
|
|
|
for(auto &mod : mods) {
|
|
|
|
QDesktopServices::openUrl(mod.websiteUrl);
|
2022-07-17 04:44:54 +05:30
|
|
|
}
|
|
|
|
}
|
2022-10-24 16:38:38 +05:30
|
|
|
|
2022-10-26 08:48:14 +05:30
|
|
|
/// @brief update UI with current status of the blocked mod detection
|
2022-10-24 16:38:38 +05:30
|
|
|
void BlockedModsDialog::update() {
|
|
|
|
QString text;
|
|
|
|
QString span;
|
|
|
|
|
|
|
|
for (auto &mod : mods) {
|
|
|
|
if (mod.matched) {
|
|
|
|
// ✔ -> html for HEAVY CHECK MARK : ✔
|
|
|
|
span = QString("<span style=\"color:green\"> ✔ Found at %1 </span>").arg(mod.localPath);
|
|
|
|
} else {
|
|
|
|
// ✘ -> html for HEAVY BALLOT X : ✘
|
|
|
|
span = QString("<span style=\"color:red\"> ✘ Not Found </span>");
|
|
|
|
}
|
|
|
|
text += QString("%1: <a href='%2'>%2</a> <p>Hash: %3 %4</p> <br/>").arg(mod.name, mod.websiteUrl, mod.hash, span);
|
|
|
|
}
|
|
|
|
|
|
|
|
ui->textBrowser->setText(text);
|
2022-10-25 23:29:37 +05:30
|
|
|
|
|
|
|
if (allModsMatched()) {
|
|
|
|
ui->labelModsFound->setText("All mods found ✔");
|
|
|
|
} else {
|
|
|
|
ui->labelModsFound->setText("Please download the missing mods.");
|
|
|
|
}
|
2022-10-24 16:38:38 +05:30
|
|
|
}
|
|
|
|
|
2022-10-26 08:48:14 +05:30
|
|
|
/// @brief Signal fired when a watched direcotry has changed
|
|
|
|
/// @param path the path to the changed directory
|
2022-10-25 13:49:19 +05:30
|
|
|
void BlockedModsDialog::directoryChanged(QString path) {
|
|
|
|
qDebug() << "Directory changed: " << path;
|
2022-10-26 08:48:14 +05:30
|
|
|
scanPath(path);
|
2022-10-25 13:49:19 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-10-26 08:48:14 +05:30
|
|
|
/// @brief add the user downloads folder and the global mods folder to the filesystem watcher
|
2022-10-25 13:49:19 +05:30
|
|
|
void BlockedModsDialog::setupWatch() {
|
|
|
|
const QString downloadsFolder = QStandardPaths::writableLocation(QStandardPaths::DownloadLocation);
|
|
|
|
const QString modsFolder = APPLICATION->settings()->get("CentralModsDir").toString();
|
|
|
|
watcher.addPath(downloadsFolder);
|
|
|
|
watcher.addPath(modsFolder);
|
|
|
|
}
|
|
|
|
|
2022-10-26 08:48:14 +05:30
|
|
|
|
|
|
|
/// @brief scan all watched folder
|
|
|
|
void BlockedModsDialog::scanPaths() {
|
2022-10-25 13:49:19 +05:30
|
|
|
for (auto &dir : watcher.directories()) {
|
2022-10-26 08:48:14 +05:30
|
|
|
scanPath(dir);
|
2022-10-25 13:49:19 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-26 08:48:14 +05:30
|
|
|
/// @brief Scan the directory at path, skip paths that do not contain a file name
|
|
|
|
/// of a blocked mod we are looking for
|
|
|
|
/// @param path the directory to scan
|
|
|
|
void BlockedModsDialog::scanPath(QString path) {
|
2022-10-25 13:49:19 +05:30
|
|
|
|
|
|
|
QDir scan_dir(path);
|
|
|
|
QDirIterator scan_it(path, QDir::Filter::Files | QDir::Filter::Hidden, QDirIterator::NoIteratorFlags);
|
|
|
|
while (scan_it.hasNext()) {
|
|
|
|
QString file = scan_it.next();
|
|
|
|
|
|
|
|
if (!checkValidPath(file)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto hash_task = Hashing::createBlockedModHasher(file, ModPlatform::Provider::FLAME, "sha1");
|
|
|
|
|
|
|
|
qDebug() << "Creating Hash task for path: " << file;
|
|
|
|
|
|
|
|
connect(hash_task.get(), &Task::succeeded, [this, hash_task, file] {
|
|
|
|
checkMatchHash(hash_task->getResult(), file);
|
|
|
|
});
|
|
|
|
connect(hash_task.get(), &Task::failed, [this, hash_task, file] {
|
|
|
|
qDebug() << "Failed to hash path: " << file;
|
|
|
|
});
|
|
|
|
|
|
|
|
hashing_task->addTask(hash_task);
|
|
|
|
}
|
|
|
|
|
|
|
|
hashing_task->start();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2022-10-26 08:48:14 +05:30
|
|
|
/// @brief check if the conputed hash for the provided path matches a blocked
|
|
|
|
/// mod we are looking for
|
|
|
|
/// @param hash the computed hash for the provided path
|
|
|
|
/// @param path the path to the local file being compared
|
2022-10-25 13:49:19 +05:30
|
|
|
void BlockedModsDialog::checkMatchHash(QString hash, QString path) {
|
|
|
|
bool match = false;
|
|
|
|
|
|
|
|
qDebug() << "Checking for match on hash: " << hash << " | From path:" << path;
|
|
|
|
|
|
|
|
for (auto &mod : mods) {
|
|
|
|
if (mod.matched) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (mod.hash.compare(hash, Qt::CaseInsensitive) == 0) {
|
|
|
|
mod.matched = true;
|
|
|
|
mod.localPath = path;
|
|
|
|
match = true;
|
|
|
|
|
|
|
|
qDebug() << "Hash match found: " << mod.name << " " << hash << " | From path:" << path;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (match) {
|
|
|
|
update();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-26 08:48:14 +05:30
|
|
|
/// @brief Check if the name of the file at path matches the naem of a blocked mod we are searching for
|
|
|
|
/// @param path the path to check
|
|
|
|
/// @return boolean: did the path match the name of a blocked mod?
|
2022-10-25 13:49:19 +05:30
|
|
|
bool BlockedModsDialog::checkValidPath(QString path) {
|
|
|
|
|
|
|
|
QFileInfo file = QFileInfo(path);
|
|
|
|
QString filename = file.fileName();
|
|
|
|
|
|
|
|
for (auto &mod : mods) {
|
|
|
|
if (mod.name.compare(filename, Qt::CaseInsensitive) == 0) {
|
|
|
|
qDebug() << "Name match found: " << mod.name << " | From path:" << path;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-10-26 08:48:14 +05:30
|
|
|
/// @brief have we found all the mods we're lookign for?
|
|
|
|
/// @return boolean
|
2022-10-25 13:49:19 +05:30
|
|
|
bool BlockedModsDialog::allModsMatched() {
|
|
|
|
for (auto &mod : mods) {
|
|
|
|
if (!mod.matched)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2022-10-26 08:48:14 +05:30
|
|
|
/// qDebug print support for the BlockedMod struct
|
2022-10-24 16:38:38 +05:30
|
|
|
QDebug operator<<(QDebug debug, const BlockedMod &m) {
|
|
|
|
QDebugStateSaver saver(debug);
|
|
|
|
|
|
|
|
debug.nospace() << "{ name: " << m.name << ", websiteUrl: " << m.websiteUrl
|
|
|
|
<< ", hash: " << m.hash << ", matched: " << m.matched
|
|
|
|
<< ", localPath: " << m.localPath <<"}";
|
|
|
|
|
|
|
|
return debug;
|
2022-10-26 08:48:14 +05:30
|
|
|
}
|