Merge branch 'develop' of github.com:MultiMC/MultiMC5 into feature_news
Conflicts: CMakeLists.txt
This commit is contained in:
commit
9d8006b597
1
.gitignore
vendored
1
.gitignore
vendored
@ -10,6 +10,7 @@ resources/CMakeFiles
|
|||||||
resources/MultiMCLauncher.jar
|
resources/MultiMCLauncher.jar
|
||||||
*~
|
*~
|
||||||
*.swp
|
*.swp
|
||||||
|
html/
|
||||||
|
|
||||||
# Ctags File
|
# Ctags File
|
||||||
tags
|
tags
|
||||||
|
@ -6,6 +6,7 @@ IF(WIN32)
|
|||||||
ENDIF()
|
ENDIF()
|
||||||
|
|
||||||
project(MultiMC)
|
project(MultiMC)
|
||||||
|
enable_testing()
|
||||||
|
|
||||||
######## Set CMake options ########
|
######## Set CMake options ########
|
||||||
SET(CMAKE_AUTOMOC ON)
|
SET(CMAKE_AUTOMOC ON)
|
||||||
@ -304,8 +305,6 @@ logic/net/NetJob.h
|
|||||||
logic/net/NetJob.cpp
|
logic/net/NetJob.cpp
|
||||||
logic/net/HttpMetaCache.h
|
logic/net/HttpMetaCache.h
|
||||||
logic/net/HttpMetaCache.cpp
|
logic/net/HttpMetaCache.cpp
|
||||||
logic/net/S3ListBucket.h
|
|
||||||
logic/net/S3ListBucket.cpp
|
|
||||||
logic/net/PasteUpload.h
|
logic/net/PasteUpload.h
|
||||||
logic/net/PasteUpload.cpp
|
logic/net/PasteUpload.cpp
|
||||||
logic/net/URLConstants.h
|
logic/net/URLConstants.h
|
||||||
@ -472,6 +471,15 @@ IF(WIN32)
|
|||||||
)
|
)
|
||||||
ENDIF(WIN32)
|
ENDIF(WIN32)
|
||||||
|
|
||||||
|
OPTION(MultiMC_CODE_COVERAGE "Compiles for code coverage" OFF)
|
||||||
|
IF(MultiMC_CODE_COVERAGE)
|
||||||
|
SET(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -g -O0 --coverage")
|
||||||
|
SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -g -O0 --coverage")
|
||||||
|
SET(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -g -O0 --coverage")
|
||||||
|
SET(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -g -O0 --coverage")
|
||||||
|
SET(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -O0 --coverage")
|
||||||
|
ENDIF(MultiMC_CODE_COVERAGE)
|
||||||
|
|
||||||
# Tell CMake that MultiMCLauncher.jar is generated.
|
# Tell CMake that MultiMCLauncher.jar is generated.
|
||||||
SET_SOURCE_FILES_PROPERTIES(${PROJECT_BINARY_DIR}/depends/launcher/MultiMCLauncher.jar GENERATED)
|
SET_SOURCE_FILES_PROPERTIES(${PROJECT_BINARY_DIR}/depends/launcher/MultiMCLauncher.jar GENERATED)
|
||||||
SET_SOURCE_FILES_PROPERTIES(${PROJECT_BINARY_DIR}/depends/javacheck/JavaCheck.jar GENERATED)
|
SET_SOURCE_FILES_PROPERTIES(${PROJECT_BINARY_DIR}/depends/javacheck/JavaCheck.jar GENERATED)
|
||||||
@ -482,14 +490,18 @@ CONFIGURE_FILE(generated.qrc.in generated.qrc)
|
|||||||
QT5_ADD_RESOURCES(GENERATED_QRC ${CMAKE_CURRENT_BINARY_DIR}/generated.qrc)
|
QT5_ADD_RESOURCES(GENERATED_QRC ${CMAKE_CURRENT_BINARY_DIR}/generated.qrc)
|
||||||
QT5_ADD_RESOURCES(GRAPHICS_QRC graphics.qrc)
|
QT5_ADD_RESOURCES(GRAPHICS_QRC graphics.qrc)
|
||||||
|
|
||||||
|
# Add common library
|
||||||
|
ADD_LIBRARY(MultiMC_common STATIC ${MULTIMC_SOURCES} ${MULTIMC_UI} ${GENERATED_QRC} ${GRAPHICS_QRC})
|
||||||
|
|
||||||
# Add executable
|
# Add executable
|
||||||
ADD_EXECUTABLE(MultiMC MACOSX_BUNDLE WIN32
|
ADD_EXECUTABLE(MultiMC MACOSX_BUNDLE WIN32 main.cpp ${MULTIMC_RCS})
|
||||||
${MULTIMC_SOURCES} ${MULTIMC_UI} ${GRAPHICS_QRC} ${GENERATED_QRC} ${MULTIMC_RCS})
|
|
||||||
|
|
||||||
# Link
|
# Link
|
||||||
TARGET_LINK_LIBRARIES(MultiMC xz-embedded unpack200 quazip libUtil libSettings libGroupView ${MultiMC_LINK_ADDITIONAL_LIBS})
|
TARGET_LINK_LIBRARIES(MultiMC MultiMC_common)
|
||||||
|
TARGET_LINK_LIBRARIES(MultiMC_common xz-embedded unpack200 quazip libUtil libSettings libGroupView ${MultiMC_LINK_ADDITIONAL_LIBS})
|
||||||
QT5_USE_MODULES(MultiMC Core Widgets Network Xml WebKit ${MultiMC_QT_ADDITIONAL_MODULES})
|
QT5_USE_MODULES(MultiMC Core Widgets Network Xml WebKit ${MultiMC_QT_ADDITIONAL_MODULES})
|
||||||
ADD_DEPENDENCIES(MultiMC MultiMCLauncher JavaCheck)
|
QT5_USE_MODULES(MultiMC_common Core Widgets Network Xml WebKit ${MultiMC_QT_ADDITIONAL_MODULES})
|
||||||
|
ADD_DEPENDENCIES(MultiMC_common MultiMCLauncher JavaCheck)
|
||||||
|
|
||||||
################################ INSTALLATION AND PACKAGING ################################
|
################################ INSTALLATION AND PACKAGING ################################
|
||||||
|
|
||||||
@ -648,3 +660,6 @@ ENDIF()
|
|||||||
add_custom_target (translations DEPENDS ${QM_FILES})
|
add_custom_target (translations DEPENDS ${QM_FILES})
|
||||||
|
|
||||||
install(FILES ${QM_FILES} DESTINATION ${CMAKE_INSTALL_PREFIX}/translations)
|
install(FILES ${QM_FILES} DESTINATION ${CMAKE_INSTALL_PREFIX}/translations)
|
||||||
|
|
||||||
|
# Tests
|
||||||
|
add_subdirectory(tests)
|
||||||
|
39
MultiMC.cpp
39
MultiMC.cpp
@ -9,7 +9,6 @@
|
|||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
#include <QStringList>
|
#include <QStringList>
|
||||||
|
|
||||||
#include "gui/MainWindow.h"
|
|
||||||
#include "gui/dialogs/VersionSelectDialog.h"
|
#include "gui/dialogs/VersionSelectDialog.h"
|
||||||
#include "logic/lists/InstanceList.h"
|
#include "logic/lists/InstanceList.h"
|
||||||
#include "logic/auth/MojangAccountList.h"
|
#include "logic/auth/MojangAccountList.h"
|
||||||
@ -37,7 +36,7 @@
|
|||||||
#include "config.h"
|
#include "config.h"
|
||||||
using namespace Util::Commandline;
|
using namespace Util::Commandline;
|
||||||
|
|
||||||
MultiMC::MultiMC(int &argc, char **argv) : QApplication(argc, argv),
|
MultiMC::MultiMC(int &argc, char **argv, const QString &root) : QApplication(argc, argv),
|
||||||
m_version{VERSION_MAJOR, VERSION_MINOR, VERSION_BUILD, VERSION_CHANNEL, VERSION_BUILD_TYPE}
|
m_version{VERSION_MAJOR, VERSION_MINOR, VERSION_BUILD, VERSION_CHANNEL, VERSION_BUILD_TYPE}
|
||||||
{
|
{
|
||||||
setOrganizationName("MultiMC");
|
setOrganizationName("MultiMC");
|
||||||
@ -138,7 +137,9 @@ MultiMC::MultiMC(int &argc, char **argv) : QApplication(argc, argv),
|
|||||||
}
|
}
|
||||||
|
|
||||||
// change directory
|
// change directory
|
||||||
QDir::setCurrent(args["dir"].toString());
|
QDir::setCurrent(args["dir"].toString().isEmpty() ?
|
||||||
|
(root.isEmpty() ? QDir::currentPath() : QDir::current().absoluteFilePath(root))
|
||||||
|
: args["dir"].toString());
|
||||||
|
|
||||||
// init the logger
|
// init the logger
|
||||||
initLogger();
|
initLogger();
|
||||||
@ -480,37 +481,5 @@ QString MultiMC::getExitUpdatePath() const
|
|||||||
return m_updateOnExitPath;
|
return m_updateOnExitPath;
|
||||||
}
|
}
|
||||||
|
|
||||||
int main_gui(MultiMC &app)
|
|
||||||
{
|
|
||||||
// show main window
|
|
||||||
MainWindow mainWin;
|
|
||||||
mainWin.restoreState(QByteArray::fromBase64(MMC->settings()->get("MainWindowState").toByteArray()));
|
|
||||||
mainWin.restoreGeometry(QByteArray::fromBase64(MMC->settings()->get("MainWindowGeometry").toByteArray()));
|
|
||||||
mainWin.show();
|
|
||||||
mainWin.checkSetDefaultJava();
|
|
||||||
auto exitCode = app.exec();
|
|
||||||
|
|
||||||
// Update if necessary.
|
|
||||||
if (!app.getExitUpdatePath().isEmpty())
|
|
||||||
app.installUpdates(app.getExitUpdatePath(), false);
|
|
||||||
|
|
||||||
return exitCode;
|
|
||||||
}
|
|
||||||
|
|
||||||
int main(int argc, char *argv[])
|
|
||||||
{
|
|
||||||
// initialize Qt
|
|
||||||
MultiMC app(argc, argv);
|
|
||||||
|
|
||||||
switch (app.status())
|
|
||||||
{
|
|
||||||
case MultiMC::Initialized:
|
|
||||||
return main_gui(app);
|
|
||||||
case MultiMC::Failed:
|
|
||||||
return 1;
|
|
||||||
case MultiMC::Succeeded:
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#include "MultiMC.moc"
|
#include "MultiMC.moc"
|
||||||
|
@ -46,7 +46,7 @@ public:
|
|||||||
};
|
};
|
||||||
|
|
||||||
public:
|
public:
|
||||||
MultiMC(int &argc, char **argv);
|
MultiMC(int &argc, char **argv, const QString &root = QString());
|
||||||
virtual ~MultiMC();
|
virtual ~MultiMC();
|
||||||
|
|
||||||
std::shared_ptr<SettingsObject> settings()
|
std::shared_ptr<SettingsObject> settings()
|
||||||
|
@ -51,6 +51,14 @@ add_definitions(-DLIBSETTINGS_LIBRARY)
|
|||||||
|
|
||||||
set(CMAKE_POSITION_INDEPENDENT_CODE ON)
|
set(CMAKE_POSITION_INDEPENDENT_CODE ON)
|
||||||
|
|
||||||
|
IF(MultiMC_CODE_COVERAGE)
|
||||||
|
SET(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -g -O0 --coverage")
|
||||||
|
SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -g -O0 --coverage")
|
||||||
|
SET(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -g -O0 --coverage")
|
||||||
|
SET(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -g -O0 --coverage")
|
||||||
|
SET(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -O0 --coverage")
|
||||||
|
ENDIF(MultiMC_CODE_COVERAGE)
|
||||||
|
|
||||||
add_library(libSettings STATIC ${LIBSETTINGS_SOURCES} ${LIBSETTINGS_HEADERS} ${LIBSETTINGS_HEADERS_PRIVATE})
|
add_library(libSettings STATIC ${LIBSETTINGS_SOURCES} ${LIBSETTINGS_HEADERS} ${LIBSETTINGS_HEADERS_PRIVATE})
|
||||||
qt5_use_modules(libSettings Core)
|
qt5_use_modules(libSettings Core)
|
||||||
target_link_libraries(libSettings)
|
target_link_libraries(libSettings)
|
||||||
|
@ -47,6 +47,14 @@ add_definitions(-DLIBUTIL_LIBRARY)
|
|||||||
|
|
||||||
set(CMAKE_POSITION_INDEPENDENT_CODE ON)
|
set(CMAKE_POSITION_INDEPENDENT_CODE ON)
|
||||||
|
|
||||||
|
IF(MultiMC_CODE_COVERAGE)
|
||||||
|
SET(CMAKE_C_FLAGS "${CMAKE_C_FLAGS} -g -O0 --coverage")
|
||||||
|
SET(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -g -O0 --coverage")
|
||||||
|
SET(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -g -O0 --coverage")
|
||||||
|
SET(CMAKE_CXX_FLAGS_RELEASE "${CMAKE_CXX_FLAGS_RELEASE} -g -O0 --coverage")
|
||||||
|
SET(CMAKE_EXE_LINKER_FLAGS "${CMAKE_EXE_LINKER_FLAGS} -O0 --coverage")
|
||||||
|
ENDIF(MultiMC_CODE_COVERAGE)
|
||||||
|
|
||||||
add_library(libUtil STATIC ${LIBUTIL_SOURCES})
|
add_library(libUtil STATIC ${LIBUTIL_SOURCES})
|
||||||
# qt5_use_modules(libUtil Core Network)
|
# qt5_use_modules(libUtil Core Network)
|
||||||
qt5_use_modules(libUtil Core)
|
qt5_use_modules(libUtil Core)
|
||||||
|
@ -20,7 +20,6 @@
|
|||||||
|
|
||||||
#include "MainWindow.h"
|
#include "MainWindow.h"
|
||||||
#include "ui_MainWindow.h"
|
#include "ui_MainWindow.h"
|
||||||
#include "keyring.h"
|
|
||||||
|
|
||||||
#include <QMenu>
|
#include <QMenu>
|
||||||
#include <QMessageBox>
|
#include <QMessageBox>
|
||||||
@ -186,7 +185,8 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWi
|
|||||||
accountMenu = new QMenu(this);
|
accountMenu = new QMenu(this);
|
||||||
manageAccountsAction = new QAction(tr("Manage Accounts"), this);
|
manageAccountsAction = new QAction(tr("Manage Accounts"), this);
|
||||||
manageAccountsAction->setCheckable(false);
|
manageAccountsAction->setCheckable(false);
|
||||||
connect(manageAccountsAction, SIGNAL(triggered(bool)), this, SLOT(on_actionManageAccounts_triggered()));
|
connect(manageAccountsAction, SIGNAL(triggered(bool)), this,
|
||||||
|
SLOT(on_actionManageAccounts_triggered()));
|
||||||
|
|
||||||
repopulateAccountsMenu();
|
repopulateAccountsMenu();
|
||||||
|
|
||||||
@ -195,7 +195,8 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWi
|
|||||||
accountMenuButton->setMenu(accountMenu);
|
accountMenuButton->setMenu(accountMenu);
|
||||||
accountMenuButton->setPopupMode(QToolButton::InstantPopup);
|
accountMenuButton->setPopupMode(QToolButton::InstantPopup);
|
||||||
accountMenuButton->setToolButtonStyle(Qt::ToolButtonTextBesideIcon);
|
accountMenuButton->setToolButtonStyle(Qt::ToolButtonTextBesideIcon);
|
||||||
accountMenuButton->setIcon(QPixmap(":/icons/toolbar/noaccount").scaled(48, 48, Qt::KeepAspectRatio));
|
accountMenuButton->setIcon(
|
||||||
|
QPixmap(":/icons/toolbar/noaccount").scaled(48, 48, Qt::KeepAspectRatio));
|
||||||
|
|
||||||
QWidgetAction *accountMenuButtonAction = new QWidgetAction(this);
|
QWidgetAction *accountMenuButtonAction = new QWidgetAction(this);
|
||||||
accountMenuButtonAction->setDefaultWidget(accountMenuButton);
|
accountMenuButtonAction->setDefaultWidget(accountMenuButton);
|
||||||
@ -203,9 +204,12 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWi
|
|||||||
ui->mainToolBar->addAction(accountMenuButtonAction);
|
ui->mainToolBar->addAction(accountMenuButtonAction);
|
||||||
|
|
||||||
// Update the menu when the active account changes.
|
// Update the menu when the active account changes.
|
||||||
// Shouldn't have to use lambdas here like this, but if I don't, the compiler throws a fit. Template hell sucks...
|
// Shouldn't have to use lambdas here like this, but if I don't, the compiler throws a fit.
|
||||||
connect(MMC->accounts().get(), &MojangAccountList::activeAccountChanged, [this] { activeAccountChanged(); });
|
// Template hell sucks...
|
||||||
connect(MMC->accounts().get(), &MojangAccountList::listChanged, [this] { repopulateAccountsMenu(); });
|
connect(MMC->accounts().get(), &MojangAccountList::activeAccountChanged, [this]
|
||||||
|
{ activeAccountChanged(); });
|
||||||
|
connect(MMC->accounts().get(), &MojangAccountList::listChanged, [this]
|
||||||
|
{ repopulateAccountsMenu(); });
|
||||||
|
|
||||||
std::shared_ptr<MojangAccountList> accounts = MMC->accounts();
|
std::shared_ptr<MojangAccountList> accounts = MMC->accounts();
|
||||||
|
|
||||||
@ -221,8 +225,7 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWi
|
|||||||
{
|
{
|
||||||
auto meta = MMC->metacache()->resolveEntry("skins", profile.name + ".png");
|
auto meta = MMC->metacache()->resolveEntry("skins", profile.name + ".png");
|
||||||
auto action = CacheDownload::make(
|
auto action = CacheDownload::make(
|
||||||
QUrl("http://" + URLConstants::SKINS_BASE + profile.name + ".png"),
|
QUrl("http://" + URLConstants::SKINS_BASE + profile.name + ".png"), meta);
|
||||||
meta);
|
|
||||||
job->addNetAction(action);
|
job->addNetAction(action);
|
||||||
meta->stale = true;
|
meta->stale = true;
|
||||||
}
|
}
|
||||||
@ -249,7 +252,8 @@ MainWindow::MainWindow(QWidget *parent) : QMainWindow(parent), ui(new Ui::MainWi
|
|||||||
|
|
||||||
// set up the updater object.
|
// set up the updater object.
|
||||||
auto updater = MMC->updateChecker();
|
auto updater = MMC->updateChecker();
|
||||||
QObject::connect(updater.get(), &UpdateChecker::updateAvailable, this, &MainWindow::updateAvailable);
|
QObject::connect(updater.get(), &UpdateChecker::updateAvailable, this,
|
||||||
|
&MainWindow::updateAvailable);
|
||||||
// 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();
|
||||||
@ -363,7 +367,8 @@ void MainWindow::changeActiveAccount()
|
|||||||
QAction *sAction = (QAction *)sender();
|
QAction *sAction = (QAction *)sender();
|
||||||
// Profile's associated Mojang username
|
// Profile's associated Mojang username
|
||||||
// Will need to change when profiles are properly implemented
|
// Will need to change when profiles are properly implemented
|
||||||
if (sAction->data().type() != QVariant::Type::String) return;
|
if (sAction->data().type() != QVariant::Type::String)
|
||||||
|
return;
|
||||||
|
|
||||||
QVariant data = sAction->data();
|
QVariant data = sAction->data();
|
||||||
QString id = "";
|
QString id = "";
|
||||||
@ -394,7 +399,8 @@ void MainWindow::activeAccountChanged()
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Set the icon to the "no account" icon.
|
// Set the icon to the "no account" icon.
|
||||||
accountMenuButton->setIcon(QPixmap(":/icons/toolbar/noaccount").scaled(48, 48, Qt::KeepAspectRatio));
|
accountMenuButton->setIcon(
|
||||||
|
QPixmap(":/icons/toolbar/noaccount").scaled(48, 48, Qt::KeepAspectRatio));
|
||||||
}
|
}
|
||||||
|
|
||||||
bool MainWindow::eventFilter(QObject *obj, QEvent *ev)
|
bool MainWindow::eventFilter(QObject *obj, QEvent *ev)
|
||||||
@ -448,8 +454,10 @@ void MainWindow::updateAvailable(QString repo, QString versionName, int versionI
|
|||||||
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.";
|
||||||
// TODO: If the user chooses to update on exit, we should download updates in the background.
|
// TODO: If the user chooses to update on exit, we should download updates in the
|
||||||
// Doing so is a bit complicated, because we'd have to make sure it finished downloading before actually exiting MultiMC.
|
// background.
|
||||||
|
// Doing so is a bit complicated, because we'd have to make sure it finished downloading
|
||||||
|
// before actually exiting MultiMC.
|
||||||
ProgressDialog updateDlg(this);
|
ProgressDialog updateDlg(this);
|
||||||
DownloadUpdateTask updateTask(repo, versionId, &updateDlg);
|
DownloadUpdateTask updateTask(repo, versionId, &updateDlg);
|
||||||
// If the task succeeds, install the updates.
|
// If the task succeeds, install the updates.
|
||||||
@ -525,36 +533,44 @@ void MainWindow::on_actionAddInstance_triggered()
|
|||||||
{
|
{
|
||||||
errorMsg += "An instance with the given directory name already exists.";
|
errorMsg += "An instance with the given directory name already exists.";
|
||||||
CustomMessageBox::selectable(this, tr("Error"), errorMsg, QMessageBox::Warning)->show();
|
CustomMessageBox::selectable(this, tr("Error"), errorMsg, QMessageBox::Warning)->show();
|
||||||
break;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
case InstanceFactory::CantCreateDir:
|
case InstanceFactory::CantCreateDir:
|
||||||
{
|
{
|
||||||
errorMsg += "Failed to create the instance directory.";
|
errorMsg += "Failed to create the instance directory.";
|
||||||
CustomMessageBox::selectable(this, tr("Error"), errorMsg, QMessageBox::Warning)->show();
|
CustomMessageBox::selectable(this, tr("Error"), errorMsg, QMessageBox::Warning)->show();
|
||||||
break;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
{
|
{
|
||||||
errorMsg += QString("Unknown instance loader error %1").arg(error);
|
errorMsg += QString("Unknown instance loader error %1").arg(error);
|
||||||
CustomMessageBox::selectable(this, tr("Error"), errorMsg, QMessageBox::Warning)->show();
|
CustomMessageBox::selectable(this, tr("Error"), errorMsg, QMessageBox::Warning)->show();
|
||||||
break;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
std::shared_ptr<MojangAccountList> accounts = MMC->accounts();
|
if (MMC->accounts()->anyAccountIsValid())
|
||||||
MojangAccountPtr account = accounts->activeAccount();
|
|
||||||
if(account.get() != nullptr && account->accountStatus() != NotVerified)
|
|
||||||
{
|
{
|
||||||
ProgressDialog loadDialog(this);
|
ProgressDialog loadDialog(this);
|
||||||
auto update = newInstance->doUpdate(false);
|
auto update = newInstance->doUpdate(false);
|
||||||
connect(update.get(), &Task::failed , [this](QString reason) {
|
connect(update.get(), &Task::failed, [this](QString reason)
|
||||||
|
{
|
||||||
QString error = QString("Instance load failed: %1").arg(reason);
|
QString error = QString("Instance load failed: %1").arg(reason);
|
||||||
CustomMessageBox::selectable(this, tr("Error"), error, QMessageBox::Warning)->show();
|
CustomMessageBox::selectable(this, tr("Error"), error, QMessageBox::Warning)
|
||||||
|
->show();
|
||||||
});
|
});
|
||||||
loadDialog.exec(update.get());
|
loadDialog.exec(update.get());
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
CustomMessageBox::selectable(
|
||||||
|
this, tr("Error"),
|
||||||
|
tr("MultiMC cannot download Minecraft or update instances unless you have at least "
|
||||||
|
"one account added.\nPlease add your Mojang or Minecraft account."),
|
||||||
|
QMessageBox::Warning)->show();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::on_actionCopyInstance_triggered()
|
void MainWindow::on_actionCopyInstance_triggered()
|
||||||
@ -629,8 +645,14 @@ void MainWindow::on_actionChangeInstGroup_triggered()
|
|||||||
|
|
||||||
bool ok = false;
|
bool ok = false;
|
||||||
QString name(m_selectedInstance->group());
|
QString name(m_selectedInstance->group());
|
||||||
name = QInputDialog::getText(this, tr("Group name"), tr("Enter a new group name."),
|
auto groups = MMC->instances()->getGroups();
|
||||||
QLineEdit::Normal, name, &ok);
|
groups.insert(0, "");
|
||||||
|
groups.sort(Qt::CaseInsensitive);
|
||||||
|
int foo = groups.indexOf(name);
|
||||||
|
|
||||||
|
name = QInputDialog::getItem(this, tr("Group name"), tr("Enter a new group name."), groups,
|
||||||
|
foo, true, &ok);
|
||||||
|
name = name.simplified();
|
||||||
if (ok)
|
if (ok)
|
||||||
m_selectedInstance->setGroupPost(name);
|
m_selectedInstance->setGroupPost(name);
|
||||||
}
|
}
|
||||||
@ -814,8 +836,10 @@ void MainWindow::doLaunch()
|
|||||||
if (accounts->count() <= 0)
|
if (accounts->count() <= 0)
|
||||||
{
|
{
|
||||||
// Tell the user they need to log in at least one account in order to play.
|
// Tell the user they need to log in at least one account in order to play.
|
||||||
auto reply = CustomMessageBox::selectable(this, tr("No Accounts"),
|
auto reply = CustomMessageBox::selectable(
|
||||||
tr("In order to play Minecraft, you must have at least one Mojang or Minecraft account logged in to MultiMC."
|
this, tr("No Accounts"),
|
||||||
|
tr("In order to play Minecraft, you must have at least one Mojang or Minecraft "
|
||||||
|
"account logged in to MultiMC."
|
||||||
"Would you like to open the account manager to add an account now?"),
|
"Would you like to open the account manager to add an account now?"),
|
||||||
QMessageBox::Information, QMessageBox::Yes | QMessageBox::No)->exec();
|
QMessageBox::Information, QMessageBox::Yes | QMessageBox::No)->exec();
|
||||||
|
|
||||||
@ -863,7 +887,8 @@ void MainWindow::doLaunch()
|
|||||||
account->downgrade();
|
account->downgrade();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (loginWithPassword(account, tr("Your account is currently not logged in. Please enter your password to log in again.")))
|
if (loginWithPassword(account, tr("Your account is currently not logged in. Please enter "
|
||||||
|
"your password to log in again.")))
|
||||||
updateInstance(m_selectedInstance, account);
|
updateInstance(m_selectedInstance, account);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -872,7 +897,8 @@ bool MainWindow::loginWithPassword(MojangAccountPtr account, const QString& erro
|
|||||||
EditAccountDialog passDialog(errorMsg, this, EditAccountDialog::PasswordField);
|
EditAccountDialog passDialog(errorMsg, this, EditAccountDialog::PasswordField);
|
||||||
if (passDialog.exec() == QDialog::Accepted)
|
if (passDialog.exec() == QDialog::Accepted)
|
||||||
{
|
{
|
||||||
// To refresh the token, we just create an authenticate task with the given account and the user's password.
|
// To refresh the token, we just create an authenticate task with the given account and
|
||||||
|
// the user's password.
|
||||||
ProgressDialog progDialog(this);
|
ProgressDialog progDialog(this);
|
||||||
auto task = account->login(passDialog.password());
|
auto task = account->login(passDialog.password());
|
||||||
progDialog.exec(task.get());
|
progDialog.exec(task.get());
|
||||||
@ -894,15 +920,14 @@ void MainWindow::updateInstance(BaseInstance* instance, MojangAccountPtr account
|
|||||||
if (!updateTask)
|
if (!updateTask)
|
||||||
{
|
{
|
||||||
launchInstance(instance, account);
|
launchInstance(instance, account);
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
else
|
|
||||||
{
|
|
||||||
ProgressDialog tDialog(this);
|
ProgressDialog tDialog(this);
|
||||||
connect(updateTask.get(), &Task::succeeded, [this, instance, account] { launchInstance(instance, account); });
|
connect(updateTask.get(), &Task::succeeded, [this, instance, account]
|
||||||
|
{ launchInstance(instance, account); });
|
||||||
connect(updateTask.get(), SIGNAL(failed(QString)), SLOT(onGameUpdateError(QString)));
|
connect(updateTask.get(), SIGNAL(failed(QString)), SLOT(onGameUpdateError(QString)));
|
||||||
tDialog.exec(updateTask.get());
|
tDialog.exec(updateTask.get());
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
void MainWindow::launchInstance(BaseInstance *instance, MojangAccountPtr account)
|
void MainWindow::launchInstance(BaseInstance *instance, MojangAccountPtr account)
|
||||||
{
|
{
|
||||||
@ -989,13 +1014,31 @@ void MainWindow::on_actionChangeInstMCVersion_triggered()
|
|||||||
this, tr("Are you sure?"),
|
this, tr("Are you sure?"),
|
||||||
tr("This will remove any library/version customization you did previously. "
|
tr("This will remove any library/version customization you did previously. "
|
||||||
"This includes things like Forge install and similar."),
|
"This includes things like Forge install and similar."),
|
||||||
QMessageBox::Warning, QMessageBox::Ok, QMessageBox::Abort)->exec();
|
QMessageBox::Warning, QMessageBox::Ok | QMessageBox::Abort,
|
||||||
|
QMessageBox::Abort)->exec();
|
||||||
|
|
||||||
if (result != QMessageBox::Ok)
|
if (result != QMessageBox::Ok)
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
m_selectedInstance->setIntendedVersionId(vselect.selectedVersion()->descriptor());
|
m_selectedInstance->setIntendedVersionId(vselect.selectedVersion()->descriptor());
|
||||||
}
|
}
|
||||||
|
if (!MMC->accounts()->anyAccountIsValid())
|
||||||
|
{
|
||||||
|
CustomMessageBox::selectable(
|
||||||
|
this, tr("Error"),
|
||||||
|
tr("MultiMC cannot download Minecraft or update instances unless you have at least "
|
||||||
|
"one account added.\nPlease add your Mojang or Minecraft account."),
|
||||||
|
QMessageBox::Warning)->show();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
auto updateTask = m_selectedInstance->doUpdate(false /*only_prepare*/);
|
||||||
|
if (!updateTask)
|
||||||
|
{
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
ProgressDialog tDialog(this);
|
||||||
|
connect(updateTask.get(), SIGNAL(failed(QString)), SLOT(onGameUpdateError(QString)));
|
||||||
|
tDialog.exec(updateTask.get());
|
||||||
}
|
}
|
||||||
|
|
||||||
void MainWindow::on_actionChangeInstLWJGLVersion_triggered()
|
void MainWindow::on_actionChangeInstLWJGLVersion_triggered()
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
<rect>
|
<rect>
|
||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>637</width>
|
<width>706</width>
|
||||||
<height>579</height>
|
<height>579</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
@ -17,7 +17,7 @@
|
|||||||
</size>
|
</size>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
<property name="windowTitle">
|
||||||
<string>Dialog</string>
|
<string>About MultiMC</string>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout">
|
<layout class="QVBoxLayout" name="verticalLayout">
|
||||||
<item>
|
<item>
|
||||||
@ -95,13 +95,16 @@
|
|||||||
</item>
|
</item>
|
||||||
<item>
|
<item>
|
||||||
<widget class="QToolBox" name="toolBox">
|
<widget class="QToolBox" name="toolBox">
|
||||||
|
<property name="currentIndex">
|
||||||
|
<number>0</number>
|
||||||
|
</property>
|
||||||
<widget class="QWidget" name="aboutPage">
|
<widget class="QWidget" name="aboutPage">
|
||||||
<property name="geometry">
|
<property name="geometry">
|
||||||
<rect>
|
<rect>
|
||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>619</width>
|
<width>688</width>
|
||||||
<height>329</height>
|
<height>313</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<attribute name="label">
|
<attribute name="label">
|
||||||
@ -159,8 +162,8 @@
|
|||||||
<rect>
|
<rect>
|
||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>619</width>
|
<width>688</width>
|
||||||
<height>329</height>
|
<height>313</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<attribute name="label">
|
<attribute name="label">
|
||||||
@ -176,7 +179,7 @@
|
|||||||
<string><!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd">
|
<string><!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd">
|
||||||
<html><head><meta name="qrichtext" content="1" /><style type="text/css">
|
<html><head><meta name="qrichtext" content="1" /><style type="text/css">
|
||||||
p, li { white-space: pre-wrap; }
|
p, li { white-space: pre-wrap; }
|
||||||
</style></head><body style=" font-family:'Bitstream Vera Sans'; font-size:11pt; font-weight:400; font-style:normal;">
|
</style></head><body style=" font-family:'Cantarell'; font-size:11pt; font-weight:400; font-style:normal;">
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu';">Andrew Okin &lt;</span><a href="mailto:forkk@forkk.net"><span style=" font-family:'Ubuntu'; text-decoration: underline; color:#0000ff;">forkk@forkk.net</span></a><span style=" font-family:'Ubuntu';">&gt;</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu';">Andrew Okin &lt;</span><a href="mailto:forkk@forkk.net"><span style=" font-family:'Ubuntu'; text-decoration: underline; color:#0000ff;">forkk@forkk.net</span></a><span style=" font-family:'Ubuntu';">&gt;</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu';">Petr Mrázek &lt;</span><a href="mailto:peterix@gmail.com"><span style=" font-family:'Ubuntu'; text-decoration: underline; color:#0000ff;">peterix@gmail.com</span></a><span style=" font-family:'Ubuntu';">&gt;</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu';">Petr Mrázek &lt;</span><a href="mailto:peterix@gmail.com"><span style=" font-family:'Ubuntu'; text-decoration: underline; color:#0000ff;">peterix@gmail.com</span></a><span style=" font-family:'Ubuntu';">&gt;</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu';">Orochimarufan &lt;</span><a href="mailto:orochimarufan.x3@gmail.com"><span style=" font-family:'Ubuntu'; text-decoration: underline; color:#0000ff;">orochimarufan.x3@gmail.com</span></a><span style=" font-family:'Ubuntu';">&gt;</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu';">Orochimarufan &lt;</span><a href="mailto:orochimarufan.x3@gmail.com"><span style=" font-family:'Ubuntu'; text-decoration: underline; color:#0000ff;">orochimarufan.x3@gmail.com</span></a><span style=" font-family:'Ubuntu';">&gt;</span></p>
|
||||||
@ -203,8 +206,8 @@ p, li { white-space: pre-wrap; }
|
|||||||
<rect>
|
<rect>
|
||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>619</width>
|
<width>688</width>
|
||||||
<height>329</height>
|
<height>313</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<attribute name="label">
|
<attribute name="label">
|
||||||
@ -219,6 +222,11 @@ p, li { white-space: pre-wrap; }
|
|||||||
<height>0</height>
|
<height>0</height>
|
||||||
</size>
|
</size>
|
||||||
</property>
|
</property>
|
||||||
|
<property name="font">
|
||||||
|
<font>
|
||||||
|
<family>DejaVu Sans Mono</family>
|
||||||
|
</font>
|
||||||
|
</property>
|
||||||
<property name="readOnly">
|
<property name="readOnly">
|
||||||
<bool>true</bool>
|
<bool>true</bool>
|
||||||
</property>
|
</property>
|
||||||
@ -226,139 +234,128 @@ p, li { white-space: pre-wrap; }
|
|||||||
<string><!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd">
|
<string><!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0//EN" "http://www.w3.org/TR/REC-html40/strict.dtd">
|
||||||
<html><head><meta name="qrichtext" content="1" /><style type="text/css">
|
<html><head><meta name="qrichtext" content="1" /><style type="text/css">
|
||||||
p, li { white-space: pre-wrap; }
|
p, li { white-space: pre-wrap; }
|
||||||
</style></head><body style=" font-family:'Bitstream Vera Sans'; font-size:11pt; font-weight:400; font-style:normal;">
|
</style></head><body style=" font-family:'DejaVu Sans Mono'; font-size:11pt; font-weight:400; font-style:normal;">
|
||||||
<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:18pt; font-weight:600;">MultiMC</span></p>
|
<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Bitstream Vera Sans'; font-size:18pt; font-weight:600;">MultiMC</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu'; font-size:10pt;">Copyright 2012 MultiMC Contributors</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">Copyright 2012 MultiMC Contributors</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu'; font-size:10pt;">Licensed under the Apache License, Version 2.0 (the &quot;License&quot;);</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">Licensed under the Apache License, Version 2.0 (the &quot;License&quot;);</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu'; font-size:10pt;">you may not use this file except in compliance with the License.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">you may not use this file except in compliance with the License.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu'; font-size:10pt;">You may obtain a copy of the License at</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">You may obtain a copy of the License at</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:'Ubuntu'; font-size:10pt;"><br /></p>
|
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu'; font-size:10pt;"> http://www.apache.org/licenses/LICENSE-2.0</span></p>
|
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:'Ubuntu'; font-size:10pt;"><br /></p>
|
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu'; font-size:10pt;">Unless required by applicable law or agreed to in writing, software</span></p>
|
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu'; font-size:10pt;">distributed under the License is distributed on an &quot;AS IS&quot; BASIS,</span></p>
|
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu'; font-size:10pt;">WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.</span></p>
|
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu'; font-size:10pt;">See the License for the specific language governing permissions and</span></p>
|
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Ubuntu'; font-size:10pt;">limitations under the License.</span></p>
|
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-family:'Ubuntu'; font-size:10pt;"><br /></p>
|
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
||||||
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> http://www.apache.org/licenses/LICENSE-2.0</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
||||||
<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:18pt; font-weight:600;">QSLog</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">Unless required by applicable law or agreed to in writing, software</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">Copyright (c) 2010, Razvan Petru</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">distributed under the License is distributed on an &quot;AS IS&quot; BASIS,</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">All rights reserved.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.</span></p>
|
||||||
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">See the License for the specific language governing permissions and</span></p>
|
||||||
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">limitations under the License.</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">Redistribution and use in source and binary forms, with or without modification,</span></p>
|
<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Bitstream Vera Sans'; font-size:18pt; font-weight:600;">QSLog</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">are permitted provided that the following conditions are met:</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">Copyright (c) 2010, Razvan Petru</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">All rights reserved.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">* Redistributions of source code must retain the above copyright notice, this</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> list of conditions and the following disclaimer.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">Redistribution and use in source and binary forms, with or without modification,</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">* Redistributions in binary form must reproduce the above copyright notice, this</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">are permitted provided that the following conditions are met:</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> list of conditions and the following disclaimer in the documentation and/or other</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> materials provided with the distribution.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">* Redistributions of source code must retain the above copyright notice, this</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">* The name of the contributors may not be used to endorse or promote products</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> list of conditions and the following disclaimer.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> derived from this software without specific prior written permission.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">* Redistributions in binary form must reproduce the above copyright notice, this</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> list of conditions and the following disclaimer in the documentation and/or other</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS &quot;AS IS&quot; AND</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> materials provided with the distribution.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">* The name of the contributors may not be used to endorse or promote products</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> derived from this software without specific prior written permission.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS &quot;AS IS&quot; AND</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">OF THE POSSIBILITY OF SUCH DAMAGE.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE</span></p>
|
||||||
<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:18pt; font-weight:600;">Group View (instance view)</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">/**</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">OF THE POSSIBILITY OF SUCH DAMAGE.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * Copyright (C) 2007 Rafael Fernández López &lt;ereslibre@kde.org&gt;</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * Copyright (C) 2007 John Tapsell &lt;tapsell@kde.org&gt;</span></p>
|
<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Bitstream Vera Sans'; font-size:18pt; font-weight:600;">Group View (instance view)</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> *</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> /*</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * This library is free software; you can redistribute it and/or</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * Copyright (C) 2007 Rafael Fernández López &lt;ereslibre@kde.org&gt;</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * modify it under the terms of the GNU Library General Public</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * Copyright (C) 2007 John Tapsell &lt;tapsell@kde.org&gt;</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * License as published by the Free Software Foundation; either</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> *</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * version 2 of the License, or (at your option) any later version.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * This library is free software; you can redistribute it and/or</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> *</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * modify it under the terms of the GNU Library General Public</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * This library is distributed in the hope that it will be useful,</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * License as published by the Free Software Foundation; either</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * but WITHOUT ANY WARRANTY; without even the implied warranty of</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * version 2 of the License, or (at your option) any later version.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> *</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * Library General Public License for more details.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * This library is distributed in the hope that it will be useful,</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> *</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * but WITHOUT ANY WARRANTY; without even the implied warranty of</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * You should have received a copy of the GNU Library General Public License</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * along with this library; see the file COPYING.LIB. If not, write to</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * Library General Public License for more details.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> *</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * Boston, MA 02110-1301, USA.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * You should have received a copy of the GNU Library General Public License</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> */</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * along with this library; see the file COPYING.LIB. If not, write to</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * Boston, MA 02110-1301, USA.</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> */</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:18pt; font-weight:600;">Pack200</span></p>
|
<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Bitstream Vera Sans'; font-size:18pt; font-weight:600;">Pack200</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">The GNU General Public License (GPL)</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">The GNU General Public License (GPL)</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">Version 2, June 1991</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">Version 2, June 1991</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">+ &quot;CLASSPATH&quot; EXCEPTION TO THE GPL</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">+ &quot;CLASSPATH&quot; EXCEPTION TO THE GPL</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">Certain source files distributed by Oracle America and/or its affiliates are</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">Certain source files distributed by Oracle America and/or its affiliates are</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">subject to the following clarification and special exception to the GPL, but</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">subject to the following clarification and special exception to the GPL, but</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">only where Oracle has expressly included in the particular source file's header</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">only where Oracle has expressly included in the particular source file's header</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">the words &quot;Oracle designates this particular file as subject to the &quot;Classpath&quot;</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">the words &quot;Oracle designates this particular file as subject to the &quot;Classpath&quot;</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">exception as provided by Oracle in the LICENSE file that accompanied this code.&quot;</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">exception as provided by Oracle in the LICENSE file that accompanied this code.&quot;</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> Linking this library statically or dynamically with other modules is making</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> Linking this library statically or dynamically with other modules is making</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> a combined work based on this library. Thus, the terms and conditions of</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> a combined work based on this library. Thus, the terms and conditions of</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> the GNU General Public License cover the whole combination.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> the GNU General Public License cover the whole combination.</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> As a special exception, the copyright holders of this library give you</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> As a special exception, the copyright holders of this library give you</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> permission to link this library with independent modules to produce an</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> permission to link this library with independent modules to produce an</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> executable, regardless of the license terms of these independent modules,</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> executable, regardless of the license terms of these independent modules,</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> and to copy and distribute the resulting executable under terms of your</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> and to copy and distribute the resulting executable under terms of your</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> choice, provided that you also meet, for each linked independent module,</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> choice, provided that you also meet, for each linked independent module,</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> the terms and conditions of the license of that module. An independent</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> the terms and conditions of the license of that module. An independent</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> module is a module which is not derived from or based on this library. If</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> module is a module which is not derived from or based on this library. If</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> you modify this library, you may extend this exception to your version of</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> you modify this library, you may extend this exception to your version of</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> the library, but you are not obligated to do so. If you do not wish to do</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> the library, but you are not obligated to do so. If you do not wish to do</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> so, delete this exception statement from your version.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> so, delete this exception statement from your version.</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Bitstream Vera Sans'; font-size:18pt; font-weight:600;">Quazip</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">Copyright (C) 2005-2011 Sergey A. Tachenov</span></p>
|
||||||
<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:18pt; font-weight:600;">Quazip</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">Copyright (C) 2005-2011 Sergey A. Tachenov</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">This program is free software; you can redistribute it and/or modify it</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">under the terms of the GNU Lesser General Public License as published by</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">This program is free software; you can redistribute it and/or modify it</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">the Free Software Foundation; either version 2 of the License, or (at</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">under the terms of the GNU Lesser General Public License as published by</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">your option) any later version.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">the Free Software Foundation; either version 2 of the License, or (at</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">your option) any later version.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">This program is distributed in the hope that it will be useful, but</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">WITHOUT ANY WARRANTY; without even the implied warranty of</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">This program is distributed in the hope that it will be useful, but</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">WITHOUT ANY WARRANTY; without even the implied warranty of</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">General Public License for more details.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU Lesser</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">General Public License for more details.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">You should have received a copy of the GNU Lesser General Public License</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">along with this program; if not, write to the Free Software Foundation,</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">You should have received a copy of the GNU Lesser General Public License</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">along with this program; if not, write to the Free Software Foundation,</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">See COPYING file for the full LGPL text.</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">See COPYING file for the full LGPL text.</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">Original ZIP package is copyrighted by Gilles Vollant, see</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">quazip/(un)zip.h files for details, basically it's zlib license.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">Original ZIP package is copyrighted by Gilles Vollant, see</span></p>
|
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:8pt;"><br /></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">quazip/(un)zip.h files for details, basically it's zlib license.</span></p>
|
<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-family:'Bitstream Vera Sans'; font-size:18pt; font-weight:600;">xz-minidec</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;">/*</span></p>
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * XZ decompressor</span></p>
|
||||||
<p align="center" style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:18pt; font-weight:600;">xz-minidec</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> *</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;">/*</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * Authors: Lasse Collin &lt;lasse.collin@tukaani.org&gt;</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * XZ decompressor</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * Igor Pavlov &lt;http://7-zip.org/&gt;</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> *</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> *</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * Authors: Lasse Collin &lt;lasse.collin@tukaani.org&gt;</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * This file has been put into the public domain.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * Igor Pavlov &lt;http://7-zip.org/&gt;</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> * You can do whatever you want with this file.</span></p>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> *</span></p>
|
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:8pt;"> */</span></p></body></html></string>
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * This file has been put into the public domain.</span></p>
|
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> * You can do whatever you want with this file.</span></p>
|
|
||||||
<p style=" margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px;"><span style=" font-size:10pt;"> */</span></p>
|
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p>
|
|
||||||
<p style="-qt-paragraph-type:empty; margin-top:0px; margin-bottom:0px; margin-left:0px; margin-right:0px; -qt-block-indent:0; text-indent:0px; font-size:10pt;"><br /></p></body></html></string>
|
|
||||||
</property>
|
</property>
|
||||||
</widget>
|
</widget>
|
||||||
</item>
|
</item>
|
||||||
@ -404,7 +401,6 @@ p, li { white-space: pre-wrap; }
|
|||||||
</widget>
|
</widget>
|
||||||
<resources>
|
<resources>
|
||||||
<include location="../../graphics.qrc"/>
|
<include location="../../graphics.qrc"/>
|
||||||
<include location="../../graphics.qrc"/>
|
|
||||||
</resources>
|
</resources>
|
||||||
<connections/>
|
<connections/>
|
||||||
</ui>
|
</ui>
|
||||||
|
@ -36,8 +36,12 @@ AccountListDialog::AccountListDialog(QWidget *parent)
|
|||||||
ui->setupUi(this);
|
ui->setupUi(this);
|
||||||
|
|
||||||
m_accounts = MMC->accounts();
|
m_accounts = MMC->accounts();
|
||||||
// TODO: Make the "Active?" column show checkboxes or radio buttons.
|
|
||||||
ui->listView->setModel(m_accounts.get());
|
ui->listView->setModel(m_accounts.get());
|
||||||
|
ui->listView->header()->setSectionResizeMode(QHeaderView::ResizeToContents);
|
||||||
|
|
||||||
|
// Expand the account column
|
||||||
|
ui->listView->header()->setSectionResizeMode(1, QHeaderView::Stretch);
|
||||||
|
|
||||||
QItemSelectionModel* selectionModel = ui->listView->selectionModel();
|
QItemSelectionModel* selectionModel = ui->listView->selectionModel();
|
||||||
|
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
<property name="windowTitle">
|
||||||
<string>Dialog</string>
|
<string>Manage Accounts</string>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout_2">
|
<layout class="QVBoxLayout" name="verticalLayout_2">
|
||||||
<item>
|
<item>
|
||||||
|
@ -7,11 +7,11 @@
|
|||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>400</width>
|
<width>400</width>
|
||||||
<height>128</height>
|
<height>148</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
<property name="windowTitle">
|
||||||
<string>Dialog</string>
|
<string>Edit Account</string>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout">
|
<layout class="QVBoxLayout" name="verticalLayout">
|
||||||
<item>
|
<item>
|
||||||
|
@ -7,11 +7,11 @@
|
|||||||
<x>0</x>
|
<x>0</x>
|
||||||
<y>0</y>
|
<y>0</y>
|
||||||
<width>526</width>
|
<width>526</width>
|
||||||
<height>622</height>
|
<height>637</height>
|
||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
<property name="windowTitle">
|
||||||
<string/>
|
<string>Instance Settings</string>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout">
|
<layout class="QVBoxLayout" name="verticalLayout">
|
||||||
<item>
|
<item>
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
<property name="windowTitle">
|
||||||
<string>Dialog</string>
|
<string>Edit Mods</string>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout">
|
<layout class="QVBoxLayout" name="verticalLayout">
|
||||||
<item>
|
<item>
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
<property name="windowTitle">
|
||||||
<string>Dialog</string>
|
<string>Manage Lwjgl Versions</string>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout">
|
<layout class="QVBoxLayout" name="verticalLayout">
|
||||||
<item>
|
<item>
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
<property name="windowTitle">
|
||||||
<string>Dialog</string>
|
<string>Manage Mods</string>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QGridLayout" name="gridLayout">
|
<layout class="QGridLayout" name="gridLayout">
|
||||||
<item row="0" column="0">
|
<item row="0" column="0">
|
||||||
|
@ -11,7 +11,7 @@
|
|||||||
</rect>
|
</rect>
|
||||||
</property>
|
</property>
|
||||||
<property name="windowTitle">
|
<property name="windowTitle">
|
||||||
<string>Dialog</string>
|
<string>Choose Version</string>
|
||||||
</property>
|
</property>
|
||||||
<layout class="QVBoxLayout" name="verticalLayout">
|
<layout class="QVBoxLayout" name="verticalLayout">
|
||||||
<item>
|
<item>
|
||||||
|
@ -149,7 +149,7 @@ public:
|
|||||||
*/
|
*/
|
||||||
virtual SettingsObject &settings() const;
|
virtual SettingsObject &settings() const;
|
||||||
|
|
||||||
/// returns a valid update task if update is needed, NULL otherwise
|
/// returns a valid update task
|
||||||
virtual std::shared_ptr<Task> doUpdate(bool only_prepare) = 0;
|
virtual std::shared_ptr<Task> doUpdate(bool only_prepare) = 0;
|
||||||
|
|
||||||
/// returns a valid minecraft process, ready for launch with the given account.
|
/// returns a valid minecraft process, ready for launch with the given account.
|
||||||
|
@ -152,8 +152,17 @@ QStringList OneSixInstance::processMinecraftArgs(MojangAccountPtr account)
|
|||||||
token_mapping["game_directory"] = absRootDir;
|
token_mapping["game_directory"] = absRootDir;
|
||||||
QString absAssetsDir = QDir("assets/").absolutePath();
|
QString absAssetsDir = QDir("assets/").absolutePath();
|
||||||
token_mapping["game_assets"] = reconstructAssets(d->version).absolutePath();
|
token_mapping["game_assets"] = reconstructAssets(d->version).absolutePath();
|
||||||
//TODO: this is something new and not even fully implemented in the vanilla launcher.
|
|
||||||
token_mapping["user_properties"] = "{ }";
|
auto user = account->user();
|
||||||
|
QJsonObject userAttrs;
|
||||||
|
for(auto key: user.properties.keys())
|
||||||
|
{
|
||||||
|
auto array = QJsonArray::fromStringList(user.properties.values(key));
|
||||||
|
userAttrs.insert(key, array);
|
||||||
|
}
|
||||||
|
QJsonDocument value(userAttrs);
|
||||||
|
|
||||||
|
token_mapping["user_properties"] = value.toJson(QJsonDocument::Compact);
|
||||||
token_mapping["user_type"] = account->currentProfile()->legacy ? "legacy" : "mojang";
|
token_mapping["user_type"] = account->currentProfile()->legacy ? "legacy" : "mojang";
|
||||||
// 1.7.3+ assets tokens
|
// 1.7.3+ assets tokens
|
||||||
token_mapping["assets_root"] = absAssetsDir;
|
token_mapping["assets_root"] = absAssetsDir;
|
||||||
|
@ -54,11 +54,9 @@ void OneSixUpdate::executeTask()
|
|||||||
|
|
||||||
if (m_only_prepare)
|
if (m_only_prepare)
|
||||||
{
|
{
|
||||||
if (m_inst->shouldUpdate())
|
/*
|
||||||
{
|
* FIXME: in offline mode, do not proceed!
|
||||||
emitFailed("Unable to update instance in offline mode.");
|
*/
|
||||||
return;
|
|
||||||
}
|
|
||||||
setStatus("Testing the Java installation.");
|
setStatus("Testing the Java installation.");
|
||||||
QString java_path = m_inst->settings().get("JavaPath").toString();
|
QString java_path = m_inst->settings().get("JavaPath").toString();
|
||||||
|
|
||||||
@ -243,11 +241,13 @@ void OneSixUpdate::assetIndexFinished()
|
|||||||
auto objectDL = MD5EtagDownload::make(
|
auto objectDL = MD5EtagDownload::make(
|
||||||
QUrl("http://" + URLConstants::RESOURCE_BASE + objectName),
|
QUrl("http://" + URLConstants::RESOURCE_BASE + objectName),
|
||||||
objectFile.filePath());
|
objectFile.filePath());
|
||||||
|
objectDL->m_total_progress = object.size;
|
||||||
dls.append(objectDL);
|
dls.append(objectDL);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if(dls.size())
|
if(dls.size())
|
||||||
{
|
{
|
||||||
|
setStatus("Getting the assets files from Mojang...");
|
||||||
auto job = new NetJob("Assets for " + inst->name());
|
auto job = new NetJob("Assets for " + inst->name());
|
||||||
for(auto dl: dls)
|
for(auto dl: dls)
|
||||||
job->addNetAction(dl);
|
job->addNetAction(dl);
|
||||||
|
@ -68,6 +68,7 @@ MojangAccountPtr MojangAccount::loadFromJson(const QJsonObject &object)
|
|||||||
User u;
|
User u;
|
||||||
QJsonObject userStructure = object.value("user").toObject();
|
QJsonObject userStructure = object.value("user").toObject();
|
||||||
u.id = userStructure.value("id").toString();
|
u.id = userStructure.value("id").toString();
|
||||||
|
/*
|
||||||
QJsonObject propMap = userStructure.value("properties").toObject();
|
QJsonObject propMap = userStructure.value("properties").toObject();
|
||||||
for(auto key: propMap.keys())
|
for(auto key: propMap.keys())
|
||||||
{
|
{
|
||||||
@ -75,6 +76,7 @@ MojangAccountPtr MojangAccount::loadFromJson(const QJsonObject &object)
|
|||||||
for(auto value: values)
|
for(auto value: values)
|
||||||
u.properties.insert(key, value.toString());
|
u.properties.insert(key, value.toString());
|
||||||
}
|
}
|
||||||
|
*/
|
||||||
account->m_user = u;
|
account->m_user = u;
|
||||||
}
|
}
|
||||||
account->m_username = username;
|
account->m_username = username;
|
||||||
@ -119,6 +121,7 @@ QJsonObject MojangAccount::saveToJson() const
|
|||||||
QJsonObject userStructure;
|
QJsonObject userStructure;
|
||||||
{
|
{
|
||||||
userStructure.insert("id", m_user.id);
|
userStructure.insert("id", m_user.id);
|
||||||
|
/*
|
||||||
QJsonObject userAttrs;
|
QJsonObject userAttrs;
|
||||||
for(auto key: m_user.properties.keys())
|
for(auto key: m_user.properties.keys())
|
||||||
{
|
{
|
||||||
@ -126,6 +129,7 @@ QJsonObject MojangAccount::saveToJson() const
|
|||||||
userAttrs.insert(key, array);
|
userAttrs.insert(key, array);
|
||||||
}
|
}
|
||||||
userStructure.insert("properties", userAttrs);
|
userStructure.insert("properties", userAttrs);
|
||||||
|
*/
|
||||||
}
|
}
|
||||||
json.insert("user", userStructure);
|
json.insert("user", userStructure);
|
||||||
|
|
||||||
|
@ -122,6 +122,11 @@ public: /* queries */
|
|||||||
return m_profiles;
|
return m_profiles;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const User & user()
|
||||||
|
{
|
||||||
|
return m_user;
|
||||||
|
}
|
||||||
|
|
||||||
//! Get the session ID required for legacy Minecraft versions
|
//! Get the session ID required for legacy Minecraft versions
|
||||||
QString sessionId() const
|
QString sessionId() const
|
||||||
{
|
{
|
||||||
|
@ -22,10 +22,12 @@
|
|||||||
#include <QJsonArray>
|
#include <QJsonArray>
|
||||||
#include <QJsonObject>
|
#include <QJsonObject>
|
||||||
#include <QJsonParseError>
|
#include <QJsonParseError>
|
||||||
|
#include <QDir>
|
||||||
|
|
||||||
#include "logger/QsLog.h"
|
#include "logger/QsLog.h"
|
||||||
|
|
||||||
#include "logic/auth/MojangAccount.h"
|
#include "logic/auth/MojangAccount.h"
|
||||||
|
#include <pathutils.h>
|
||||||
|
|
||||||
#define ACCOUNT_LIST_FORMAT_VERSION 2
|
#define ACCOUNT_LIST_FORMAT_VERSION 2
|
||||||
|
|
||||||
@ -148,9 +150,6 @@ QVariant MojangAccountList::data(const QModelIndex &index, int role) const
|
|||||||
case Qt::DisplayRole:
|
case Qt::DisplayRole:
|
||||||
switch (index.column())
|
switch (index.column())
|
||||||
{
|
{
|
||||||
case ActiveColumn:
|
|
||||||
return account == m_activeAccount;
|
|
||||||
|
|
||||||
case NameColumn:
|
case NameColumn:
|
||||||
return account->username();
|
return account->username();
|
||||||
|
|
||||||
@ -164,6 +163,13 @@ QVariant MojangAccountList::data(const QModelIndex &index, int role) const
|
|||||||
case PointerRole:
|
case PointerRole:
|
||||||
return qVariantFromValue(account);
|
return qVariantFromValue(account);
|
||||||
|
|
||||||
|
case Qt::CheckStateRole:
|
||||||
|
switch (index.column())
|
||||||
|
{
|
||||||
|
case ActiveColumn:
|
||||||
|
return account == m_activeAccount;
|
||||||
|
}
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return QVariant();
|
return QVariant();
|
||||||
}
|
}
|
||||||
@ -212,6 +218,36 @@ int MojangAccountList::columnCount(const QModelIndex &parent) const
|
|||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Qt::ItemFlags MojangAccountList::flags(const QModelIndex &index) const
|
||||||
|
{
|
||||||
|
if (index.row() < 0 || index.row() >= rowCount(index) || !index.isValid())
|
||||||
|
{
|
||||||
|
return Qt::NoItemFlags;
|
||||||
|
}
|
||||||
|
|
||||||
|
return Qt::ItemIsUserCheckable | Qt::ItemIsEnabled | Qt::ItemIsSelectable;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool MojangAccountList::setData(const QModelIndex &index, const QVariant &value, int role)
|
||||||
|
{
|
||||||
|
if (index.row() < 0 || index.row() >= rowCount(index) || !index.isValid())
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(role == Qt::CheckStateRole)
|
||||||
|
{
|
||||||
|
if(value == Qt::Checked)
|
||||||
|
{
|
||||||
|
MojangAccountPtr account = this->at(index.row());
|
||||||
|
this->setActiveAccount(account->username());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
emit dataChanged(index, index);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
void MojangAccountList::updateListData(QList<MojangAccountPtr> versions)
|
void MojangAccountList::updateListData(QList<MojangAccountPtr> versions)
|
||||||
{
|
{
|
||||||
beginResetModel();
|
beginResetModel();
|
||||||
@ -311,6 +347,18 @@ bool MojangAccountList::saveList(const QString &filePath)
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// make sure the parent folder exists
|
||||||
|
if(!ensureFilePathExists(path))
|
||||||
|
return false;
|
||||||
|
|
||||||
|
// make sure the file wasn't overwritten with a folder before (fixes a bug)
|
||||||
|
QFileInfo finfo(path);
|
||||||
|
if(finfo.isDir())
|
||||||
|
{
|
||||||
|
QDir badDir(path);
|
||||||
|
badDir.removeRecursively();
|
||||||
|
}
|
||||||
|
|
||||||
QLOG_INFO() << "Writing account list to" << path;
|
QLOG_INFO() << "Writing account list to" << path;
|
||||||
|
|
||||||
QLOG_DEBUG() << "Building JSON data structure.";
|
QLOG_DEBUG() << "Building JSON data structure.";
|
||||||
@ -366,3 +414,13 @@ void MojangAccountList::setListFilePath(QString path, bool autosave)
|
|||||||
m_listFilePath = path;
|
m_listFilePath = path;
|
||||||
m_autosave = autosave;
|
m_autosave = autosave;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool MojangAccountList::anyAccountIsValid()
|
||||||
|
{
|
||||||
|
for(auto account:m_accounts)
|
||||||
|
{
|
||||||
|
if(account->accountStatus() != NotVerified)
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
@ -64,6 +64,8 @@ public:
|
|||||||
virtual QVariant headerData(int section, Qt::Orientation orientation, int role) const;
|
virtual QVariant headerData(int section, Qt::Orientation orientation, int role) const;
|
||||||
virtual int rowCount(const QModelIndex &parent) const;
|
virtual int rowCount(const QModelIndex &parent) const;
|
||||||
virtual int columnCount(const QModelIndex &parent) const;
|
virtual int columnCount(const QModelIndex &parent) const;
|
||||||
|
virtual Qt::ItemFlags flags(const QModelIndex &index) const;
|
||||||
|
virtual bool setData(const QModelIndex &index, const QVariant &value, int role);
|
||||||
|
|
||||||
/*!
|
/*!
|
||||||
* Adds a the given Mojang account to the account list.
|
* Adds a the given Mojang account to the account list.
|
||||||
@ -125,6 +127,11 @@ public:
|
|||||||
*/
|
*/
|
||||||
virtual void setActiveAccount(const QString &username);
|
virtual void setActiveAccount(const QString &username);
|
||||||
|
|
||||||
|
/*!
|
||||||
|
* Returns true if any of the account is at least Validated
|
||||||
|
*/
|
||||||
|
bool anyAccountIsValid();
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
/*!
|
/*!
|
||||||
* Signal emitted to indicate that the account list has changed.
|
* Signal emitted to indicate that the account list has changed.
|
||||||
|
@ -162,23 +162,17 @@ bool AuthenticateTask::processResponse(QJsonObject responseData)
|
|||||||
}
|
}
|
||||||
|
|
||||||
// this is what the vanilla launcher passes to the userProperties launch param
|
// this is what the vanilla launcher passes to the userProperties launch param
|
||||||
// doesn't seem to be used for anything so far? I don't get any of this data on my account
|
|
||||||
// (peterixxx)
|
|
||||||
// is it a good idea to log this?
|
|
||||||
if (responseData.contains("user"))
|
if (responseData.contains("user"))
|
||||||
{
|
{
|
||||||
User u;
|
User u;
|
||||||
auto obj = responseData.value("user").toObject();
|
auto obj = responseData.value("user").toObject();
|
||||||
u.id = obj.value("id").toString();
|
u.id = obj.value("id").toString();
|
||||||
QLOG_DEBUG() << "User ID: " << u.id ;
|
|
||||||
auto propArray = obj.value("properties").toArray();
|
auto propArray = obj.value("properties").toArray();
|
||||||
QLOG_DEBUG() << "User Properties: ";
|
|
||||||
for (auto prop : propArray)
|
for (auto prop : propArray)
|
||||||
{
|
{
|
||||||
auto propTuple = prop.toObject();
|
auto propTuple = prop.toObject();
|
||||||
auto name = propTuple.value("name").toString();
|
auto name = propTuple.value("name").toString();
|
||||||
auto value = propTuple.value("value").toString();
|
auto value = propTuple.value("value").toString();
|
||||||
QLOG_DEBUG() << name << " : " << value;
|
|
||||||
u.properties.insert(name, value);
|
u.properties.insert(name, value);
|
||||||
}
|
}
|
||||||
m_account->m_user = u;
|
m_account->m_user = u;
|
||||||
|
@ -112,20 +112,21 @@ bool RefreshTask::processResponse(QJsonObject responseData)
|
|||||||
// this is what the vanilla launcher passes to the userProperties launch param
|
// this is what the vanilla launcher passes to the userProperties launch param
|
||||||
if (responseData.contains("user"))
|
if (responseData.contains("user"))
|
||||||
{
|
{
|
||||||
|
User u;
|
||||||
auto obj = responseData.value("user").toObject();
|
auto obj = responseData.value("user").toObject();
|
||||||
auto userId = obj.value("id").toString();
|
u.id = obj.value("id").toString();
|
||||||
auto propArray = obj.value("properties").toArray();
|
auto propArray = obj.value("properties").toArray();
|
||||||
QLOG_DEBUG() << "User ID: " << userId;
|
|
||||||
QLOG_DEBUG() << "User Properties: ";
|
|
||||||
for (auto prop : propArray)
|
for (auto prop : propArray)
|
||||||
{
|
{
|
||||||
auto propTuple = prop.toObject();
|
auto propTuple = prop.toObject();
|
||||||
auto name = propTuple.value("name").toString();
|
auto name = propTuple.value("name").toString();
|
||||||
auto value = propTuple.value("value").toString();
|
auto value = propTuple.value("value").toString();
|
||||||
QLOG_DEBUG() << name << " : " << value;
|
u.properties.insert(name, value);
|
||||||
}
|
}
|
||||||
|
m_account->m_user = u;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// We've made it through the minefield of possible errors. Return true to indicate that
|
// We've made it through the minefield of possible errors. Return true to indicate that
|
||||||
// we've succeeded.
|
// we've succeeded.
|
||||||
QLOG_DEBUG() << "Finished reading refresh response.";
|
QLOG_DEBUG() << "Finished reading refresh response.";
|
||||||
|
@ -36,11 +36,16 @@ const static int GROUP_FILE_FORMAT_VERSION = 1;
|
|||||||
InstanceList::InstanceList(const QString &instDir, QObject *parent)
|
InstanceList::InstanceList(const QString &instDir, QObject *parent)
|
||||||
: QAbstractListModel(parent), m_instDir(instDir)
|
: QAbstractListModel(parent), m_instDir(instDir)
|
||||||
{
|
{
|
||||||
|
connect(MMC, &MultiMC::aboutToQuit, this, &InstanceList::saveGroupList);
|
||||||
|
|
||||||
|
if (!QDir::current().exists(m_instDir))
|
||||||
|
{
|
||||||
|
QDir::current().mkpath(m_instDir);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
InstanceList::~InstanceList()
|
InstanceList::~InstanceList()
|
||||||
{
|
{
|
||||||
saveGroupList();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int InstanceList::rowCount(const QModelIndex &parent) const
|
int InstanceList::rowCount(const QModelIndex &parent) const
|
||||||
@ -112,6 +117,11 @@ void InstanceList::groupChanged()
|
|||||||
saveGroupList();
|
saveGroupList();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
QStringList InstanceList::getGroups()
|
||||||
|
{
|
||||||
|
return m_groups.toList();
|
||||||
|
}
|
||||||
|
|
||||||
void InstanceList::saveGroupList()
|
void InstanceList::saveGroupList()
|
||||||
{
|
{
|
||||||
QString groupFileName = m_instDir + "/instgroups.json";
|
QString groupFileName = m_instDir + "/instgroups.json";
|
||||||
@ -121,7 +131,7 @@ void InstanceList::saveGroupList()
|
|||||||
if (!groupFile.open(QIODevice::WriteOnly | QIODevice::Truncate))
|
if (!groupFile.open(QIODevice::WriteOnly | QIODevice::Truncate))
|
||||||
{
|
{
|
||||||
// An error occurred. Ignore it.
|
// An error occurred. Ignore it.
|
||||||
QLOG_ERROR() << "Failed to read instance group file.";
|
QLOG_ERROR() << "Failed to save instance group file.";
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
QTextStream out(&groupFile);
|
QTextStream out(&groupFile);
|
||||||
@ -132,6 +142,10 @@ void InstanceList::saveGroupList()
|
|||||||
QString group = instance->group();
|
QString group = instance->group();
|
||||||
if (group.isEmpty())
|
if (group.isEmpty())
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
|
// keep a list/set of groups for choosing
|
||||||
|
m_groups.insert(group);
|
||||||
|
|
||||||
if (!groupMap.count(group))
|
if (!groupMap.count(group))
|
||||||
{
|
{
|
||||||
QSet<QString> set;
|
QSet<QString> set;
|
||||||
@ -248,6 +262,9 @@ void InstanceList::loadGroupList(QMap<QString, QString> &groupMap)
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// keep a list/set of groups for choosing
|
||||||
|
m_groups.insert(groupName);
|
||||||
|
|
||||||
// Iterate through the list of instances in the group.
|
// Iterate through the list of instances in the group.
|
||||||
QJsonArray instancesArray = groupObj.value("instances").toArray();
|
QJsonArray instancesArray = groupObj.value("instances").toArray();
|
||||||
|
|
||||||
|
@ -17,6 +17,7 @@
|
|||||||
|
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
#include <QAbstractListModel>
|
#include <QAbstractListModel>
|
||||||
|
#include <QSet>
|
||||||
#include "categorizedsortfilterproxymodel.h"
|
#include "categorizedsortfilterproxymodel.h"
|
||||||
#include <QIcon>
|
#include <QIcon>
|
||||||
|
|
||||||
@ -29,6 +30,9 @@ class InstanceList : public QAbstractListModel
|
|||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
private:
|
private:
|
||||||
void loadGroupList(QMap<QString, QString> &groupList);
|
void loadGroupList(QMap<QString, QString> &groupList);
|
||||||
|
|
||||||
|
private
|
||||||
|
slots:
|
||||||
void saveGroupList();
|
void saveGroupList();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
@ -94,6 +98,9 @@ public:
|
|||||||
InstancePtr getInstanceById(QString id) const;
|
InstancePtr getInstanceById(QString id) const;
|
||||||
|
|
||||||
QModelIndex getInstanceIndexById(const QString &id) const;
|
QModelIndex getInstanceIndexById(const QString &id) const;
|
||||||
|
|
||||||
|
// FIXME: instead of iterating through all instances and forming a set, keep the set around
|
||||||
|
QStringList getGroups();
|
||||||
signals:
|
signals:
|
||||||
void dataIsInvalid();
|
void dataIsInvalid();
|
||||||
|
|
||||||
@ -113,6 +120,7 @@ private:
|
|||||||
protected:
|
protected:
|
||||||
QString m_instDir;
|
QString m_instDir;
|
||||||
QList<InstancePtr> m_instances;
|
QList<InstancePtr> m_instances;
|
||||||
|
QSet<QString> m_groups;
|
||||||
};
|
};
|
||||||
|
|
||||||
class InstanceProxyModel : public KCategorizedSortFilterProxyModel
|
class InstanceProxyModel : public KCategorizedSortFilterProxyModel
|
||||||
|
@ -42,7 +42,9 @@ void ByteArrayDownload::start()
|
|||||||
|
|
||||||
void ByteArrayDownload::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
|
void ByteArrayDownload::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
|
||||||
{
|
{
|
||||||
emit progress(index_within_job, bytesReceived, bytesTotal);
|
m_total_progress = bytesTotal;
|
||||||
|
m_progress = bytesReceived;
|
||||||
|
emit progress(m_index_within_job, bytesReceived, bytesTotal);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ByteArrayDownload::downloadError(QNetworkReply::NetworkError error)
|
void ByteArrayDownload::downloadError(QNetworkReply::NetworkError error)
|
||||||
@ -62,14 +64,14 @@ void ByteArrayDownload::downloadFinished()
|
|||||||
m_status = Job_Finished;
|
m_status = Job_Finished;
|
||||||
m_data = m_reply->readAll();
|
m_data = m_reply->readAll();
|
||||||
m_reply.reset();
|
m_reply.reset();
|
||||||
emit succeeded(index_within_job);
|
emit succeeded(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// else the download failed
|
// else the download failed
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
m_reply.reset();
|
m_reply.reset();
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -35,14 +35,14 @@ void CacheDownload::start()
|
|||||||
{
|
{
|
||||||
if (!m_entry->stale)
|
if (!m_entry->stale)
|
||||||
{
|
{
|
||||||
emit succeeded(index_within_job);
|
emit succeeded(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
m_output_file.setFileName(m_target_path);
|
m_output_file.setFileName(m_target_path);
|
||||||
// if there already is a file and md5 checking is in effect and it can be opened
|
// if there already is a file and md5 checking is in effect and it can be opened
|
||||||
if (!ensureFilePathExists(m_target_path))
|
if (!ensureFilePathExists(m_target_path))
|
||||||
{
|
{
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
QLOG_INFO() << "Downloading " << m_url.toString();
|
QLOG_INFO() << "Downloading " << m_url.toString();
|
||||||
@ -69,7 +69,9 @@ void CacheDownload::start()
|
|||||||
|
|
||||||
void CacheDownload::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
|
void CacheDownload::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
|
||||||
{
|
{
|
||||||
emit progress(index_within_job, bytesReceived, bytesTotal);
|
m_total_progress = bytesTotal;
|
||||||
|
m_progress = bytesReceived;
|
||||||
|
emit progress(m_index_within_job, bytesReceived, bytesTotal);
|
||||||
}
|
}
|
||||||
|
|
||||||
void CacheDownload::downloadError(QNetworkReply::NetworkError error)
|
void CacheDownload::downloadError(QNetworkReply::NetworkError error)
|
||||||
@ -116,7 +118,7 @@ void CacheDownload::downloadFinished()
|
|||||||
MMC->metacache()->updateEntry(m_entry);
|
MMC->metacache()->updateEntry(m_entry);
|
||||||
|
|
||||||
m_reply.reset();
|
m_reply.reset();
|
||||||
emit succeeded(index_within_job);
|
emit succeeded(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// else the download failed
|
// else the download failed
|
||||||
@ -125,7 +127,7 @@ void CacheDownload::downloadFinished()
|
|||||||
m_output_file.close();
|
m_output_file.close();
|
||||||
m_output_file.remove();
|
m_output_file.remove();
|
||||||
m_reply.reset();
|
m_reply.reset();
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -140,7 +142,7 @@ void CacheDownload::downloadReadyRead()
|
|||||||
* Can't open the file... the job failed
|
* Can't open the file... the job failed
|
||||||
*/
|
*/
|
||||||
m_reply->abort();
|
m_reply->abort();
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -68,7 +68,7 @@ void ForgeMirrors::deferToFixedList()
|
|||||||
"https://www.creeperhost.net/link.php?id=1",
|
"https://www.creeperhost.net/link.php?id=1",
|
||||||
"http://new.creeperrepo.net/forge/maven/"});
|
"http://new.creeperrepo.net/forge/maven/"});
|
||||||
injectDownloads();
|
injectDownloads();
|
||||||
emit succeeded(index_within_job);
|
emit succeeded(m_index_within_job);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ForgeMirrors::parseMirrorList()
|
void ForgeMirrors::parseMirrorList()
|
||||||
@ -88,7 +88,7 @@ void ForgeMirrors::parseMirrorList()
|
|||||||
if(!m_mirrors.size())
|
if(!m_mirrors.size())
|
||||||
deferToFixedList();
|
deferToFixedList();
|
||||||
injectDownloads();
|
injectDownloads();
|
||||||
emit succeeded(index_within_job);
|
emit succeeded(m_index_within_job);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ForgeMirrors::injectDownloads()
|
void ForgeMirrors::injectDownloads()
|
||||||
@ -108,7 +108,9 @@ void ForgeMirrors::injectDownloads()
|
|||||||
|
|
||||||
void ForgeMirrors::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
|
void ForgeMirrors::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
|
||||||
{
|
{
|
||||||
emit progress(index_within_job, bytesReceived, bytesTotal);
|
m_total_progress = bytesTotal;
|
||||||
|
m_progress = bytesReceived;
|
||||||
|
emit progress(m_index_within_job, bytesReceived, bytesTotal);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ForgeMirrors::downloadReadyRead()
|
void ForgeMirrors::downloadReadyRead()
|
||||||
|
@ -44,20 +44,20 @@ void ForgeXzDownload::start()
|
|||||||
if (!m_entry->stale)
|
if (!m_entry->stale)
|
||||||
{
|
{
|
||||||
m_status = Job_Finished;
|
m_status = Job_Finished;
|
||||||
emit succeeded(index_within_job);
|
emit succeeded(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// can we actually create the real, final file?
|
// can we actually create the real, final file?
|
||||||
if (!ensureFilePathExists(m_target_path))
|
if (!ensureFilePathExists(m_target_path))
|
||||||
{
|
{
|
||||||
m_status = Job_Failed;
|
m_status = Job_Failed;
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (m_mirrors.empty())
|
if (m_mirrors.empty())
|
||||||
{
|
{
|
||||||
m_status = Job_Failed;
|
m_status = Job_Failed;
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -80,7 +80,9 @@ void ForgeXzDownload::start()
|
|||||||
|
|
||||||
void ForgeXzDownload::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
|
void ForgeXzDownload::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
|
||||||
{
|
{
|
||||||
emit progress(index_within_job, bytesReceived, bytesTotal);
|
m_total_progress = bytesTotal;
|
||||||
|
m_progress = bytesReceived;
|
||||||
|
emit progress(m_index_within_job, bytesReceived, bytesTotal);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ForgeXzDownload::downloadError(QNetworkReply::NetworkError error)
|
void ForgeXzDownload::downloadError(QNetworkReply::NetworkError error)
|
||||||
@ -100,7 +102,7 @@ void ForgeXzDownload::failAndTryNextMirror()
|
|||||||
m_mirror_index = next;
|
m_mirror_index = next;
|
||||||
|
|
||||||
updateUrl();
|
updateUrl();
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
}
|
}
|
||||||
|
|
||||||
void ForgeXzDownload::updateUrl()
|
void ForgeXzDownload::updateUrl()
|
||||||
@ -148,7 +150,7 @@ void ForgeXzDownload::downloadFinished()
|
|||||||
m_status = Job_Failed;
|
m_status = Job_Failed;
|
||||||
m_pack200_xz_file.remove();
|
m_pack200_xz_file.remove();
|
||||||
m_reply.reset();
|
m_reply.reset();
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -175,7 +177,7 @@ void ForgeXzDownload::downloadReadyRead()
|
|||||||
* Can't open the file... the job failed
|
* Can't open the file... the job failed
|
||||||
*/
|
*/
|
||||||
m_reply->abort();
|
m_reply->abort();
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -347,5 +349,5 @@ void ForgeXzDownload::decompressAndInstall()
|
|||||||
MMC->metacache()->updateEntry(m_entry);
|
MMC->metacache()->updateEntry(m_entry);
|
||||||
|
|
||||||
m_reply.reset();
|
m_reply.reset();
|
||||||
emit succeeded(index_within_job);
|
emit succeeded(m_index_within_job);
|
||||||
}
|
}
|
||||||
|
@ -44,7 +44,7 @@ void MD5EtagDownload::start()
|
|||||||
if (m_check_md5 && hash == m_expected_md5)
|
if (m_check_md5 && hash == m_expected_md5)
|
||||||
{
|
{
|
||||||
QLOG_INFO() << "Skipping " << m_url.toString() << ": md5 match.";
|
QLOG_INFO() << "Skipping " << m_url.toString() << ": md5 match.";
|
||||||
emit succeeded(index_within_job);
|
emit succeeded(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@ -54,7 +54,7 @@ void MD5EtagDownload::start()
|
|||||||
}
|
}
|
||||||
if (!ensureFilePathExists(filename))
|
if (!ensureFilePathExists(filename))
|
||||||
{
|
{
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -68,7 +68,7 @@ void MD5EtagDownload::start()
|
|||||||
// Plus, this way, we don't end up starting a download for a file we can't open.
|
// Plus, this way, we don't end up starting a download for a file we can't open.
|
||||||
if (!m_output_file.open(QIODevice::WriteOnly))
|
if (!m_output_file.open(QIODevice::WriteOnly))
|
||||||
{
|
{
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -86,7 +86,9 @@ void MD5EtagDownload::start()
|
|||||||
|
|
||||||
void MD5EtagDownload::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
|
void MD5EtagDownload::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
|
||||||
{
|
{
|
||||||
emit progress(index_within_job, bytesReceived, bytesTotal);
|
m_total_progress = bytesTotal;
|
||||||
|
m_progress = bytesReceived;
|
||||||
|
emit progress(m_index_within_job, bytesReceived, bytesTotal);
|
||||||
}
|
}
|
||||||
|
|
||||||
void MD5EtagDownload::downloadError(QNetworkReply::NetworkError error)
|
void MD5EtagDownload::downloadError(QNetworkReply::NetworkError error)
|
||||||
@ -107,7 +109,7 @@ void MD5EtagDownload::downloadFinished()
|
|||||||
|
|
||||||
QLOG_INFO() << "Finished " << m_url.toString() << " got " << m_reply->rawHeader("ETag").constData();
|
QLOG_INFO() << "Finished " << m_url.toString() << " got " << m_reply->rawHeader("ETag").constData();
|
||||||
m_reply.reset();
|
m_reply.reset();
|
||||||
emit succeeded(index_within_job);
|
emit succeeded(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
// else the download failed
|
// else the download failed
|
||||||
@ -115,7 +117,7 @@ void MD5EtagDownload::downloadFinished()
|
|||||||
{
|
{
|
||||||
m_output_file.close();
|
m_output_file.close();
|
||||||
m_reply.reset();
|
m_reply.reset();
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -130,7 +132,7 @@ void MD5EtagDownload::downloadReadyRead()
|
|||||||
* Can't open the file... the job failed
|
* Can't open the file... the job failed
|
||||||
*/
|
*/
|
||||||
m_reply->abort();
|
m_reply->abort();
|
||||||
emit failed(index_within_job);
|
emit failed(m_index_within_job);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -38,6 +38,19 @@ protected:
|
|||||||
public:
|
public:
|
||||||
virtual ~NetAction() {};
|
virtual ~NetAction() {};
|
||||||
|
|
||||||
|
public:
|
||||||
|
virtual qint64 totalProgress() const
|
||||||
|
{
|
||||||
|
return m_total_progress;
|
||||||
|
}
|
||||||
|
virtual qint64 currentProgress() const
|
||||||
|
{
|
||||||
|
return m_progress;
|
||||||
|
}
|
||||||
|
virtual qint64 numberOfFailures() const
|
||||||
|
{
|
||||||
|
return m_failures;
|
||||||
|
}
|
||||||
public:
|
public:
|
||||||
/// the network reply
|
/// the network reply
|
||||||
std::shared_ptr<QNetworkReply> m_reply;
|
std::shared_ptr<QNetworkReply> m_reply;
|
||||||
@ -46,10 +59,16 @@ public:
|
|||||||
QUrl m_url;
|
QUrl m_url;
|
||||||
|
|
||||||
/// The file's status
|
/// The file's status
|
||||||
JobStatus m_status;
|
JobStatus m_status = Job_NotStarted;
|
||||||
|
|
||||||
/// index within the parent job
|
/// index within the parent job
|
||||||
int index_within_job = 0;
|
int m_index_within_job = 0;
|
||||||
|
|
||||||
|
qint64 m_progress = 0;
|
||||||
|
qint64 m_total_progress = 1;
|
||||||
|
|
||||||
|
/// number of failures up to this point
|
||||||
|
int m_failures = 0;
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void started(int index);
|
void started(int index);
|
||||||
|
@ -36,10 +36,16 @@ public:
|
|||||||
template <typename T> bool addNetAction(T action)
|
template <typename T> bool addNetAction(T action)
|
||||||
{
|
{
|
||||||
NetActionPtr base = std::static_pointer_cast<NetAction>(action);
|
NetActionPtr base = std::static_pointer_cast<NetAction>(action);
|
||||||
base->index_within_job = downloads.size();
|
base->m_index_within_job = downloads.size();
|
||||||
downloads.append(action);
|
downloads.append(action);
|
||||||
parts_progress.append(part_info());
|
part_info pi;
|
||||||
total_progress++;
|
{
|
||||||
|
pi.current_progress = base->currentProgress();
|
||||||
|
pi.total_progress = base->totalProgress();
|
||||||
|
pi.failures = base->numberOfFailures();
|
||||||
|
}
|
||||||
|
parts_progress.append(pi);
|
||||||
|
total_progress += pi.total_progress;
|
||||||
// if this is already running, the action needs to be started right away!
|
// if this is already running, the action needs to be started right away!
|
||||||
if (isRunning())
|
if (isRunning())
|
||||||
{
|
{
|
||||||
|
@ -78,7 +78,9 @@ bool PasteUpload::parseResult(QJsonDocument doc, QString *parseError)
|
|||||||
parseError = new QString(object.value("error").toString());
|
parseError = new QString(object.value("error").toString());
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
// FIXME: not the place for GUI things.
|
||||||
QString pasteUrl = object.value("paste").toObject().value("link").toString();
|
QString pasteUrl = object.value("paste").toObject().value("link").toString();
|
||||||
QDesktopServices::openUrl(pasteUrl);
|
QDesktopServices::openUrl(pasteUrl);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,175 +0,0 @@
|
|||||||
/* Copyright 2013 MultiMC Contributors
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "S3ListBucket.h"
|
|
||||||
#include "MultiMC.h"
|
|
||||||
#include "logger/QsLog.h"
|
|
||||||
#include <QUrlQuery>
|
|
||||||
#include <qxmlstream.h>
|
|
||||||
#include <QDomDocument>
|
|
||||||
|
|
||||||
inline QDomElement getDomElementByTagName(QDomElement parent, QString tagname)
|
|
||||||
{
|
|
||||||
QDomNodeList elementList = parent.elementsByTagName(tagname);
|
|
||||||
if (elementList.count())
|
|
||||||
return elementList.at(0).toElement();
|
|
||||||
else
|
|
||||||
return QDomElement();
|
|
||||||
}
|
|
||||||
|
|
||||||
S3ListBucket::S3ListBucket(QUrl url) : NetAction()
|
|
||||||
{
|
|
||||||
m_url = url;
|
|
||||||
m_status = Job_NotStarted;
|
|
||||||
}
|
|
||||||
|
|
||||||
void S3ListBucket::start()
|
|
||||||
{
|
|
||||||
QUrl finalUrl = m_url;
|
|
||||||
if (current_marker.size())
|
|
||||||
{
|
|
||||||
QUrlQuery query;
|
|
||||||
query.addQueryItem("marker", current_marker);
|
|
||||||
finalUrl.setQuery(query);
|
|
||||||
}
|
|
||||||
QNetworkRequest request(finalUrl);
|
|
||||||
request.setHeader(QNetworkRequest::UserAgentHeader, "MultiMC/5.0 (Uncached)");
|
|
||||||
auto worker = MMC->qnam();
|
|
||||||
QNetworkReply *rep = worker->get(request);
|
|
||||||
|
|
||||||
m_reply = std::shared_ptr<QNetworkReply>(rep);
|
|
||||||
connect(rep, SIGNAL(downloadProgress(qint64, qint64)),
|
|
||||||
SLOT(downloadProgress(qint64, qint64)));
|
|
||||||
connect(rep, SIGNAL(finished()), SLOT(downloadFinished()));
|
|
||||||
connect(rep, SIGNAL(error(QNetworkReply::NetworkError)),
|
|
||||||
SLOT(downloadError(QNetworkReply::NetworkError)));
|
|
||||||
connect(rep, SIGNAL(readyRead()), SLOT(downloadReadyRead()));
|
|
||||||
}
|
|
||||||
|
|
||||||
void S3ListBucket::downloadProgress(qint64 bytesReceived, qint64 bytesTotal)
|
|
||||||
{
|
|
||||||
emit progress(index_within_job, bytesSoFar + bytesReceived, bytesSoFar + bytesTotal);
|
|
||||||
}
|
|
||||||
|
|
||||||
void S3ListBucket::downloadError(QNetworkReply::NetworkError error)
|
|
||||||
{
|
|
||||||
// error happened during download.
|
|
||||||
QLOG_ERROR() << "Error getting URL:" << m_url.toString().toLocal8Bit()
|
|
||||||
<< "Network error: " << error;
|
|
||||||
m_status = Job_Failed;
|
|
||||||
}
|
|
||||||
|
|
||||||
void S3ListBucket::processValidReply()
|
|
||||||
{
|
|
||||||
QLOG_TRACE() << "GOT: " << m_url.toString() << " marker:" << current_marker;
|
|
||||||
auto readContents = [&](QXmlStreamReader & xml)
|
|
||||||
{
|
|
||||||
QString Key, ETag, Size;
|
|
||||||
while (!(xml.tokenType() == QXmlStreamReader::EndElement && xml.name() == "Contents"))
|
|
||||||
{
|
|
||||||
if (xml.tokenType() == QXmlStreamReader::StartElement)
|
|
||||||
{
|
|
||||||
if (xml.name() == "Key")
|
|
||||||
{
|
|
||||||
Key = xml.readElementText();
|
|
||||||
}
|
|
||||||
if (xml.name() == "ETag")
|
|
||||||
{
|
|
||||||
ETag = xml.readElementText();
|
|
||||||
}
|
|
||||||
if (xml.name() == "Size")
|
|
||||||
{
|
|
||||||
Size = xml.readElementText();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
xml.readNext();
|
|
||||||
}
|
|
||||||
if (xml.error() != QXmlStreamReader::NoError)
|
|
||||||
return;
|
|
||||||
objects.append({Key, ETag, Size.toLongLong()});
|
|
||||||
};
|
|
||||||
|
|
||||||
// nothing went wrong...
|
|
||||||
QByteArray ba = m_reply->readAll();
|
|
||||||
|
|
||||||
QString xmlErrorMsg;
|
|
||||||
|
|
||||||
bool is_truncated = false;
|
|
||||||
QXmlStreamReader xml(ba);
|
|
||||||
while (!xml.atEnd() && !xml.hasError())
|
|
||||||
{
|
|
||||||
/* Read next element.*/
|
|
||||||
QXmlStreamReader::TokenType token = xml.readNext();
|
|
||||||
/* If token is just StartDocument, we'll go to next.*/
|
|
||||||
if (token == QXmlStreamReader::StartDocument)
|
|
||||||
{
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
if (token == QXmlStreamReader::StartElement)
|
|
||||||
{
|
|
||||||
/* If it's named person, we'll dig the information from there.*/
|
|
||||||
if (xml.name() == "Contents")
|
|
||||||
{
|
|
||||||
readContents(xml);
|
|
||||||
}
|
|
||||||
else if (xml.name() == "IsTruncated")
|
|
||||||
{
|
|
||||||
is_truncated = (xml.readElementText() == "true");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (xml.hasError())
|
|
||||||
{
|
|
||||||
QLOG_ERROR() << "Failed to process s3.amazonaws.com/Minecraft.Resources. XML error:"
|
|
||||||
<< xml.errorString() << ba;
|
|
||||||
emit failed(index_within_job);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (is_truncated)
|
|
||||||
{
|
|
||||||
current_marker = objects.last().Key;
|
|
||||||
bytesSoFar += m_reply->size();
|
|
||||||
m_reply.reset();
|
|
||||||
start();
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
m_status = Job_Finished;
|
|
||||||
m_reply.reset();
|
|
||||||
emit succeeded(index_within_job);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
void S3ListBucket::downloadFinished()
|
|
||||||
{
|
|
||||||
// if the download succeeded
|
|
||||||
if (m_status != Job_Failed)
|
|
||||||
{
|
|
||||||
processValidReply();
|
|
||||||
}
|
|
||||||
// else the download failed
|
|
||||||
else
|
|
||||||
{
|
|
||||||
m_reply.reset();
|
|
||||||
emit failed(index_within_job);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void S3ListBucket::downloadReadyRead()
|
|
||||||
{
|
|
||||||
// ~_~
|
|
||||||
}
|
|
@ -1,57 +0,0 @@
|
|||||||
/* Copyright 2013 MultiMC Contributors
|
|
||||||
*
|
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
||||||
* you may not use this file except in compliance with the License.
|
|
||||||
* You may obtain a copy of the License at
|
|
||||||
*
|
|
||||||
* http://www.apache.org/licenses/LICENSE-2.0
|
|
||||||
*
|
|
||||||
* Unless required by applicable law or agreed to in writing, software
|
|
||||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
||||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
||||||
* See the License for the specific language governing permissions and
|
|
||||||
* limitations under the License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
#pragma once
|
|
||||||
#include "NetAction.h"
|
|
||||||
|
|
||||||
struct S3Object
|
|
||||||
{
|
|
||||||
QString Key;
|
|
||||||
QString ETag;
|
|
||||||
qlonglong size;
|
|
||||||
};
|
|
||||||
|
|
||||||
typedef std::shared_ptr<class S3ListBucket> S3ListBucketPtr;
|
|
||||||
class S3ListBucket : public NetAction
|
|
||||||
{
|
|
||||||
Q_OBJECT
|
|
||||||
public:
|
|
||||||
S3ListBucket(QUrl url);
|
|
||||||
static S3ListBucketPtr make(QUrl url)
|
|
||||||
{
|
|
||||||
return S3ListBucketPtr(new S3ListBucket(url));
|
|
||||||
}
|
|
||||||
|
|
||||||
public:
|
|
||||||
QList<S3Object> objects;
|
|
||||||
|
|
||||||
public
|
|
||||||
slots:
|
|
||||||
virtual void start() override;
|
|
||||||
|
|
||||||
protected
|
|
||||||
slots:
|
|
||||||
virtual void downloadProgress(qint64 bytesReceived, qint64 bytesTotal) override;
|
|
||||||
virtual void downloadError(QNetworkReply::NetworkError error) override;
|
|
||||||
virtual void downloadFinished() override;
|
|
||||||
virtual void downloadReadyRead() override;
|
|
||||||
|
|
||||||
private:
|
|
||||||
void processValidReply();
|
|
||||||
|
|
||||||
private:
|
|
||||||
qint64 bytesSoFar = 0;
|
|
||||||
QString current_marker;
|
|
||||||
};
|
|
38
main.cpp
Normal file
38
main.cpp
Normal file
@ -0,0 +1,38 @@
|
|||||||
|
#include "MultiMC.h"
|
||||||
|
#include "gui/MainWindow.h"
|
||||||
|
|
||||||
|
int main_gui(MultiMC &app)
|
||||||
|
{
|
||||||
|
// show main window
|
||||||
|
MainWindow mainWin;
|
||||||
|
mainWin.restoreState(QByteArray::fromBase64(MMC->settings()->get("MainWindowState").toByteArray()));
|
||||||
|
mainWin.restoreGeometry(QByteArray::fromBase64(MMC->settings()->get("MainWindowGeometry").toByteArray()));
|
||||||
|
mainWin.show();
|
||||||
|
mainWin.checkSetDefaultJava();
|
||||||
|
auto exitCode = app.exec();
|
||||||
|
|
||||||
|
// Update if necessary.
|
||||||
|
if (!app.getExitUpdatePath().isEmpty())
|
||||||
|
app.installUpdates(app.getExitUpdatePath(), false);
|
||||||
|
|
||||||
|
return exitCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char *argv[])
|
||||||
|
{
|
||||||
|
// initialize Qt
|
||||||
|
MultiMC app(argc, argv);
|
||||||
|
|
||||||
|
Q_INIT_RESOURCE(graphics);
|
||||||
|
Q_INIT_RESOURCE(generated);
|
||||||
|
|
||||||
|
switch (app.status())
|
||||||
|
{
|
||||||
|
case MultiMC::Initialized:
|
||||||
|
return main_gui(app);
|
||||||
|
case MultiMC::Failed:
|
||||||
|
return 1;
|
||||||
|
case MultiMC::Succeeded:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
@ -34,7 +34,7 @@ macro(ADD_UPDATER_TEST CLASS)
|
|||||||
set(TEST_TARGET updater_${CLASS})
|
set(TEST_TARGET updater_${CLASS})
|
||||||
add_executable(${TEST_TARGET} ${CLASS}.cpp)
|
add_executable(${TEST_TARGET} ${CLASS}.cpp)
|
||||||
target_link_libraries(${TEST_TARGET} updatershared)
|
target_link_libraries(${TEST_TARGET} updatershared)
|
||||||
add_test(${TEST_TARGET} ${TEST_TARGET})
|
add_test(NAME ${TEST_TARGET} COMMAND ${TEST_TARGET})
|
||||||
if (APPLE)
|
if (APPLE)
|
||||||
set_target_properties(${TEST_TARGET} PROPERTIES LINK_FLAGS "-framework Security -framework Cocoa")
|
set_target_properties(${TEST_TARGET} PROPERTIES LINK_FLAGS "-framework Security -framework Cocoa")
|
||||||
endif()
|
endif()
|
||||||
|
82
tests/CMakeLists.txt
Normal file
82
tests/CMakeLists.txt
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
# run the unit tests with `make test`
|
||||||
|
find_package(Qt5 COMPONENTS Test Core Network Widgets)
|
||||||
|
|
||||||
|
include_directories(${MMC_SRC})
|
||||||
|
|
||||||
|
unset(MultiMC_TESTS)
|
||||||
|
macro(add_unit_test name)
|
||||||
|
unset(srcs)
|
||||||
|
foreach(arg ${testname} ${ARGN})
|
||||||
|
list(APPEND srcs ${CMAKE_CURRENT_SOURCE_DIR}/${arg})
|
||||||
|
endforeach()
|
||||||
|
add_executable(tst_${name} ${srcs})
|
||||||
|
qt5_use_modules(tst_${name} Test Core Network Widgets)
|
||||||
|
target_link_libraries(tst_${name} MultiMC_common)
|
||||||
|
list(APPEND MultiMC_TESTS tst_${name})
|
||||||
|
add_test(NAME ${name} COMMAND tst_${name})
|
||||||
|
endmacro()
|
||||||
|
|
||||||
|
# Tests START #
|
||||||
|
|
||||||
|
add_unit_test(pathutils tst_pathutils.cpp)
|
||||||
|
add_unit_test(userutils tst_userutils.cpp)
|
||||||
|
|
||||||
|
# Tests END #
|
||||||
|
|
||||||
|
set(COVERAGE_SOURCE_DIRS
|
||||||
|
${MMC_SRC}/logic/*
|
||||||
|
${MMC_SRC}/logic/auth/*
|
||||||
|
${MMC_SRC}/logic/auth/flows/*
|
||||||
|
${MMC_SRC}/logic/lists/*
|
||||||
|
${MMC_SRC}/logic/net/*
|
||||||
|
${MMC_SRC}/logic/tasks/*
|
||||||
|
${MMC_SRC}/gui/*
|
||||||
|
${MMC_SRC}/gui/dialogs/*
|
||||||
|
${MMC_SRC}/gui/widgets/*
|
||||||
|
${MMC_SRC}/depends/settings/include/*
|
||||||
|
${MMC_SRC}/depends/settings/src/*
|
||||||
|
${MMC_SRC}/depends/util/include/*
|
||||||
|
${MMC_SRC}/depends/util/src/*
|
||||||
|
)
|
||||||
|
|
||||||
|
if(MultiMC_CODE_COVERAGE)
|
||||||
|
unset(MultiMC_RUN_TESTS)
|
||||||
|
unset(MultiMC_TEST_COVERAGE_FILES)
|
||||||
|
|
||||||
|
foreach(test ${MultiMC_TESTS})
|
||||||
|
add_custom_target(MultiMC_RUN_TEST_${test}
|
||||||
|
COMMAND lcov -d ${CMAKE_CURRENT_BINARY_DIR} -z -q # clean test
|
||||||
|
&& lcov -d ${MMC_BIN} -z -q # clean common
|
||||||
|
&& lcov -d ${MMC_BIN}/depends/settings/CMakeFiles/libSettings.dir -z -q # clean settings
|
||||||
|
&& lcov -d ${MMC_BIN}/depends/utils/CMakeFiles/libUtil.dir -z -q # clean utils
|
||||||
|
&& ${MMC_BIN}/${test} -o coverage_${test}.out,xml # run test
|
||||||
|
&& lcov -q --checksum -b ${MMC_SRC} -d ${CMAKE_CURRENT_BINARY_DIR} -c -o coverage_${test}.info # generate for test
|
||||||
|
&& lcov -q --checksum -b ${MMC_SRC} -d ${MMC_BIN} -c -o coverage_common.info # generate for common
|
||||||
|
&& lcov -q --checksum -b ${MMC_SRC} -d ${MMC_BIN}/depends/settings/CMakeFiles/libSettings.dir -c -o coverage_settings.info # generate for settings
|
||||||
|
&& lcov -q --checksum -b ${MMC_SRC} -d ${MMC_BIN}/depends/util/CMakeFiles/libUtil.dir -c -o coverage_utils.info # generate for utils
|
||||||
|
&& lcov -q --checksum -b ${MMC_SRC} -d .
|
||||||
|
-a coverage_${test}.info -a coverage_common.info -a coverage_settings.info -a coverage_utils.info
|
||||||
|
-o coverage_${test}-combined.info # combine test and common
|
||||||
|
&& lcov -q --checksum -b ${MMC_SRC} --list-full-path --extract coverage_${test}-combined.info ${COVERAGE_SOURCE_DIRS} -o coverage_${test}-stripped.info # strip
|
||||||
|
WORKING_DIRECTORY ${CMAKE_CURRENT_BINARY_DIR}
|
||||||
|
VERBATIM
|
||||||
|
DEPENDS ${test}
|
||||||
|
COMMENT "Running ${test}..."
|
||||||
|
)
|
||||||
|
list(APPEND MultiMC_TEST_COVERAGE_FILES coverage_${test}-stripped.info)
|
||||||
|
list(APPEND MultiMC_RUN_TESTS MultiMC_RUN_TEST_${test})
|
||||||
|
endforeach(test)
|
||||||
|
|
||||||
|
add_custom_target(MultiMC_GENERATE_COVERAGE
|
||||||
|
DEPENDS ${MultiMC_RUN_TESTS}
|
||||||
|
COMMENT "Generating coverage files..."
|
||||||
|
)
|
||||||
|
add_custom_target(MultiMC_GENERATE_COVERAGE_HTML
|
||||||
|
COMMAND genhtml -t "MultiMC 5 Test Coverage" --num-spaces 4 --demangle-cpp --legend -o ${MMC_SRC}/html/coverage ${MultiMC_TEST_COVERAGE_FILES}
|
||||||
|
DEPENDS MultiMC_GENERATE_COVERAGE
|
||||||
|
COMMENT "Generating test coverage html..."
|
||||||
|
)
|
||||||
|
add_custom_target(MultiMC_RUN_TESTS DEPENDS MultiMC_GENERATE_COVERAGE_HTML)
|
||||||
|
endif(MultiMC_CODE_COVERAGE)
|
||||||
|
|
||||||
|
add_subdirectory(data)
|
31
tests/TestUtil.h
Normal file
31
tests/TestUtil.h
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <QFile>
|
||||||
|
#include <QCoreApplication>
|
||||||
|
#include <QTest>
|
||||||
|
#include <QDir>
|
||||||
|
|
||||||
|
#include "MultiMC.h"
|
||||||
|
|
||||||
|
struct TestsInternal
|
||||||
|
{
|
||||||
|
static QByteArray readFile(const QString &fileName)
|
||||||
|
{
|
||||||
|
QFile f(fileName);
|
||||||
|
f.open(QFile::ReadOnly);
|
||||||
|
return f.readAll();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
#define MULTIMC_GET_TEST_FILE(file) TestsInternal::readFile(QFINDTESTDATA( file ))
|
||||||
|
|
||||||
|
#define QTEST_GUILESS_MAIN_MULTIMC(TestObject) \
|
||||||
|
int main(int argc, char *argv[]) \
|
||||||
|
{ \
|
||||||
|
char *argv_[] = { argv[0] }; \
|
||||||
|
int argc_ = 1; \
|
||||||
|
MultiMC app(argc_, argv_, QDir::temp().absoluteFilePath("MultiMC_Test")); \
|
||||||
|
app.setAttribute(Qt::AA_Use96Dpi, true); \
|
||||||
|
TestObject tc; \
|
||||||
|
return QTest::qExec(&tc, argc, argv); \
|
||||||
|
}
|
4
tests/data/CMakeLists.txt
Normal file
4
tests/data/CMakeLists.txt
Normal file
@ -0,0 +1,4 @@
|
|||||||
|
add_custom_target(MultiMC_Test_Data
|
||||||
|
ALL
|
||||||
|
COMMAND ${CMAKE_COMMAND} -E copy_directory ${CMAKE_CURRENT_SOURCE_DIR} ${CMAKE_CURRENT_BINARY_DIR}
|
||||||
|
)
|
6
tests/data/tst_userutils-test_createShortcut-unix
Executable file
6
tests/data/tst_userutils-test_createShortcut-unix
Executable file
@ -0,0 +1,6 @@
|
|||||||
|
[Desktop Entry]
|
||||||
|
Type=Application
|
||||||
|
TryExec=asdfDest
|
||||||
|
Exec=asdfDest 'arg1' 'arg2'
|
||||||
|
Name=asdf
|
||||||
|
Icon=
|
76
tests/tst_pathutils.cpp
Normal file
76
tests/tst_pathutils.cpp
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
#include <QTest>
|
||||||
|
#include "TestUtil.h"
|
||||||
|
|
||||||
|
#include "depends/util/include/pathutils.h"
|
||||||
|
|
||||||
|
class PathUtilsTest : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
private
|
||||||
|
slots:
|
||||||
|
void initTestCase()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
void cleanupTestCase()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void test_PathCombine1_data()
|
||||||
|
{
|
||||||
|
QTest::addColumn<QString>("result");
|
||||||
|
QTest::addColumn<QString>("path1");
|
||||||
|
QTest::addColumn<QString>("path2");
|
||||||
|
|
||||||
|
#if defined(Q_OS_UNIX)
|
||||||
|
QTest::newRow("unix 1") << "/abc/def/ghi/jkl" << "/abc/def" << "ghi/jkl";
|
||||||
|
QTest::newRow("unix 2") << "/abc/def/ghi/jkl" << "/abc/def/" << "ghi/jkl";
|
||||||
|
#elif defined(Q_OS_WIN)
|
||||||
|
QTest::newRow("win, from C:") << "C:\\abc" << "C:" << "abc\\def";
|
||||||
|
QTest::newRow("win 1") << "C:\\abc\\def\\ghi\\jkl" << "C:\\abc\\def" << "ghi\\jkl";
|
||||||
|
QTest::newRow("win 2") << "C:\\abc\\def\\ghi\\jkl" << "C:\\abc\\def\\" << "ghi\\jkl";
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
void test_PathCombine1()
|
||||||
|
{
|
||||||
|
QFETCH(QString, result);
|
||||||
|
QFETCH(QString, path1);
|
||||||
|
QFETCH(QString, path2);
|
||||||
|
|
||||||
|
QCOMPARE(PathCombine(path1, path2), result);
|
||||||
|
}
|
||||||
|
|
||||||
|
void test_PathCombine2_data()
|
||||||
|
{
|
||||||
|
QTest::addColumn<QString>("result");
|
||||||
|
QTest::addColumn<QString>("path1");
|
||||||
|
QTest::addColumn<QString>("path2");
|
||||||
|
QTest::addColumn<QString>("path3");
|
||||||
|
|
||||||
|
#if defined(Q_OS_UNIX)
|
||||||
|
QTest::newRow("unix 1") << "/abc/def/ghi/jkl" << "/abc" << "def" << "ghi/jkl";
|
||||||
|
QTest::newRow("unix 2") << "/abc/def/ghi/jkl" << "/abc/" << "def" << "ghi/jkl";
|
||||||
|
QTest::newRow("unix 3") << "/abc/def/ghi/jkl" << "/abc" << "def/" << "ghi/jkl";
|
||||||
|
QTest::newRow("unix 4") << "/abc/def/ghi/jkl" << "/abc/" << "def/" << "ghi/jkl";
|
||||||
|
#elif defined(Q_OS_WIN)
|
||||||
|
QTest::newRow("win 1") << "C:\\abc\\def\\ghi\\jkl" << "C:\\abc" << "def" << "ghi\\jkl";
|
||||||
|
QTest::newRow("win 2") << "C:\\abc\\def\\ghi\\jkl" << "C:\\abc\\" << "def" << "ghi\\jkl";
|
||||||
|
QTest::newRow("win 3") << "C:\\abc\\def\\ghi\\jkl" << "C:\\abc" << "def\\" << "ghi\\jkl";
|
||||||
|
QTest::newRow("win 4") << "C:\\abc\\def\\ghi\\jkl" << "C:\\abc\\" << "def" << "ghi\\jkl";
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
void test_PathCombine2()
|
||||||
|
{
|
||||||
|
QFETCH(QString, result);
|
||||||
|
QFETCH(QString, path1);
|
||||||
|
QFETCH(QString, path2);
|
||||||
|
QFETCH(QString, path3);
|
||||||
|
|
||||||
|
QCOMPARE(PathCombine(path1, path2, path3), result);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
QTEST_GUILESS_MAIN_MULTIMC(PathUtilsTest)
|
||||||
|
|
||||||
|
#include "tst_pathutils.moc"
|
66
tests/tst_userutils.cpp
Normal file
66
tests/tst_userutils.cpp
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
#include <QTest>
|
||||||
|
#include <QStandardPaths>
|
||||||
|
#include "TestUtil.h"
|
||||||
|
|
||||||
|
#include "depends/util/include/userutils.h"
|
||||||
|
|
||||||
|
class UserUtilsTest : public QObject
|
||||||
|
{
|
||||||
|
Q_OBJECT
|
||||||
|
private
|
||||||
|
slots:
|
||||||
|
void initTestCase()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
void cleanupTestCase()
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
void test_getDesktop()
|
||||||
|
{
|
||||||
|
QCOMPARE(Util::getDesktopDir(), QStandardPaths::writableLocation(QStandardPaths::DesktopLocation));
|
||||||
|
}
|
||||||
|
|
||||||
|
void test_createShortcut_data()
|
||||||
|
{
|
||||||
|
QTest::addColumn<QString>("location");
|
||||||
|
QTest::addColumn<QString>("dest");
|
||||||
|
QTest::addColumn<QStringList>("args");
|
||||||
|
QTest::addColumn<QString>("name");
|
||||||
|
QTest::addColumn<QString>("iconLocation");
|
||||||
|
QTest::addColumn<QByteArray>("result");
|
||||||
|
|
||||||
|
QTest::newRow("unix") << QDir::currentPath()
|
||||||
|
<< "asdfDest"
|
||||||
|
<< (QStringList() << "arg1" << "arg2")
|
||||||
|
<< "asdf"
|
||||||
|
<< QString()
|
||||||
|
#if defined(Q_OS_LINUX)
|
||||||
|
<< MULTIMC_GET_TEST_FILE("data/tst_userutils-test_createShortcut-unix")
|
||||||
|
#elif defined(Q_OS_WIN)
|
||||||
|
<< QString()
|
||||||
|
#endif
|
||||||
|
;
|
||||||
|
}
|
||||||
|
|
||||||
|
void test_createShortcut()
|
||||||
|
{
|
||||||
|
QFETCH(QString, location);
|
||||||
|
QFETCH(QString, dest);
|
||||||
|
QFETCH(QStringList, args);
|
||||||
|
QFETCH(QString, name);
|
||||||
|
QFETCH(QString, iconLocation);
|
||||||
|
QFETCH(QByteArray, result);
|
||||||
|
|
||||||
|
QVERIFY(Util::createShortCut(location, dest, args, name, iconLocation));
|
||||||
|
QCOMPARE(QString::fromLocal8Bit(TestsInternal::readFile(location + QDir::separator() + name + ".desktop")), QString::fromLocal8Bit(result));
|
||||||
|
|
||||||
|
//QDir().remove(location);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
QTEST_GUILESS_MAIN_MULTIMC(UserUtilsTest)
|
||||||
|
|
||||||
|
#include "tst_userutils.moc"
|
Loading…
x
Reference in New Issue
Block a user