mirror of
https://github.com/PrismLauncher/PrismLauncher.git
synced 2025-05-11 12:04:41 +02:00
implement packignore for curseforge/modrinth export (#3295)
This commit is contained in:
commit
ecf770d4ae
@ -40,12 +40,11 @@
|
|||||||
#include <QFileSystemModel>
|
#include <QFileSystemModel>
|
||||||
#include <QSortFilterProxyModel>
|
#include <QSortFilterProxyModel>
|
||||||
#include <QStack>
|
#include <QStack>
|
||||||
#include <algorithm>
|
|
||||||
#include "FileSystem.h"
|
#include "FileSystem.h"
|
||||||
#include "SeparatorPrefixTree.h"
|
#include "SeparatorPrefixTree.h"
|
||||||
#include "StringUtils.h"
|
#include "StringUtils.h"
|
||||||
|
|
||||||
FileIgnoreProxy::FileIgnoreProxy(QString root, QObject* parent) : QSortFilterProxyModel(parent), root(root) {}
|
FileIgnoreProxy::FileIgnoreProxy(QString root, QObject* parent) : QSortFilterProxyModel(parent), m_root(root) {}
|
||||||
// NOTE: Sadly, we have to do sorting ourselves.
|
// NOTE: Sadly, we have to do sorting ourselves.
|
||||||
bool FileIgnoreProxy::lessThan(const QModelIndex& left, const QModelIndex& right) const
|
bool FileIgnoreProxy::lessThan(const QModelIndex& left, const QModelIndex& right) const
|
||||||
{
|
{
|
||||||
@ -104,10 +103,10 @@ QVariant FileIgnoreProxy::data(const QModelIndex& index, int role) const
|
|||||||
if (index.column() == 0 && role == Qt::CheckStateRole) {
|
if (index.column() == 0 && role == Qt::CheckStateRole) {
|
||||||
QFileSystemModel* fsm = qobject_cast<QFileSystemModel*>(sourceModel());
|
QFileSystemModel* fsm = qobject_cast<QFileSystemModel*>(sourceModel());
|
||||||
auto blockedPath = relPath(fsm->filePath(sourceIndex));
|
auto blockedPath = relPath(fsm->filePath(sourceIndex));
|
||||||
auto cover = blocked.cover(blockedPath);
|
auto cover = m_blocked.cover(blockedPath);
|
||||||
if (!cover.isNull()) {
|
if (!cover.isNull()) {
|
||||||
return QVariant(Qt::Unchecked);
|
return QVariant(Qt::Unchecked);
|
||||||
} else if (blocked.exists(blockedPath)) {
|
} else if (m_blocked.exists(blockedPath)) {
|
||||||
return QVariant(Qt::PartiallyChecked);
|
return QVariant(Qt::PartiallyChecked);
|
||||||
} else {
|
} else {
|
||||||
return QVariant(Qt::Checked);
|
return QVariant(Qt::Checked);
|
||||||
@ -130,7 +129,7 @@ bool FileIgnoreProxy::setData(const QModelIndex& index, const QVariant& value, i
|
|||||||
|
|
||||||
QString FileIgnoreProxy::relPath(const QString& path) const
|
QString FileIgnoreProxy::relPath(const QString& path) const
|
||||||
{
|
{
|
||||||
return QDir(root).relativeFilePath(path);
|
return QDir(m_root).relativeFilePath(path);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool FileIgnoreProxy::setFilterState(QModelIndex index, Qt::CheckState state)
|
bool FileIgnoreProxy::setFilterState(QModelIndex index, Qt::CheckState state)
|
||||||
@ -146,18 +145,18 @@ bool FileIgnoreProxy::setFilterState(QModelIndex index, Qt::CheckState state)
|
|||||||
bool changed = false;
|
bool changed = false;
|
||||||
if (state == Qt::Unchecked) {
|
if (state == Qt::Unchecked) {
|
||||||
// blocking a path
|
// blocking a path
|
||||||
auto& node = blocked.insert(blockedPath);
|
auto& node = m_blocked.insert(blockedPath);
|
||||||
// get rid of all blocked nodes below
|
// get rid of all blocked nodes below
|
||||||
node.clear();
|
node.clear();
|
||||||
changed = true;
|
changed = true;
|
||||||
} else if (state == Qt::Checked || state == Qt::PartiallyChecked) {
|
} else if (state == Qt::Checked || state == Qt::PartiallyChecked) {
|
||||||
if (!blocked.remove(blockedPath)) {
|
if (!m_blocked.remove(blockedPath)) {
|
||||||
auto cover = blocked.cover(blockedPath);
|
auto cover = m_blocked.cover(blockedPath);
|
||||||
qDebug() << "Blocked by cover" << cover;
|
qDebug() << "Blocked by cover" << cover;
|
||||||
// uncover
|
// uncover
|
||||||
blocked.remove(cover);
|
m_blocked.remove(cover);
|
||||||
// block all contents, except for any cover
|
// block all contents, except for any cover
|
||||||
QModelIndex rootIndex = fsm->index(FS::PathCombine(root, cover));
|
QModelIndex rootIndex = fsm->index(FS::PathCombine(m_root, cover));
|
||||||
QModelIndex doing = rootIndex;
|
QModelIndex doing = rootIndex;
|
||||||
int row = 0;
|
int row = 0;
|
||||||
QStack<QModelIndex> todo;
|
QStack<QModelIndex> todo;
|
||||||
@ -179,7 +178,7 @@ bool FileIgnoreProxy::setFilterState(QModelIndex index, Qt::CheckState state)
|
|||||||
todo.push(node);
|
todo.push(node);
|
||||||
} else {
|
} else {
|
||||||
// or just block this one.
|
// or just block this one.
|
||||||
blocked.insert(relpath);
|
m_blocked.insert(relpath);
|
||||||
}
|
}
|
||||||
row++;
|
row++;
|
||||||
}
|
}
|
||||||
@ -229,7 +228,7 @@ bool FileIgnoreProxy::shouldExpand(QModelIndex index)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
auto blockedPath = relPath(fsm->filePath(sourceIndex));
|
auto blockedPath = relPath(fsm->filePath(sourceIndex));
|
||||||
auto found = blocked.find(blockedPath);
|
auto found = m_blocked.find(blockedPath);
|
||||||
if (found) {
|
if (found) {
|
||||||
return !found->leaf();
|
return !found->leaf();
|
||||||
}
|
}
|
||||||
@ -239,8 +238,8 @@ bool FileIgnoreProxy::shouldExpand(QModelIndex index)
|
|||||||
void FileIgnoreProxy::setBlockedPaths(QStringList paths)
|
void FileIgnoreProxy::setBlockedPaths(QStringList paths)
|
||||||
{
|
{
|
||||||
beginResetModel();
|
beginResetModel();
|
||||||
blocked.clear();
|
m_blocked.clear();
|
||||||
blocked.insert(paths);
|
m_blocked.insert(paths);
|
||||||
endResetModel();
|
endResetModel();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -272,5 +271,30 @@ bool FileIgnoreProxy::ignoreFile(QFileInfo fileInfo) const
|
|||||||
|
|
||||||
bool FileIgnoreProxy::filterFile(const QString& fileName) const
|
bool FileIgnoreProxy::filterFile(const QString& fileName) const
|
||||||
{
|
{
|
||||||
return blocked.covers(fileName) || ignoreFile(QFileInfo(QDir(root), fileName));
|
return m_blocked.covers(fileName) || ignoreFile(QFileInfo(QDir(m_root), fileName));
|
||||||
|
}
|
||||||
|
|
||||||
|
void FileIgnoreProxy::loadBlockedPathsFromFile(const QString& fileName)
|
||||||
|
{
|
||||||
|
QFile ignoreFile(fileName);
|
||||||
|
if (!ignoreFile.open(QIODevice::ReadOnly)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
auto ignoreData = ignoreFile.readAll();
|
||||||
|
auto string = QString::fromUtf8(ignoreData);
|
||||||
|
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
||||||
|
setBlockedPaths(string.split('\n', Qt::SkipEmptyParts));
|
||||||
|
#else
|
||||||
|
setBlockedPaths(string.split('\n', QString::SkipEmptyParts));
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
|
void FileIgnoreProxy::saveBlockedPathsToFile(const QString& fileName)
|
||||||
|
{
|
||||||
|
auto ignoreData = blockedPaths().toStringList().join('\n').toUtf8();
|
||||||
|
try {
|
||||||
|
FS::write(fileName, ignoreData);
|
||||||
|
} catch (const Exception& e) {
|
||||||
|
qWarning() << e.cause();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -61,8 +61,8 @@ class FileIgnoreProxy : public QSortFilterProxyModel {
|
|||||||
|
|
||||||
void setBlockedPaths(QStringList paths);
|
void setBlockedPaths(QStringList paths);
|
||||||
|
|
||||||
inline const SeparatorPrefixTree<'/'>& blockedPaths() const { return blocked; }
|
inline const SeparatorPrefixTree<'/'>& blockedPaths() const { return m_blocked; }
|
||||||
inline SeparatorPrefixTree<'/'>& blockedPaths() { return blocked; }
|
inline SeparatorPrefixTree<'/'>& blockedPaths() { return m_blocked; }
|
||||||
|
|
||||||
// list of file names that need to be removed completely from model
|
// list of file names that need to be removed completely from model
|
||||||
inline QStringList& ignoreFilesWithName() { return m_ignoreFiles; }
|
inline QStringList& ignoreFilesWithName() { return m_ignoreFiles; }
|
||||||
@ -71,6 +71,10 @@ class FileIgnoreProxy : public QSortFilterProxyModel {
|
|||||||
|
|
||||||
bool filterFile(const QString& fileName) const;
|
bool filterFile(const QString& fileName) const;
|
||||||
|
|
||||||
|
void loadBlockedPathsFromFile(const QString& fileName);
|
||||||
|
|
||||||
|
void saveBlockedPathsToFile(const QString& fileName);
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
bool filterAcceptsColumn(int source_column, const QModelIndex& source_parent) const;
|
bool filterAcceptsColumn(int source_column, const QModelIndex& source_parent) const;
|
||||||
bool filterAcceptsRow(int source_row, const QModelIndex& source_parent) const;
|
bool filterAcceptsRow(int source_row, const QModelIndex& source_parent) const;
|
||||||
@ -78,8 +82,8 @@ class FileIgnoreProxy : public QSortFilterProxyModel {
|
|||||||
bool ignoreFile(QFileInfo file) const;
|
bool ignoreFile(QFileInfo file) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
const QString root;
|
const QString m_root;
|
||||||
SeparatorPrefixTree<'/'> blocked;
|
SeparatorPrefixTree<'/'> m_blocked;
|
||||||
QStringList m_ignoreFiles;
|
QStringList m_ignoreFiles;
|
||||||
SeparatorPrefixTree<'/'> m_ignoreFilePaths;
|
SeparatorPrefixTree<'/'> m_ignoreFilePaths;
|
||||||
};
|
};
|
||||||
|
@ -60,40 +60,40 @@
|
|||||||
#include "SeparatorPrefixTree.h"
|
#include "SeparatorPrefixTree.h"
|
||||||
|
|
||||||
ExportInstanceDialog::ExportInstanceDialog(InstancePtr instance, QWidget* parent)
|
ExportInstanceDialog::ExportInstanceDialog(InstancePtr instance, QWidget* parent)
|
||||||
: QDialog(parent), ui(new Ui::ExportInstanceDialog), m_instance(instance)
|
: QDialog(parent), m_ui(new Ui::ExportInstanceDialog), m_instance(instance)
|
||||||
{
|
{
|
||||||
ui->setupUi(this);
|
m_ui->setupUi(this);
|
||||||
auto model = new QFileSystemModel(this);
|
auto model = new QFileSystemModel(this);
|
||||||
model->setIconProvider(&icons);
|
model->setIconProvider(&m_icons);
|
||||||
auto root = instance->instanceRoot();
|
auto root = instance->instanceRoot();
|
||||||
proxyModel = new FileIgnoreProxy(root, this);
|
m_proxyModel = new FileIgnoreProxy(root, this);
|
||||||
proxyModel->setSourceModel(model);
|
m_proxyModel->setSourceModel(model);
|
||||||
auto prefix = QDir(instance->instanceRoot()).relativeFilePath(instance->gameRoot());
|
auto prefix = QDir(instance->instanceRoot()).relativeFilePath(instance->gameRoot());
|
||||||
proxyModel->ignoreFilesWithPath().insert({ FS::PathCombine(prefix, "logs"), FS::PathCombine(prefix, "crash-reports") });
|
for (auto path : { "logs", "crash-reports", ".cache", ".fabric", ".quilt" }) {
|
||||||
proxyModel->ignoreFilesWithName().append({ ".DS_Store", "thumbs.db", "Thumbs.db" });
|
m_proxyModel->ignoreFilesWithPath().insert(FS::PathCombine(prefix, path));
|
||||||
proxyModel->ignoreFilesWithPath().insert(
|
}
|
||||||
{ FS::PathCombine(prefix, ".cache"), FS::PathCombine(prefix, ".fabric"), FS::PathCombine(prefix, ".quilt") });
|
m_proxyModel->ignoreFilesWithName().append({ ".DS_Store", "thumbs.db", "Thumbs.db" });
|
||||||
loadPackIgnore();
|
m_proxyModel->loadBlockedPathsFromFile(ignoreFileName());
|
||||||
|
|
||||||
ui->treeView->setModel(proxyModel);
|
m_ui->treeView->setModel(m_proxyModel);
|
||||||
ui->treeView->setRootIndex(proxyModel->mapFromSource(model->index(root)));
|
m_ui->treeView->setRootIndex(m_proxyModel->mapFromSource(model->index(root)));
|
||||||
ui->treeView->sortByColumn(0, Qt::AscendingOrder);
|
m_ui->treeView->sortByColumn(0, Qt::AscendingOrder);
|
||||||
|
|
||||||
connect(proxyModel, SIGNAL(rowsInserted(QModelIndex, int, int)), SLOT(rowsInserted(QModelIndex, int, int)));
|
connect(m_proxyModel, SIGNAL(rowsInserted(QModelIndex, int, int)), SLOT(rowsInserted(QModelIndex, int, int)));
|
||||||
|
|
||||||
model->setFilter(QDir::AllEntries | QDir::NoDotAndDotDot | QDir::AllDirs | QDir::Hidden);
|
model->setFilter(QDir::AllEntries | QDir::NoDotAndDotDot | QDir::AllDirs | QDir::Hidden);
|
||||||
model->setRootPath(root);
|
model->setRootPath(root);
|
||||||
auto headerView = ui->treeView->header();
|
auto headerView = m_ui->treeView->header();
|
||||||
headerView->setSectionResizeMode(QHeaderView::ResizeToContents);
|
headerView->setSectionResizeMode(QHeaderView::ResizeToContents);
|
||||||
headerView->setSectionResizeMode(0, QHeaderView::Stretch);
|
headerView->setSectionResizeMode(0, QHeaderView::Stretch);
|
||||||
|
|
||||||
ui->buttonBox->button(QDialogButtonBox::Cancel)->setText(tr("Cancel"));
|
m_ui->buttonBox->button(QDialogButtonBox::Cancel)->setText(tr("Cancel"));
|
||||||
ui->buttonBox->button(QDialogButtonBox::Ok)->setText(tr("OK"));
|
m_ui->buttonBox->button(QDialogButtonBox::Ok)->setText(tr("OK"));
|
||||||
}
|
}
|
||||||
|
|
||||||
ExportInstanceDialog::~ExportInstanceDialog()
|
ExportInstanceDialog::~ExportInstanceDialog()
|
||||||
{
|
{
|
||||||
delete ui;
|
delete m_ui;
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Save icon to instance's folder is needed
|
/// Save icon to instance's folder is needed
|
||||||
@ -144,7 +144,7 @@ void ExportInstanceDialog::doExport()
|
|||||||
|
|
||||||
auto files = QFileInfoList();
|
auto files = QFileInfoList();
|
||||||
if (!MMCZip::collectFileListRecursively(m_instance->instanceRoot(), nullptr, &files,
|
if (!MMCZip::collectFileListRecursively(m_instance->instanceRoot(), nullptr, &files,
|
||||||
std::bind(&FileIgnoreProxy::filterFile, proxyModel, std::placeholders::_1))) {
|
std::bind(&FileIgnoreProxy::filterFile, m_proxyModel, std::placeholders::_1))) {
|
||||||
QMessageBox::warning(this, tr("Error"), tr("Unable to export instance"));
|
QMessageBox::warning(this, tr("Error"), tr("Unable to export instance"));
|
||||||
QDialog::done(QDialog::Rejected);
|
QDialog::done(QDialog::Rejected);
|
||||||
return;
|
return;
|
||||||
@ -164,7 +164,7 @@ void ExportInstanceDialog::doExport()
|
|||||||
|
|
||||||
void ExportInstanceDialog::done(int result)
|
void ExportInstanceDialog::done(int result)
|
||||||
{
|
{
|
||||||
savePackIgnore();
|
m_proxyModel->saveBlockedPathsToFile(ignoreFileName());
|
||||||
if (result == QDialog::Accepted) {
|
if (result == QDialog::Accepted) {
|
||||||
doExport();
|
doExport();
|
||||||
return;
|
return;
|
||||||
@ -176,13 +176,13 @@ void ExportInstanceDialog::rowsInserted(QModelIndex parent, int top, int bottom)
|
|||||||
{
|
{
|
||||||
// WARNING: possible off-by-one?
|
// WARNING: possible off-by-one?
|
||||||
for (int i = top; i < bottom; i++) {
|
for (int i = top; i < bottom; i++) {
|
||||||
auto node = proxyModel->index(i, 0, parent);
|
auto node = m_proxyModel->index(i, 0, parent);
|
||||||
if (proxyModel->shouldExpand(node)) {
|
if (m_proxyModel->shouldExpand(node)) {
|
||||||
auto expNode = node.parent();
|
auto expNode = node.parent();
|
||||||
if (!expNode.isValid()) {
|
if (!expNode.isValid()) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
ui->treeView->expand(node);
|
m_ui->treeView->expand(node);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -191,30 +191,3 @@ QString ExportInstanceDialog::ignoreFileName()
|
|||||||
{
|
{
|
||||||
return FS::PathCombine(m_instance->instanceRoot(), ".packignore");
|
return FS::PathCombine(m_instance->instanceRoot(), ".packignore");
|
||||||
}
|
}
|
||||||
|
|
||||||
void ExportInstanceDialog::loadPackIgnore()
|
|
||||||
{
|
|
||||||
auto filename = ignoreFileName();
|
|
||||||
QFile ignoreFile(filename);
|
|
||||||
if (!ignoreFile.open(QIODevice::ReadOnly)) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
auto ignoreData = ignoreFile.readAll();
|
|
||||||
auto string = QString::fromUtf8(ignoreData);
|
|
||||||
#if QT_VERSION >= QT_VERSION_CHECK(5, 14, 0)
|
|
||||||
proxyModel->setBlockedPaths(string.split('\n', Qt::SkipEmptyParts));
|
|
||||||
#else
|
|
||||||
proxyModel->setBlockedPaths(string.split('\n', QString::SkipEmptyParts));
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
void ExportInstanceDialog::savePackIgnore()
|
|
||||||
{
|
|
||||||
auto ignoreData = proxyModel->blockedPaths().toStringList().join('\n').toUtf8();
|
|
||||||
auto filename = ignoreFileName();
|
|
||||||
try {
|
|
||||||
FS::write(filename, ignoreData);
|
|
||||||
} catch (const Exception& e) {
|
|
||||||
qWarning() << e.cause();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -60,15 +60,13 @@ class ExportInstanceDialog : public QDialog {
|
|||||||
|
|
||||||
private:
|
private:
|
||||||
void doExport();
|
void doExport();
|
||||||
void loadPackIgnore();
|
|
||||||
void savePackIgnore();
|
|
||||||
QString ignoreFileName();
|
QString ignoreFileName();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
Ui::ExportInstanceDialog* ui;
|
Ui::ExportInstanceDialog* m_ui;
|
||||||
InstancePtr m_instance;
|
InstancePtr m_instance;
|
||||||
FileIgnoreProxy* proxyModel;
|
FileIgnoreProxy* m_proxyModel;
|
||||||
FastFileIconProvider icons;
|
FastFileIconProvider m_icons;
|
||||||
|
|
||||||
private slots:
|
private slots:
|
||||||
void rowsInserted(QModelIndex parent, int top, int bottom);
|
void rowsInserted(QModelIndex parent, int top, int bottom);
|
||||||
|
@ -29,104 +29,102 @@
|
|||||||
#include <QJsonDocument>
|
#include <QJsonDocument>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
#include <QPushButton>
|
#include <QPushButton>
|
||||||
#include "FastFileIconProvider.h"
|
|
||||||
#include "FileSystem.h"
|
#include "FileSystem.h"
|
||||||
#include "MMCZip.h"
|
#include "MMCZip.h"
|
||||||
#include "modplatform/modrinth/ModrinthPackExportTask.h"
|
#include "modplatform/modrinth/ModrinthPackExportTask.h"
|
||||||
|
|
||||||
ExportPackDialog::ExportPackDialog(InstancePtr instance, QWidget* parent, ModPlatform::ResourceProvider provider)
|
ExportPackDialog::ExportPackDialog(InstancePtr instance, QWidget* parent, ModPlatform::ResourceProvider provider)
|
||||||
: QDialog(parent), instance(instance), ui(new Ui::ExportPackDialog), m_provider(provider)
|
: QDialog(parent), m_instance(instance), m_ui(new Ui::ExportPackDialog), m_provider(provider)
|
||||||
{
|
{
|
||||||
Q_ASSERT(m_provider == ModPlatform::ResourceProvider::MODRINTH || m_provider == ModPlatform::ResourceProvider::FLAME);
|
Q_ASSERT(m_provider == ModPlatform::ResourceProvider::MODRINTH || m_provider == ModPlatform::ResourceProvider::FLAME);
|
||||||
|
|
||||||
ui->setupUi(this);
|
m_ui->setupUi(this);
|
||||||
ui->name->setPlaceholderText(instance->name());
|
m_ui->name->setPlaceholderText(instance->name());
|
||||||
ui->name->setText(instance->settings()->get("ExportName").toString());
|
m_ui->name->setText(instance->settings()->get("ExportName").toString());
|
||||||
ui->version->setText(instance->settings()->get("ExportVersion").toString());
|
m_ui->version->setText(instance->settings()->get("ExportVersion").toString());
|
||||||
ui->optionalFiles->setChecked(instance->settings()->get("ExportOptionalFiles").toBool());
|
m_ui->optionalFiles->setChecked(instance->settings()->get("ExportOptionalFiles").toBool());
|
||||||
|
|
||||||
if (m_provider == ModPlatform::ResourceProvider::MODRINTH) {
|
if (m_provider == ModPlatform::ResourceProvider::MODRINTH) {
|
||||||
setWindowTitle(tr("Export Modrinth Pack"));
|
setWindowTitle(tr("Export Modrinth Pack"));
|
||||||
|
|
||||||
ui->authorLabel->hide();
|
m_ui->authorLabel->hide();
|
||||||
ui->author->hide();
|
m_ui->author->hide();
|
||||||
|
|
||||||
ui->summary->setPlainText(instance->settings()->get("ExportSummary").toString());
|
m_ui->summary->setPlainText(instance->settings()->get("ExportSummary").toString());
|
||||||
} else {
|
} else {
|
||||||
setWindowTitle(tr("Export CurseForge Pack"));
|
setWindowTitle(tr("Export CurseForge Pack"));
|
||||||
|
|
||||||
ui->summaryLabel->hide();
|
m_ui->summaryLabel->hide();
|
||||||
ui->summary->hide();
|
m_ui->summary->hide();
|
||||||
|
|
||||||
ui->author->setText(instance->settings()->get("ExportAuthor").toString());
|
m_ui->author->setText(instance->settings()->get("ExportAuthor").toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
// ensure a valid pack is generated
|
// ensure a valid pack is generated
|
||||||
// the name and version fields mustn't be empty
|
// the name and version fields mustn't be empty
|
||||||
connect(ui->name, &QLineEdit::textEdited, this, &ExportPackDialog::validate);
|
connect(m_ui->name, &QLineEdit::textEdited, this, &ExportPackDialog::validate);
|
||||||
connect(ui->version, &QLineEdit::textEdited, this, &ExportPackDialog::validate);
|
connect(m_ui->version, &QLineEdit::textEdited, this, &ExportPackDialog::validate);
|
||||||
// the instance name can technically be empty
|
// the instance name can technically be empty
|
||||||
validate();
|
validate();
|
||||||
|
|
||||||
QFileSystemModel* model = new QFileSystemModel(this);
|
QFileSystemModel* model = new QFileSystemModel(this);
|
||||||
model->setIconProvider(&icons);
|
model->setIconProvider(&m_icons);
|
||||||
|
|
||||||
// use the game root - everything outside cannot be exported
|
// use the game root - everything outside cannot be exported
|
||||||
const QDir root(instance->gameRoot());
|
const QDir instanceRoot(instance->instanceRoot());
|
||||||
proxy = new FileIgnoreProxy(instance->gameRoot(), this);
|
m_proxy = new FileIgnoreProxy(instance->instanceRoot(), this);
|
||||||
proxy->ignoreFilesWithPath().insert({ "logs", "crash-reports", ".cache", ".fabric", ".quilt" });
|
auto prefix = QDir(instance->instanceRoot()).relativeFilePath(instance->gameRoot());
|
||||||
proxy->ignoreFilesWithName().append({ ".DS_Store", "thumbs.db", "Thumbs.db" });
|
for (auto path : { "logs", "crash-reports", ".cache", ".fabric", ".quilt" }) {
|
||||||
proxy->setSourceModel(model);
|
m_proxy->ignoreFilesWithPath().insert(FS::PathCombine(prefix, path));
|
||||||
|
}
|
||||||
|
m_proxy->ignoreFilesWithName().append({ ".DS_Store", "thumbs.db", "Thumbs.db" });
|
||||||
|
m_proxy->setSourceModel(model);
|
||||||
|
m_proxy->loadBlockedPathsFromFile(ignoreFileName());
|
||||||
|
|
||||||
const QDir::Filters filter(QDir::AllEntries | QDir::NoDotAndDotDot | QDir::AllDirs | QDir::Hidden);
|
const QDir::Filters filter(QDir::AllEntries | QDir::NoDotAndDotDot | QDir::AllDirs | QDir::Hidden);
|
||||||
|
|
||||||
for (const QString& file : root.entryList(filter)) {
|
|
||||||
if (!(file == "mods" || file == "coremods" || file == "datapacks" || file == "config" || file == "options.txt" ||
|
|
||||||
file == "servers.dat"))
|
|
||||||
proxy->blockedPaths().insert(file);
|
|
||||||
}
|
|
||||||
|
|
||||||
MinecraftInstance* mcInstance = dynamic_cast<MinecraftInstance*>(instance.get());
|
MinecraftInstance* mcInstance = dynamic_cast<MinecraftInstance*>(instance.get());
|
||||||
if (mcInstance) {
|
if (mcInstance) {
|
||||||
for (auto& resourceModel : mcInstance->resourceLists())
|
for (auto& resourceModel : mcInstance->resourceLists())
|
||||||
if (resourceModel->indexDir().exists())
|
if (resourceModel->indexDir().exists())
|
||||||
proxy->ignoreFilesWithPath().insert(root.relativeFilePath(resourceModel->indexDir().absolutePath()));
|
m_proxy->ignoreFilesWithPath().insert(instanceRoot.relativeFilePath(resourceModel->indexDir().absolutePath()));
|
||||||
}
|
}
|
||||||
|
|
||||||
ui->files->setModel(proxy);
|
m_ui->files->setModel(m_proxy);
|
||||||
ui->files->setRootIndex(proxy->mapFromSource(model->index(instance->gameRoot())));
|
m_ui->files->setRootIndex(m_proxy->mapFromSource(model->index(instance->gameRoot())));
|
||||||
ui->files->sortByColumn(0, Qt::AscendingOrder);
|
m_ui->files->sortByColumn(0, Qt::AscendingOrder);
|
||||||
|
|
||||||
model->setFilter(filter);
|
model->setFilter(filter);
|
||||||
model->setRootPath(instance->gameRoot());
|
model->setRootPath(instance->gameRoot());
|
||||||
|
|
||||||
QHeaderView* headerView = ui->files->header();
|
QHeaderView* headerView = m_ui->files->header();
|
||||||
headerView->setSectionResizeMode(QHeaderView::ResizeToContents);
|
headerView->setSectionResizeMode(QHeaderView::ResizeToContents);
|
||||||
headerView->setSectionResizeMode(0, QHeaderView::Stretch);
|
headerView->setSectionResizeMode(0, QHeaderView::Stretch);
|
||||||
|
|
||||||
ui->buttonBox->button(QDialogButtonBox::Cancel)->setText(tr("Cancel"));
|
m_ui->buttonBox->button(QDialogButtonBox::Cancel)->setText(tr("Cancel"));
|
||||||
ui->buttonBox->button(QDialogButtonBox::Ok)->setText(tr("OK"));
|
m_ui->buttonBox->button(QDialogButtonBox::Ok)->setText(tr("OK"));
|
||||||
}
|
}
|
||||||
|
|
||||||
ExportPackDialog::~ExportPackDialog()
|
ExportPackDialog::~ExportPackDialog()
|
||||||
{
|
{
|
||||||
delete ui;
|
delete m_ui;
|
||||||
}
|
}
|
||||||
|
|
||||||
void ExportPackDialog::done(int result)
|
void ExportPackDialog::done(int result)
|
||||||
{
|
{
|
||||||
auto settings = instance->settings();
|
m_proxy->saveBlockedPathsToFile(ignoreFileName());
|
||||||
settings->set("ExportName", ui->name->text());
|
auto settings = m_instance->settings();
|
||||||
settings->set("ExportVersion", ui->version->text());
|
settings->set("ExportName", m_ui->name->text());
|
||||||
settings->set("ExportOptionalFiles", ui->optionalFiles->isChecked());
|
settings->set("ExportVersion", m_ui->version->text());
|
||||||
|
settings->set("ExportOptionalFiles", m_ui->optionalFiles->isChecked());
|
||||||
|
|
||||||
if (m_provider == ModPlatform::ResourceProvider::MODRINTH)
|
if (m_provider == ModPlatform::ResourceProvider::MODRINTH)
|
||||||
settings->set("ExportSummary", ui->summary->toPlainText());
|
settings->set("ExportSummary", m_ui->summary->toPlainText());
|
||||||
else
|
else
|
||||||
settings->set("ExportAuthor", ui->author->text());
|
settings->set("ExportAuthor", m_ui->author->text());
|
||||||
|
|
||||||
if (result == Accepted) {
|
if (result == Accepted) {
|
||||||
const QString name = ui->name->text().isEmpty() ? instance->name() : ui->name->text();
|
const QString name = m_ui->name->text().isEmpty() ? m_instance->name() : m_ui->name->text();
|
||||||
const QString filename = FS::RemoveInvalidFilenameChars(name);
|
const QString filename = FS::RemoveInvalidFilenameChars(name);
|
||||||
|
|
||||||
QString output;
|
QString output;
|
||||||
@ -148,11 +146,11 @@ void ExportPackDialog::done(int result)
|
|||||||
|
|
||||||
Task* task;
|
Task* task;
|
||||||
if (m_provider == ModPlatform::ResourceProvider::MODRINTH) {
|
if (m_provider == ModPlatform::ResourceProvider::MODRINTH) {
|
||||||
task = new ModrinthPackExportTask(name, ui->version->text(), ui->summary->toPlainText(), ui->optionalFiles->isChecked(),
|
task = new ModrinthPackExportTask(name, m_ui->version->text(), m_ui->summary->toPlainText(), m_ui->optionalFiles->isChecked(),
|
||||||
instance, output, std::bind(&FileIgnoreProxy::filterFile, proxy, std::placeholders::_1));
|
m_instance, output, std::bind(&FileIgnoreProxy::filterFile, m_proxy, std::placeholders::_1));
|
||||||
} else {
|
} else {
|
||||||
task = new FlamePackExportTask(name, ui->version->text(), ui->author->text(), ui->optionalFiles->isChecked(), instance, output,
|
task = new FlamePackExportTask(name, m_ui->version->text(), m_ui->author->text(), m_ui->optionalFiles->isChecked(), m_instance,
|
||||||
std::bind(&FileIgnoreProxy::filterFile, proxy, std::placeholders::_1));
|
output, std::bind(&FileIgnoreProxy::filterFile, m_proxy, std::placeholders::_1));
|
||||||
}
|
}
|
||||||
|
|
||||||
connect(task, &Task::failed,
|
connect(task, &Task::failed,
|
||||||
@ -174,6 +172,11 @@ void ExportPackDialog::done(int result)
|
|||||||
|
|
||||||
void ExportPackDialog::validate()
|
void ExportPackDialog::validate()
|
||||||
{
|
{
|
||||||
ui->buttonBox->button(QDialogButtonBox::Ok)
|
m_ui->buttonBox->button(QDialogButtonBox::Ok)
|
||||||
->setDisabled(m_provider == ModPlatform::ResourceProvider::MODRINTH && ui->version->text().isEmpty());
|
->setDisabled(m_provider == ModPlatform::ResourceProvider::MODRINTH && m_ui->version->text().isEmpty());
|
||||||
|
}
|
||||||
|
|
||||||
|
QString ExportPackDialog::ignoreFileName()
|
||||||
|
{
|
||||||
|
return FS::PathCombine(m_instance->instanceRoot(), ".packignore");
|
||||||
}
|
}
|
||||||
|
@ -41,9 +41,12 @@ class ExportPackDialog : public QDialog {
|
|||||||
void validate();
|
void validate();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
const InstancePtr instance;
|
QString ignoreFileName();
|
||||||
Ui::ExportPackDialog* ui;
|
|
||||||
FileIgnoreProxy* proxy;
|
private:
|
||||||
FastFileIconProvider icons;
|
const InstancePtr m_instance;
|
||||||
|
Ui::ExportPackDialog* m_ui;
|
||||||
|
FileIgnoreProxy* m_proxy;
|
||||||
|
FastFileIconProvider m_icons;
|
||||||
const ModPlatform::ResourceProvider m_provider;
|
const ModPlatform::ResourceProvider m_provider;
|
||||||
};
|
};
|
||||||
|
Loading…
x
Reference in New Issue
Block a user