Merge branch 'develop' of github.com:MultiMC/MultiMC5 into develop
This commit is contained in:
commit
7d5fb1e99b
@ -144,6 +144,8 @@ SET(MultiMC_CHANLIST_URL "" CACHE STRING "URL for the channel list.")
|
|||||||
# Updater enabled?
|
# Updater enabled?
|
||||||
SET(MultiMC_UPDATER false CACHE BOOL "Whether or not the update system is enabled. If this is enabled, you must also set MultiMC_CHANLIST_URL and MultiMC_VERSION_CHANNEL in order for it to work properly.")
|
SET(MultiMC_UPDATER false CACHE BOOL "Whether or not the update system is enabled. If this is enabled, you must also set MultiMC_CHANLIST_URL and MultiMC_VERSION_CHANNEL in order for it to work properly.")
|
||||||
|
|
||||||
|
# Notification URL
|
||||||
|
SET(MultiMC_NOTIFICATION_URL "" CACHE STRING "URL for checking for notifications.")
|
||||||
|
|
||||||
# Build a version string to display in the configure logs.
|
# Build a version string to display in the configure logs.
|
||||||
SET(MultiMC_VERSION_STRING "${MultiMC_VERSION_MAJOR}.${MultiMC_VERSION_MINOR}")
|
SET(MultiMC_VERSION_STRING "${MultiMC_VERSION_MAJOR}.${MultiMC_VERSION_MINOR}")
|
||||||
@ -337,6 +339,8 @@ logic/updater/UpdateChecker.h
|
|||||||
logic/updater/UpdateChecker.cpp
|
logic/updater/UpdateChecker.cpp
|
||||||
logic/updater/DownloadUpdateTask.h
|
logic/updater/DownloadUpdateTask.h
|
||||||
logic/updater/DownloadUpdateTask.cpp
|
logic/updater/DownloadUpdateTask.cpp
|
||||||
|
logic/updater/NotificationChecker.h
|
||||||
|
logic/updater/NotificationChecker.cpp
|
||||||
|
|
||||||
# News System
|
# News System
|
||||||
logic/news/NewsChecker.h
|
logic/news/NewsChecker.h
|
||||||
|
95
MultiMC.cpp
95
MultiMC.cpp
@ -26,6 +26,7 @@
|
|||||||
#include "logic/JavaUtils.h"
|
#include "logic/JavaUtils.h"
|
||||||
|
|
||||||
#include "logic/updater/UpdateChecker.h"
|
#include "logic/updater/UpdateChecker.h"
|
||||||
|
#include "logic/updater/NotificationChecker.h"
|
||||||
|
|
||||||
#include "pathutils.h"
|
#include "pathutils.h"
|
||||||
#include "cmdutils.h"
|
#include "cmdutils.h"
|
||||||
@ -47,7 +48,7 @@
|
|||||||
|
|
||||||
using namespace Util::Commandline;
|
using namespace Util::Commandline;
|
||||||
|
|
||||||
MultiMC::MultiMC(int &argc, char **argv, const QString &root)
|
MultiMC::MultiMC(int &argc, char **argv, const QString &data_dir_override)
|
||||||
: QApplication(argc, argv), m_version{VERSION_MAJOR, VERSION_MINOR, VERSION_BUILD,
|
: QApplication(argc, argv), m_version{VERSION_MAJOR, VERSION_MINOR, VERSION_BUILD,
|
||||||
VERSION_CHANNEL, VERSION_BUILD_TYPE}
|
VERSION_CHANNEL, VERSION_BUILD_TYPE}
|
||||||
{
|
{
|
||||||
@ -60,10 +61,6 @@ MultiMC::MultiMC(int &argc, char **argv, const QString &root)
|
|||||||
// Don't quit on hiding the last window
|
// Don't quit on hiding the last window
|
||||||
this->setQuitOnLastWindowClosed(false);
|
this->setQuitOnLastWindowClosed(false);
|
||||||
|
|
||||||
// Print app header
|
|
||||||
std::cout << "MultiMC 5" << std::endl;
|
|
||||||
std::cout << "(c) 2013 MultiMC Contributors" << std::endl << std::endl;
|
|
||||||
|
|
||||||
// Commandline parsing
|
// Commandline parsing
|
||||||
QHash<QString, QVariant> args;
|
QHash<QString, QVariant> args;
|
||||||
{
|
{
|
||||||
@ -82,16 +79,6 @@ MultiMC::MultiMC(int &argc, char **argv, const QString &root)
|
|||||||
parser.addShortOpt("dir", 'd');
|
parser.addShortOpt("dir", 'd');
|
||||||
parser.addDocumentation("dir", "use the supplied directory as MultiMC root instead of "
|
parser.addDocumentation("dir", "use the supplied directory as MultiMC root instead of "
|
||||||
"the binary location (use '.' for current)");
|
"the binary location (use '.' for current)");
|
||||||
// --update
|
|
||||||
parser.addOption("update");
|
|
||||||
parser.addShortOpt("update", 'u');
|
|
||||||
parser.addDocumentation("update", "replaces the given file with the running executable",
|
|
||||||
"<path>");
|
|
||||||
// --quietupdate
|
|
||||||
parser.addSwitch("quietupdate");
|
|
||||||
parser.addShortOpt("quietupdate", 'U');
|
|
||||||
parser.addDocumentation("quietupdate",
|
|
||||||
"doesn't restart MultiMC after installing updates");
|
|
||||||
// WARNING: disabled until further notice
|
// WARNING: disabled until further notice
|
||||||
/*
|
/*
|
||||||
// --launch
|
// --launch
|
||||||
@ -129,41 +116,76 @@ MultiMC::MultiMC(int &argc, char **argv, const QString &root)
|
|||||||
m_status = MultiMC::Succeeded;
|
m_status = MultiMC::Succeeded;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
// update
|
origcwdPath = QDir::currentPath();
|
||||||
// Note: cwd is always the current executable path!
|
binPath = applicationDirPath();
|
||||||
if (!args["update"].isNull())
|
QString adjustedBy;
|
||||||
|
// change directory
|
||||||
|
QString dirParam = args["dir"].toString();
|
||||||
|
if (!data_dir_override.isEmpty())
|
||||||
{
|
{
|
||||||
std::cout << "Performing MultiMC update: " << qPrintable(args["update"].toString())
|
// the override is used for tests (although dirparam would be enough...)
|
||||||
<< std::endl;
|
// TODO: remove the need for this extra logic
|
||||||
QString cwd = QDir::currentPath();
|
adjustedBy += "Test override " + data_dir_override;
|
||||||
QDir::setCurrent(applicationDirPath());
|
dataPath = data_dir_override;
|
||||||
QFile file(applicationFilePath());
|
}
|
||||||
file.copy(args["update"].toString());
|
else if (!dirParam.isEmpty())
|
||||||
if (args["quietupdate"].toBool())
|
|
||||||
{
|
{
|
||||||
m_status = MultiMC::Succeeded;
|
// the dir param. it makes multimc data path point to whatever the user specified
|
||||||
|
// on command line
|
||||||
|
adjustedBy += "Command line " + dirParam;
|
||||||
|
dataPath = dirParam;
|
||||||
|
}
|
||||||
|
if(!ensureFolderPathExists(dataPath) || !QDir::setCurrent(dataPath))
|
||||||
|
{
|
||||||
|
// BAD STUFF. WHAT DO?
|
||||||
|
initLogger();
|
||||||
|
QLOG_ERROR() << "Failed to set work path. Will exit. NOW.";
|
||||||
|
m_status = MultiMC::Failed;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
QDir::setCurrent(cwd);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// change directory
|
{
|
||||||
QDir::setCurrent(
|
#ifdef Q_OS_LINUX
|
||||||
args["dir"].toString().isEmpty()
|
QDir foo(PathCombine(binPath, ".."));
|
||||||
? (root.isEmpty() ? QDir::currentPath() : QDir::current().absoluteFilePath(root))
|
rootPath = foo.absolutePath();
|
||||||
: args["dir"].toString());
|
#elif defined(Q_OS_WIN32)
|
||||||
|
QDir foo(PathCombine(binPath, ".."));
|
||||||
|
rootPath = foo.absolutePath();
|
||||||
|
#elif defined(Q_OS_MAC)
|
||||||
|
QDir foo(PathCombine(binPath, "../.."));
|
||||||
|
rootPath = foo.absolutePath();
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
|
||||||
// init the logger
|
// init the logger
|
||||||
initLogger();
|
initLogger();
|
||||||
|
|
||||||
|
QLOG_INFO() << "MultiMC 5, (c) 2013 MultiMC Contributors";
|
||||||
|
QLOG_INFO() << "Version : " << VERSION_STR;
|
||||||
|
QLOG_INFO() << "Git commit : " << GIT_COMMIT;
|
||||||
|
if (adjustedBy.size())
|
||||||
|
{
|
||||||
|
QLOG_INFO() << "Work dir before adjustment : " << origcwdPath;
|
||||||
|
QLOG_INFO() << "Work dir after adjustment : " << QDir::currentPath();
|
||||||
|
QLOG_INFO() << "Adjusted by : " << adjustedBy;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
QLOG_INFO() << "Work dir : " << QDir::currentPath();
|
||||||
|
}
|
||||||
|
QLOG_INFO() << "Binary path : " << binPath;
|
||||||
|
QLOG_INFO() << "Application root path : " << rootPath;
|
||||||
|
|
||||||
// load settings
|
// load settings
|
||||||
initGlobalSettings();
|
initGlobalSettings();
|
||||||
|
|
||||||
// initialize the updater
|
// initialize the updater
|
||||||
m_updateChecker.reset(new UpdateChecker());
|
m_updateChecker.reset(new UpdateChecker());
|
||||||
|
|
||||||
|
// initialize the notification checker
|
||||||
|
m_notificationChecker.reset(new NotificationChecker());
|
||||||
|
|
||||||
// initialize the news checker
|
// initialize the news checker
|
||||||
m_newsChecker.reset(new NewsChecker(NEWS_RSS_URL));
|
m_newsChecker.reset(new NewsChecker(NEWS_RSS_URL));
|
||||||
|
|
||||||
@ -319,7 +341,7 @@ void MultiMC::initLogger()
|
|||||||
QsLogging::Logger &logger = QsLogging::Logger::instance();
|
QsLogging::Logger &logger = QsLogging::Logger::instance();
|
||||||
logger.setLoggingLevel(QsLogging::TraceLevel);
|
logger.setLoggingLevel(QsLogging::TraceLevel);
|
||||||
m_fileDestination = QsLogging::DestinationFactory::MakeFileDestination(logBase.arg(0));
|
m_fileDestination = QsLogging::DestinationFactory::MakeFileDestination(logBase.arg(0));
|
||||||
m_debugDestination = QsLogging::DestinationFactory::MakeQDebugDestination();
|
m_debugDestination = QsLogging::DestinationFactory::MakeDebugOutputDestination();
|
||||||
logger.addDestination(m_fileDestination.get());
|
logger.addDestination(m_fileDestination.get());
|
||||||
logger.addDestination(m_debugDestination.get());
|
logger.addDestination(m_debugDestination.get());
|
||||||
// log all the things
|
// log all the things
|
||||||
@ -332,6 +354,7 @@ void MultiMC::initGlobalSettings()
|
|||||||
// Updates
|
// Updates
|
||||||
m_settings->registerSetting("UseDevBuilds", false);
|
m_settings->registerSetting("UseDevBuilds", false);
|
||||||
m_settings->registerSetting("AutoUpdate", true);
|
m_settings->registerSetting("AutoUpdate", true);
|
||||||
|
m_settings->registerSetting("ShownNotifications", QString());
|
||||||
|
|
||||||
// FTB
|
// FTB
|
||||||
m_settings->registerSetting("TrackFTBInstances", false);
|
m_settings->registerSetting("TrackFTBInstances", false);
|
||||||
|
35
MultiMC.h
35
MultiMC.h
@ -17,6 +17,7 @@ class QNetworkAccessManager;
|
|||||||
class ForgeVersionList;
|
class ForgeVersionList;
|
||||||
class JavaVersionList;
|
class JavaVersionList;
|
||||||
class UpdateChecker;
|
class UpdateChecker;
|
||||||
|
class NotificationChecker;
|
||||||
class NewsChecker;
|
class NewsChecker;
|
||||||
|
|
||||||
#if defined(MMC)
|
#if defined(MMC)
|
||||||
@ -90,6 +91,11 @@ public:
|
|||||||
return m_updateChecker;
|
return m_updateChecker;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::shared_ptr<NotificationChecker> notificationChecker()
|
||||||
|
{
|
||||||
|
return m_notificationChecker;
|
||||||
|
}
|
||||||
|
|
||||||
std::shared_ptr<NewsChecker> newsChecker()
|
std::shared_ptr<NewsChecker> newsChecker()
|
||||||
{
|
{
|
||||||
return m_newsChecker;
|
return m_newsChecker;
|
||||||
@ -125,6 +131,29 @@ public:
|
|||||||
*/
|
*/
|
||||||
bool openJsonEditor(const QString &filename);
|
bool openJsonEditor(const QString &filename);
|
||||||
|
|
||||||
|
/// this is the root of the 'installation'. Used for automatic updates
|
||||||
|
const QString &root()
|
||||||
|
{
|
||||||
|
return rootPath;
|
||||||
|
}
|
||||||
|
/// this is the where the binary files reside
|
||||||
|
const QString &bin()
|
||||||
|
{
|
||||||
|
return binPath;
|
||||||
|
}
|
||||||
|
/// this is the work/data path. All user data is here.
|
||||||
|
const QString &data()
|
||||||
|
{
|
||||||
|
return dataPath;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* this is the original work path before it was changed by the adjustment mechanism
|
||||||
|
*/
|
||||||
|
const QString &origcwd()
|
||||||
|
{
|
||||||
|
return origcwdPath;
|
||||||
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void initLogger();
|
void initLogger();
|
||||||
|
|
||||||
@ -143,6 +172,7 @@ private:
|
|||||||
std::shared_ptr<SettingsObject> m_settings;
|
std::shared_ptr<SettingsObject> m_settings;
|
||||||
std::shared_ptr<InstanceList> m_instances;
|
std::shared_ptr<InstanceList> m_instances;
|
||||||
std::shared_ptr<UpdateChecker> m_updateChecker;
|
std::shared_ptr<UpdateChecker> m_updateChecker;
|
||||||
|
std::shared_ptr<NotificationChecker> m_notificationChecker;
|
||||||
std::shared_ptr<NewsChecker> m_newsChecker;
|
std::shared_ptr<NewsChecker> m_newsChecker;
|
||||||
std::shared_ptr<MojangAccountList> m_accounts;
|
std::shared_ptr<MojangAccountList> m_accounts;
|
||||||
std::shared_ptr<IconList> m_icons;
|
std::shared_ptr<IconList> m_icons;
|
||||||
@ -157,6 +187,11 @@ private:
|
|||||||
|
|
||||||
QString m_updateOnExitPath;
|
QString m_updateOnExitPath;
|
||||||
|
|
||||||
|
QString rootPath;
|
||||||
|
QString binPath;
|
||||||
|
QString dataPath;
|
||||||
|
QString origcwdPath;
|
||||||
|
|
||||||
Status m_status = MultiMC::Failed;
|
Status m_status = MultiMC::Failed;
|
||||||
MultiMCVersion m_version;
|
MultiMCVersion m_version;
|
||||||
};
|
};
|
||||||
|
@ -10,6 +10,12 @@
|
|||||||
// URL for the updater's channel
|
// URL for the updater's channel
|
||||||
#define CHANLIST_URL "@MultiMC_CHANLIST_URL@"
|
#define CHANLIST_URL "@MultiMC_CHANLIST_URL@"
|
||||||
|
|
||||||
|
// URL for notifications
|
||||||
|
#define NOTIFICATION_URL "@MultiMC_NOTIFICATION_URL@"
|
||||||
|
|
||||||
|
// Used for matching notifications
|
||||||
|
#define FULL_VERSION_STR "@MultiMC_VERSION_MAJOR@.@MultiMC_VERSION_MINOR@.@MultiMC_VERSION_BUILD@"
|
||||||
|
|
||||||
// The commit hash of this build
|
// The commit hash of this build
|
||||||
#define GIT_COMMIT "@MultiMC_GIT_COMMIT@"
|
#define GIT_COMMIT "@MultiMC_GIT_COMMIT@"
|
||||||
|
|
||||||
|
@ -92,6 +92,7 @@
|
|||||||
#include "logic/assets/AssetsUtils.h"
|
#include "logic/assets/AssetsUtils.h"
|
||||||
#include "logic/assets/AssetsMigrateTask.h"
|
#include "logic/assets/AssetsMigrateTask.h"
|
||||||
#include <logic/updater/UpdateChecker.h>
|
#include <logic/updater/UpdateChecker.h>
|
||||||
|
#include <logic/updater/NotificationChecker.h>
|
||||||
#include <logic/tasks/ThreadTask.h>
|
#include <logic/tasks/ThreadTask.h>
|
||||||
|
|
||||||
MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWindow)
|
MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWindow)
|
||||||
@ -283,6 +284,9 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWi
|
|||||||
// if automatic update checks are allowed, start one.
|
// if automatic update checks are allowed, start one.
|
||||||
if (MMC->settings()->get("AutoUpdate").toBool())
|
if (MMC->settings()->get("AutoUpdate").toBool())
|
||||||
on_actionCheckUpdate_triggered();
|
on_actionCheckUpdate_triggered();
|
||||||
|
|
||||||
|
connect(MMC->notificationChecker().get(), &NotificationChecker::notificationCheckFinished,
|
||||||
|
this, &MainWindow::notificationsChanged);
|
||||||
}
|
}
|
||||||
|
|
||||||
const QString currentInstanceId = MMC->settings()->get("SelectedInstance").toString();
|
const QString currentInstanceId = MMC->settings()->get("SelectedInstance").toString();
|
||||||
@ -522,6 +526,63 @@ void MainWindow::updateAvailable(QString repo, QString versionName, int versionI
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QList<int> stringToIntList(const QString &string)
|
||||||
|
{
|
||||||
|
QStringList split = string.split(',', QString::SkipEmptyParts);
|
||||||
|
QList<int> out;
|
||||||
|
for (int i = 0; i < split.size(); ++i)
|
||||||
|
{
|
||||||
|
out.append(split.at(i).toInt());
|
||||||
|
}
|
||||||
|
return out;
|
||||||
|
}
|
||||||
|
QString intListToString(const QList<int> &list)
|
||||||
|
{
|
||||||
|
QStringList slist;
|
||||||
|
for (int i = 0; i < list.size(); ++i)
|
||||||
|
{
|
||||||
|
slist.append(QString::number(list.at(i)));
|
||||||
|
}
|
||||||
|
return slist.join(',');
|
||||||
|
}
|
||||||
|
void MainWindow::notificationsChanged()
|
||||||
|
{
|
||||||
|
QList<NotificationChecker::NotificationEntry> entries =
|
||||||
|
MMC->notificationChecker()->notificationEntries();
|
||||||
|
QList<int> shownNotifications =
|
||||||
|
stringToIntList(MMC->settings()->get("ShownNotifications").toString());
|
||||||
|
for (auto it = entries.begin(); it != entries.end(); ++it)
|
||||||
|
{
|
||||||
|
NotificationChecker::NotificationEntry entry = *it;
|
||||||
|
if (!shownNotifications.contains(entry.id) && entry.applies())
|
||||||
|
{
|
||||||
|
QMessageBox::Icon icon;
|
||||||
|
switch (entry.type)
|
||||||
|
{
|
||||||
|
case NotificationChecker::NotificationEntry::Critical:
|
||||||
|
icon = QMessageBox::Critical;
|
||||||
|
break;
|
||||||
|
case NotificationChecker::NotificationEntry::Warning:
|
||||||
|
icon = QMessageBox::Warning;
|
||||||
|
break;
|
||||||
|
case NotificationChecker::NotificationEntry::Information:
|
||||||
|
icon = QMessageBox::Information;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
QMessageBox box(icon, tr("Notification"), entry.message, QMessageBox::Close, this);
|
||||||
|
QPushButton *dontShowAgainButton = box.addButton(tr("Don't show again"), QMessageBox::AcceptRole);
|
||||||
|
box.setDefaultButton(QMessageBox::Close);
|
||||||
|
box.exec();
|
||||||
|
if (box.clickedButton() == dontShowAgainButton)
|
||||||
|
{
|
||||||
|
shownNotifications.append(entry.id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
MMC->settings()->set("ShownNotifications", intListToString(shownNotifications));
|
||||||
|
}
|
||||||
|
|
||||||
void MainWindow::downloadUpdates(QString repo, int versionId, bool installOnExit)
|
void MainWindow::downloadUpdates(QString repo, int versionId, bool installOnExit)
|
||||||
{
|
{
|
||||||
QLOG_INFO() << "Downloading updates.";
|
QLOG_INFO() << "Downloading updates.";
|
||||||
|
@ -159,6 +159,8 @@ slots:
|
|||||||
|
|
||||||
void updateAvailable(QString repo, QString versionName, int versionId);
|
void updateAvailable(QString repo, QString versionName, int versionId);
|
||||||
|
|
||||||
|
void notificationsChanged();
|
||||||
|
|
||||||
void activeAccountChanged();
|
void activeAccountChanged();
|
||||||
|
|
||||||
void changeActiveAccount();
|
void changeActiveAccount();
|
||||||
|
@ -44,9 +44,19 @@ void ModListView::setModel ( QAbstractItemModel* model )
|
|||||||
QTreeView::setModel ( model );
|
QTreeView::setModel ( model );
|
||||||
auto head = header();
|
auto head = header();
|
||||||
head->setStretchLastSection(false);
|
head->setStretchLastSection(false);
|
||||||
|
// HACK: this is true for the checkbox column of mod lists
|
||||||
|
auto string = model->headerData(0,head->orientation()).toString();
|
||||||
|
if(!string.size())
|
||||||
|
{
|
||||||
head->setSectionResizeMode(0, QHeaderView::ResizeToContents);
|
head->setSectionResizeMode(0, QHeaderView::ResizeToContents);
|
||||||
head->setSectionResizeMode(1, QHeaderView::Stretch);
|
head->setSectionResizeMode(1, QHeaderView::Stretch);
|
||||||
for(int i = 2; i < head->count(); i++)
|
for(int i = 2; i < head->count(); i++)
|
||||||
head->setSectionResizeMode(i, QHeaderView::ResizeToContents);
|
head->setSectionResizeMode(i, QHeaderView::ResizeToContents);
|
||||||
dropIndicatorPosition();
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
head->setSectionResizeMode(0, QHeaderView::Stretch);
|
||||||
|
for(int i = 1; i < head->count(); i++)
|
||||||
|
head->setSectionResizeMode(i, QHeaderView::ResizeToContents);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -416,7 +416,7 @@ QVariant ModList::data(const QModelIndex &index, int role) const
|
|||||||
switch (index.column())
|
switch (index.column())
|
||||||
{
|
{
|
||||||
case ActiveColumn:
|
case ActiveColumn:
|
||||||
return mods[row].enabled();
|
return mods[row].enabled() ? Qt::Checked: Qt::Unchecked;
|
||||||
default:
|
default:
|
||||||
return QVariant();
|
return QVariant();
|
||||||
}
|
}
|
||||||
|
@ -307,9 +307,10 @@ void InstanceList::loadForgeInstances(QMap<QString, QString> groupMap)
|
|||||||
QLOG_INFO() << "The FTB directory specified does not exist. Please check your settings";
|
QLOG_INFO() << "The FTB directory specified does not exist. Please check your settings";
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
dir.cd("ModPacks");
|
dir.cd("ModPacks");
|
||||||
QFile f(dir.absoluteFilePath("modpacks.xml"));
|
auto fpath = dir.absoluteFilePath("modpacks.xml");
|
||||||
|
QFile f(fpath);
|
||||||
|
QLOG_INFO() << "Discovering FTB instances -- " << fpath;
|
||||||
if (!f.open(QFile::ReadOnly))
|
if (!f.open(QFile::ReadOnly))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -326,6 +327,9 @@ void InstanceList::loadForgeInstances(QMap<QString, QString> groupMap)
|
|||||||
QXmlStreamAttributes attrs = reader.attributes();
|
QXmlStreamAttributes attrs = reader.attributes();
|
||||||
FTBRecord record;
|
FTBRecord record;
|
||||||
record.dir = attrs.value("dir").toString();
|
record.dir = attrs.value("dir").toString();
|
||||||
|
QDir test(dataDir.absoluteFilePath(record.dir));
|
||||||
|
if(!test.exists())
|
||||||
|
continue;
|
||||||
record.name = attrs.value("name").toString();
|
record.name = attrs.value("name").toString();
|
||||||
record.logo = attrs.value("logo").toString();
|
record.logo = attrs.value("logo").toString();
|
||||||
record.mcVersion = attrs.value("mcVersion").toString();
|
record.mcVersion = attrs.value("mcVersion").toString();
|
||||||
@ -343,11 +347,17 @@ void InstanceList::loadForgeInstances(QMap<QString, QString> groupMap)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
f.close();
|
f.close();
|
||||||
|
if(!records.size())
|
||||||
|
{
|
||||||
|
QLOG_INFO() << "No FTB instances to load.";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
QLOG_INFO() << "Loading FTB instances! -- got " << records.size();
|
||||||
// process the records we acquired.
|
// process the records we acquired.
|
||||||
for (auto record : records)
|
for (auto record : records)
|
||||||
{
|
{
|
||||||
auto instanceDir = dataDir.absoluteFilePath(record.dir);
|
auto instanceDir = dataDir.absoluteFilePath(record.dir);
|
||||||
|
QLOG_INFO() << "Loading FTB instance from " << instanceDir;
|
||||||
auto templateDir = dir.absoluteFilePath(record.dir);
|
auto templateDir = dir.absoluteFilePath(record.dir);
|
||||||
if (!QFileInfo(instanceDir).exists())
|
if (!QFileInfo(instanceDir).exists())
|
||||||
{
|
{
|
||||||
@ -361,6 +371,7 @@ void InstanceList::loadForgeInstances(QMap<QString, QString> groupMap)
|
|||||||
|
|
||||||
if (!QFileInfo(PathCombine(instanceDir, "instance.cfg")).exists())
|
if (!QFileInfo(PathCombine(instanceDir, "instance.cfg")).exists())
|
||||||
{
|
{
|
||||||
|
QLOG_INFO() << "Converting " << record.name << " as new.";
|
||||||
BaseInstance *instPtr = NULL;
|
BaseInstance *instPtr = NULL;
|
||||||
auto &factory = InstanceFactory::get();
|
auto &factory = InstanceFactory::get();
|
||||||
auto version = MMC->minecraftlist()->findVersion(record.mcVersion);
|
auto version = MMC->minecraftlist()->findVersion(record.mcVersion);
|
||||||
@ -386,6 +397,7 @@ void InstanceList::loadForgeInstances(QMap<QString, QString> groupMap)
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
QLOG_INFO() << "Loading existing " << record.name;
|
||||||
BaseInstance *instPtr = NULL;
|
BaseInstance *instPtr = NULL;
|
||||||
auto error = InstanceFactory::get().loadInstance(instPtr, instanceDir);
|
auto error = InstanceFactory::get().loadInstance(instPtr, instanceDir);
|
||||||
if (!instPtr || error != InstanceFactory::NoCreateError)
|
if (!instPtr || error != InstanceFactory::NoCreateError)
|
||||||
@ -419,7 +431,7 @@ InstanceList::InstListError InstanceList::loadList()
|
|||||||
QString subDir = iter.next();
|
QString subDir = iter.next();
|
||||||
if (!QFileInfo(PathCombine(subDir, "instance.cfg")).exists())
|
if (!QFileInfo(PathCombine(subDir, "instance.cfg")).exists())
|
||||||
continue;
|
continue;
|
||||||
|
QLOG_INFO() << "Loading MultiMC instance from " << subDir;
|
||||||
BaseInstance *instPtr = NULL;
|
BaseInstance *instPtr = NULL;
|
||||||
auto error = InstanceFactory::get().loadInstance(instPtr, subDir);
|
auto error = InstanceFactory::get().loadInstance(instPtr, subDir);
|
||||||
continueProcessInstance(instPtr, error, subDir, groupMap);
|
continueProcessInstance(instPtr, error, subDir, groupMap);
|
||||||
@ -534,7 +546,7 @@ void InstanceList::continueProcessInstance(BaseInstance *instPtr, const int erro
|
|||||||
{
|
{
|
||||||
instPtr->setGroupInitial((*iter));
|
instPtr->setGroupInitial((*iter));
|
||||||
}
|
}
|
||||||
QLOG_INFO() << "Loaded instance " << instPtr->name();
|
QLOG_INFO() << "Loaded instance " << instPtr->name() << " from " << dir.absolutePath();
|
||||||
instPtr->setParent(this);
|
instPtr->setParent(this);
|
||||||
m_instances.append(std::shared_ptr<BaseInstance>(instPtr));
|
m_instances.append(std::shared_ptr<BaseInstance>(instPtr));
|
||||||
connect(instPtr, SIGNAL(propertiesChanged(BaseInstance *)), this,
|
connect(instPtr, SIGNAL(propertiesChanged(BaseInstance *)), this,
|
||||||
|
@ -404,12 +404,11 @@ DownloadUpdateTask::processFileLists(NetJob *job,
|
|||||||
{
|
{
|
||||||
auto cache_entry = MMC->metacache()->resolveEntry("root", entry.path);
|
auto cache_entry = MMC->metacache()->resolveEntry("root", entry.path);
|
||||||
QLOG_DEBUG() << "Updater will be in " << cache_entry->getFullPath();
|
QLOG_DEBUG() << "Updater will be in " << cache_entry->getFullPath();
|
||||||
if(cache_entry->stale)
|
// force check.
|
||||||
{
|
cache_entry->stale = true;
|
||||||
auto download = CacheDownload::make(QUrl(source.url), cache_entry);
|
auto download = CacheDownload::make(QUrl(source.url), cache_entry);
|
||||||
job->addNetAction(download);
|
job->addNetAction(download);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// We need to download the file to the updatefiles folder and add a task
|
// We need to download the file to the updatefiles folder and add a task
|
||||||
|
120
logic/updater/NotificationChecker.cpp
Normal file
120
logic/updater/NotificationChecker.cpp
Normal file
@ -0,0 +1,120 @@
|
|||||||
|
#include "NotificationChecker.h"
|
||||||
|
|
||||||
|
#include <QJsonDocument>
|
||||||
|
#include <QJsonObject>
|
||||||
|
#include <QJsonArray>
|
||||||
|
|
||||||
|
#include "MultiMC.h"
|
||||||
|
#include "logic/net/CacheDownload.h"
|
||||||
|
#include "config.h"
|
||||||
|
|
||||||
|
NotificationChecker::NotificationChecker(QObject *parent)
|
||||||
|
: QObject(parent), m_notificationsUrl(QUrl(NOTIFICATION_URL))
|
||||||
|
{
|
||||||
|
// this will call checkForNotifications once the event loop is running
|
||||||
|
QMetaObject::invokeMethod(this, "checkForNotifications", Qt::QueuedConnection);
|
||||||
|
}
|
||||||
|
|
||||||
|
QUrl NotificationChecker::notificationsUrl() const
|
||||||
|
{
|
||||||
|
return m_notificationsUrl;
|
||||||
|
}
|
||||||
|
void NotificationChecker::setNotificationsUrl(const QUrl ¬ificationsUrl)
|
||||||
|
{
|
||||||
|
m_notificationsUrl = notificationsUrl;
|
||||||
|
}
|
||||||
|
|
||||||
|
QList<NotificationChecker::NotificationEntry> NotificationChecker::notificationEntries() const
|
||||||
|
{
|
||||||
|
return m_entries;
|
||||||
|
}
|
||||||
|
|
||||||
|
void NotificationChecker::checkForNotifications()
|
||||||
|
{
|
||||||
|
if (!m_notificationsUrl.isValid())
|
||||||
|
{
|
||||||
|
QLOG_ERROR() << "Failed to check for notifications. No notifications URL set."
|
||||||
|
<< "If you'd like to use MultiMC's notification system, please pass the "
|
||||||
|
"URL to CMake at compile time.";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (m_checkJob)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
m_checkJob.reset(new NetJob("Checking for notifications"));
|
||||||
|
auto entry = MMC->metacache()->resolveEntry("root", "notifications.json");
|
||||||
|
entry->stale = true;
|
||||||
|
m_checkJob->addNetAction(m_download = CacheDownload::make(m_notificationsUrl, entry));
|
||||||
|
connect(m_download.get(), &CacheDownload::succeeded, this,
|
||||||
|
&NotificationChecker::downloadSucceeded);
|
||||||
|
m_checkJob->start();
|
||||||
|
}
|
||||||
|
|
||||||
|
void NotificationChecker::downloadSucceeded(int)
|
||||||
|
{
|
||||||
|
m_entries.clear();
|
||||||
|
|
||||||
|
QFile file(m_download->m_output_file.fileName());
|
||||||
|
if (file.open(QFile::ReadOnly))
|
||||||
|
{
|
||||||
|
QJsonArray root = QJsonDocument::fromJson(file.readAll()).array();
|
||||||
|
for (auto it = root.begin(); it != root.end(); ++it)
|
||||||
|
{
|
||||||
|
QJsonObject obj = (*it).toObject();
|
||||||
|
NotificationEntry entry;
|
||||||
|
entry.id = obj.value("id").toDouble();
|
||||||
|
entry.message = obj.value("message").toString();
|
||||||
|
entry.channel = obj.value("channel").toString();
|
||||||
|
entry.buildtype = obj.value("buildtype").toString();
|
||||||
|
entry.from = obj.value("from").toString();
|
||||||
|
entry.to = obj.value("to").toString();
|
||||||
|
const QString type = obj.value("type").toString("critical");
|
||||||
|
if (type == "critical")
|
||||||
|
{
|
||||||
|
entry.type = NotificationEntry::Critical;
|
||||||
|
}
|
||||||
|
else if (type == "warning")
|
||||||
|
{
|
||||||
|
entry.type = NotificationEntry::Warning;
|
||||||
|
}
|
||||||
|
else if (type == "information")
|
||||||
|
{
|
||||||
|
entry.type = NotificationEntry::Information;
|
||||||
|
}
|
||||||
|
m_entries.append(entry);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
m_checkJob.reset();
|
||||||
|
|
||||||
|
emit notificationCheckFinished();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool NotificationChecker::NotificationEntry::applies() const
|
||||||
|
{
|
||||||
|
bool channelApplies = channel.isEmpty() || channel == VERSION_CHANNEL;
|
||||||
|
bool buildtypeApplies = buildtype.isEmpty() || buildtype == VERSION_BUILD_TYPE;
|
||||||
|
bool fromApplies =
|
||||||
|
from.isEmpty() || from == FULL_VERSION_STR || !versionLessThan(FULL_VERSION_STR, from);
|
||||||
|
bool toApplies =
|
||||||
|
to.isEmpty() || to == FULL_VERSION_STR || !versionLessThan(to, FULL_VERSION_STR);
|
||||||
|
return channelApplies && buildtypeApplies && fromApplies && toApplies;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool NotificationChecker::NotificationEntry::versionLessThan(const QString &v1,
|
||||||
|
const QString &v2)
|
||||||
|
{
|
||||||
|
QStringList l1 = v1.split('.');
|
||||||
|
QStringList l2 = v2.split('.');
|
||||||
|
while (!l1.isEmpty() && !l2.isEmpty())
|
||||||
|
{
|
||||||
|
int one = l1.isEmpty() ? 0 : l1.takeFirst().toInt();
|
||||||
|
int two = l2.isEmpty() ? 0 : l2.takeFirst().toInt();
|
||||||
|
if (one != two)
|
||||||
|
{
|
||||||
|
return one < two;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
54
logic/updater/NotificationChecker.h
Normal file
54
logic/updater/NotificationChecker.h
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
|
||||||
|
#include "logic/net/NetJob.h"
|
||||||
|
#include "logic/net/CacheDownload.h"
|
||||||
|
|
||||||
|
class NotificationChecker : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit NotificationChecker(QObject *parent = 0);
|
||||||
|
|
||||||
|
QUrl notificationsUrl() const;
|
||||||
|
void setNotificationsUrl(const QUrl ¬ificationsUrl);
|
||||||
|
|
||||||
|
struct NotificationEntry
|
||||||
|
{
|
||||||
|
int id;
|
||||||
|
QString message;
|
||||||
|
enum
|
||||||
|
{
|
||||||
|
Critical,
|
||||||
|
Warning,
|
||||||
|
Information
|
||||||
|
} type;
|
||||||
|
QString channel;
|
||||||
|
QString buildtype;
|
||||||
|
QString from;
|
||||||
|
QString to;
|
||||||
|
bool applies() const;
|
||||||
|
static bool versionLessThan(const QString &v1, const QString &v2);
|
||||||
|
};
|
||||||
|
|
||||||
|
QList<NotificationEntry> notificationEntries() const;
|
||||||
|
|
||||||
|
public
|
||||||
|
slots:
|
||||||
|
void checkForNotifications();
|
||||||
|
|
||||||
|
private
|
||||||
|
slots:
|
||||||
|
void downloadSucceeded(int);
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void notificationCheckFinished();
|
||||||
|
|
||||||
|
private:
|
||||||
|
QList<NotificationEntry> m_entries;
|
||||||
|
QUrl m_notificationsUrl;
|
||||||
|
NetJobPtr m_checkJob;
|
||||||
|
CacheDownloadPtr m_download;
|
||||||
|
};
|
@ -39,7 +39,7 @@ int main(int argc, char *argv[]) \
|
|||||||
{ \
|
{ \
|
||||||
char *argv_[] = { argv[0] _MMC_EXTRA_ARGV }; \
|
char *argv_[] = { argv[0] _MMC_EXTRA_ARGV }; \
|
||||||
int argc_ = 1 + _MMC_EXTRA_ARGC; \
|
int argc_ = 1 + _MMC_EXTRA_ARGC; \
|
||||||
MultiMC app(argc_, argv_, QDir::temp().absoluteFilePath("MultiMC_Test")); \
|
MultiMC app(argc_, argv_/*, QDir::temp().absoluteFilePath("MultiMC_Test")*/); \
|
||||||
app.setAttribute(Qt::AA_Use96Dpi, true); \
|
app.setAttribute(Qt::AA_Use96Dpi, true); \
|
||||||
TestObject tc; \
|
TestObject tc; \
|
||||||
return QTest::qExec(&tc, argc, argv); \
|
return QTest::qExec(&tc, argc, argv); \
|
||||||
|
Loading…
x
Reference in New Issue
Block a user