2023-09-05 00:18:36 +03:00
|
|
|
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
/*
|
|
|
|
* Prism Launcher - Minecraft Launcher
|
|
|
|
* Copyright (c) 2023 Trial97 <alexandru.tripon97@gmail.com>
|
|
|
|
*
|
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation, version 3.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
2023-09-05 20:13:16 +03:00
|
|
|
#include "SkinManageDialog.h"
|
|
|
|
#include "ui_SkinManageDialog.h"
|
2023-09-05 00:18:36 +03:00
|
|
|
|
|
|
|
#include <FileSystem.h>
|
|
|
|
#include <QAction>
|
|
|
|
#include <QDialog>
|
2023-09-05 20:13:16 +03:00
|
|
|
#include <QEventLoop>
|
|
|
|
#include <QFileDialog>
|
2023-09-05 00:18:36 +03:00
|
|
|
#include <QFileInfo>
|
|
|
|
#include <QKeyEvent>
|
|
|
|
#include <QListView>
|
2023-09-05 20:13:16 +03:00
|
|
|
#include <QMimeDatabase>
|
|
|
|
#include <QPainter>
|
|
|
|
#include <QUrl>
|
2023-09-05 00:18:36 +03:00
|
|
|
|
|
|
|
#include "Application.h"
|
|
|
|
#include "DesktopServices.h"
|
2023-09-05 20:13:16 +03:00
|
|
|
#include "Json.h"
|
2023-09-05 00:18:36 +03:00
|
|
|
#include "QObjectPtr.h"
|
|
|
|
|
|
|
|
#include "minecraft/auth/AccountTask.h"
|
2023-09-05 20:13:16 +03:00
|
|
|
#include "minecraft/auth/Parsers.h"
|
2023-09-05 00:18:36 +03:00
|
|
|
#include "minecraft/skins/CapeChange.h"
|
|
|
|
#include "minecraft/skins/SkinDelete.h"
|
|
|
|
#include "minecraft/skins/SkinList.h"
|
|
|
|
#include "minecraft/skins/SkinModel.h"
|
|
|
|
#include "minecraft/skins/SkinUpload.h"
|
|
|
|
|
2023-09-05 20:13:16 +03:00
|
|
|
#include "net/Download.h"
|
2023-09-05 00:18:36 +03:00
|
|
|
#include "net/NetJob.h"
|
|
|
|
#include "tasks/Task.h"
|
|
|
|
|
|
|
|
#include "ui/dialogs/CustomMessageBox.h"
|
|
|
|
#include "ui/dialogs/ProgressDialog.h"
|
|
|
|
#include "ui/instanceview/InstanceDelegate.h"
|
|
|
|
|
|
|
|
SkinManageDialog::SkinManageDialog(QWidget* parent, MinecraftAccountPtr acct)
|
|
|
|
: QDialog(parent), m_acct(acct), ui(new Ui::SkinManageDialog), m_list(this, APPLICATION->settings()->get("SkinsDir").toString(), acct)
|
|
|
|
{
|
|
|
|
ui->setupUi(this);
|
|
|
|
|
|
|
|
setWindowModality(Qt::WindowModal);
|
|
|
|
|
|
|
|
auto contentsWidget = ui->listView;
|
|
|
|
contentsWidget->setViewMode(QListView::IconMode);
|
|
|
|
contentsWidget->setFlow(QListView::LeftToRight);
|
|
|
|
contentsWidget->setIconSize(QSize(48, 48));
|
|
|
|
contentsWidget->setMovement(QListView::Static);
|
|
|
|
contentsWidget->setResizeMode(QListView::Adjust);
|
|
|
|
contentsWidget->setSelectionMode(QAbstractItemView::SingleSelection);
|
|
|
|
contentsWidget->setSpacing(5);
|
|
|
|
contentsWidget->setWordWrap(false);
|
|
|
|
contentsWidget->setWrapping(true);
|
|
|
|
contentsWidget->setUniformItemSizes(true);
|
|
|
|
contentsWidget->setTextElideMode(Qt::ElideRight);
|
|
|
|
contentsWidget->setVerticalScrollMode(QAbstractItemView::ScrollPerPixel);
|
|
|
|
contentsWidget->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff);
|
|
|
|
contentsWidget->installEventFilter(this);
|
|
|
|
contentsWidget->setItemDelegate(new ListViewDelegate());
|
|
|
|
|
|
|
|
contentsWidget->setAcceptDrops(true);
|
|
|
|
contentsWidget->setDropIndicatorShown(true);
|
|
|
|
contentsWidget->viewport()->setAcceptDrops(true);
|
|
|
|
contentsWidget->setDragDropMode(QAbstractItemView::DropOnly);
|
|
|
|
contentsWidget->setDefaultDropAction(Qt::CopyAction);
|
|
|
|
|
|
|
|
contentsWidget->installEventFilter(this);
|
|
|
|
contentsWidget->setModel(&m_list);
|
|
|
|
|
|
|
|
connect(contentsWidget, SIGNAL(doubleClicked(QModelIndex)), SLOT(activated(QModelIndex)));
|
|
|
|
|
|
|
|
connect(contentsWidget->selectionModel(), SIGNAL(selectionChanged(QItemSelection, QItemSelection)),
|
|
|
|
SLOT(selectionChanged(QItemSelection, QItemSelection)));
|
|
|
|
connect(ui->listView, &QListView::customContextMenuRequested, this, &SkinManageDialog::show_context_menu);
|
|
|
|
|
|
|
|
setupCapes();
|
|
|
|
|
|
|
|
ui->listView->setCurrentIndex(m_list.index(m_list.getSelectedAccountSkin()));
|
|
|
|
}
|
|
|
|
|
|
|
|
SkinManageDialog::~SkinManageDialog()
|
|
|
|
{
|
|
|
|
delete ui;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinManageDialog::activated(QModelIndex index)
|
|
|
|
{
|
|
|
|
m_selected_skin = index.data(Qt::UserRole).toString();
|
|
|
|
accept();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinManageDialog::selectionChanged(QItemSelection selected, QItemSelection deselected)
|
|
|
|
{
|
|
|
|
if (selected.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
QString key = selected.first().indexes().first().data(Qt::UserRole).toString();
|
|
|
|
if (key.isEmpty())
|
|
|
|
return;
|
|
|
|
m_selected_skin = key;
|
|
|
|
auto skin = m_list.skin(key);
|
|
|
|
if (!skin)
|
|
|
|
return;
|
|
|
|
ui->selectedModel->setPixmap(skin->getTexture().scaled(128, 128, Qt::KeepAspectRatio, Qt::FastTransformation));
|
|
|
|
ui->capeCombo->setCurrentIndex(m_capes_idx.value(skin->getCapeId()));
|
|
|
|
ui->steveBtn->setChecked(skin->getModel() == SkinModel::CLASSIC);
|
|
|
|
ui->alexBtn->setChecked(skin->getModel() == SkinModel::SLIM);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinManageDialog::delayed_scroll(QModelIndex model_index)
|
|
|
|
{
|
|
|
|
auto contentsWidget = ui->listView;
|
|
|
|
contentsWidget->scrollTo(model_index);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinManageDialog::on_openDirBtn_clicked()
|
|
|
|
{
|
|
|
|
DesktopServices::openDirectory(m_list.getDir(), true);
|
|
|
|
}
|
|
|
|
|
2023-09-05 20:13:16 +03:00
|
|
|
void SkinManageDialog::on_fileBtn_clicked()
|
2023-09-05 00:18:36 +03:00
|
|
|
{
|
|
|
|
auto filter = QMimeDatabase().mimeTypeForName("image/png").filterString();
|
|
|
|
QString raw_path = QFileDialog::getOpenFileName(this, tr("Select Skin Texture"), QString(), filter);
|
2023-09-05 20:13:16 +03:00
|
|
|
auto message = m_list.installSkin(raw_path, {});
|
|
|
|
if (!message.isEmpty()) {
|
|
|
|
CustomMessageBox::selectable(this, tr("Selected file is not a valid skin"), message, QMessageBox::Critical)->show();
|
2023-09-05 00:18:36 +03:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
QPixmap previewCape(QPixmap capeImage)
|
|
|
|
{
|
|
|
|
QPixmap preview = QPixmap(10, 16);
|
|
|
|
QPainter painter(&preview);
|
|
|
|
painter.drawPixmap(0, 0, capeImage.copy(1, 1, 10, 16));
|
|
|
|
return preview.scaled(80, 128, Qt::IgnoreAspectRatio, Qt::FastTransformation);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinManageDialog::setupCapes()
|
|
|
|
{
|
|
|
|
// FIXME: add a model for this, download/refresh the capes on demand
|
|
|
|
auto& accountData = *m_acct->accountData();
|
|
|
|
int index = 0;
|
|
|
|
ui->capeCombo->addItem(tr("No Cape"), QVariant());
|
|
|
|
auto currentCape = accountData.minecraftProfile.currentCape;
|
|
|
|
if (currentCape.isEmpty()) {
|
|
|
|
ui->capeCombo->setCurrentIndex(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
auto capesDir = FS::PathCombine(m_list.getDir(), "capes");
|
|
|
|
NetJob::Ptr job{ new NetJob(tr("Download capes"), APPLICATION->network()) };
|
|
|
|
bool needsToDownload = false;
|
|
|
|
for (auto& cape : accountData.minecraftProfile.capes) {
|
|
|
|
auto path = FS::PathCombine(capesDir, cape.id + ".png");
|
|
|
|
if (cape.data.size()) {
|
|
|
|
QPixmap capeImage;
|
|
|
|
if (capeImage.loadFromData(cape.data, "PNG") && capeImage.save(path)) {
|
|
|
|
m_capes[cape.id] = previewCape(capeImage);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (QFileInfo(path).exists()) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (!cape.url.isEmpty()) {
|
|
|
|
needsToDownload = true;
|
|
|
|
job->addNetAction(Net::Download::makeFile(cape.url, path));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (needsToDownload) {
|
|
|
|
ProgressDialog dlg(this);
|
|
|
|
dlg.execWithTask(job.get());
|
|
|
|
}
|
|
|
|
for (auto& cape : accountData.minecraftProfile.capes) {
|
|
|
|
index++;
|
|
|
|
QPixmap capeImage;
|
|
|
|
if (!m_capes.contains(cape.id)) {
|
|
|
|
auto path = FS::PathCombine(capesDir, cape.id + ".png");
|
|
|
|
if (QFileInfo(path).exists() && capeImage.load(path)) {
|
|
|
|
capeImage = previewCape(capeImage);
|
|
|
|
m_capes[cape.id] = capeImage;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!capeImage.isNull()) {
|
|
|
|
ui->capeCombo->addItem(capeImage, cape.alias, cape.id);
|
|
|
|
} else {
|
|
|
|
ui->capeCombo->addItem(cape.alias, cape.id);
|
|
|
|
}
|
|
|
|
|
|
|
|
m_capes_idx[cape.id] = index;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinManageDialog::on_capeCombo_currentIndexChanged(int index)
|
|
|
|
{
|
|
|
|
auto id = ui->capeCombo->currentData();
|
|
|
|
ui->capeImage->setPixmap(m_capes.value(id.toString(), {}));
|
|
|
|
if (auto skin = m_list.skin(m_selected_skin); skin) {
|
|
|
|
skin->setCapeId(id.toString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinManageDialog::on_steveBtn_toggled(bool checked)
|
|
|
|
{
|
|
|
|
if (auto skin = m_list.skin(m_selected_skin); skin) {
|
|
|
|
skin->setModel(checked ? SkinModel::CLASSIC : SkinModel::SLIM);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinManageDialog::accept()
|
|
|
|
{
|
|
|
|
auto skin = m_list.skin(m_selected_skin);
|
|
|
|
if (!skin)
|
|
|
|
reject();
|
|
|
|
auto path = skin->getPath();
|
|
|
|
|
|
|
|
ProgressDialog prog(this);
|
|
|
|
NetJob::Ptr skinUpload{ new NetJob(tr("Change skin"), APPLICATION->network(), 1) };
|
|
|
|
|
|
|
|
if (!QFile::exists(path)) {
|
|
|
|
CustomMessageBox::selectable(this, tr("Skin Upload"), tr("Skin file does not exist!"), QMessageBox::Warning)->exec();
|
|
|
|
reject();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-09-08 19:50:46 +03:00
|
|
|
skinUpload->addNetAction(SkinUpload::make(m_acct->accessToken(), skin->getPath(), skin->getModelString()));
|
2023-09-05 00:18:36 +03:00
|
|
|
|
|
|
|
auto selectedCape = skin->getCapeId();
|
|
|
|
if (selectedCape != m_acct->accountData()->minecraftProfile.currentCape) {
|
|
|
|
skinUpload->addNetAction(CapeChange::make(m_acct->accessToken(), selectedCape));
|
|
|
|
}
|
|
|
|
|
|
|
|
skinUpload->addTask(m_acct->refresh().staticCast<Task>());
|
|
|
|
if (prog.execWithTask(skinUpload.get()) != QDialog::Accepted) {
|
|
|
|
CustomMessageBox::selectable(this, tr("Skin Upload"), tr("Failed to upload skin!"), QMessageBox::Warning)->exec();
|
|
|
|
reject();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
skin->setURL(m_acct->accountData()->minecraftProfile.skin.url);
|
|
|
|
QDialog::accept();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinManageDialog::on_resetBtn_clicked()
|
|
|
|
{
|
|
|
|
ProgressDialog prog(this);
|
|
|
|
NetJob::Ptr skinReset{ new NetJob(tr("Reset skin"), APPLICATION->network(), 1) };
|
|
|
|
skinReset->addNetAction(SkinDelete::make(m_acct->accessToken()));
|
|
|
|
skinReset->addTask(m_acct->refresh().staticCast<Task>());
|
|
|
|
if (prog.execWithTask(skinReset.get()) != QDialog::Accepted) {
|
|
|
|
CustomMessageBox::selectable(this, tr("Skin Delete"), tr("Failed to delete current skin!"), QMessageBox::Warning)->exec();
|
|
|
|
reject();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
QDialog::accept();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinManageDialog::show_context_menu(const QPoint& pos)
|
|
|
|
{
|
|
|
|
QMenu myMenu(tr("Context menu"), this);
|
|
|
|
myMenu.addAction(ui->action_Rename_Skin);
|
|
|
|
myMenu.addAction(ui->action_Delete_Skin);
|
|
|
|
|
|
|
|
myMenu.exec(ui->listView->mapToGlobal(pos));
|
|
|
|
}
|
|
|
|
|
|
|
|
bool SkinManageDialog::eventFilter(QObject* obj, QEvent* ev)
|
|
|
|
{
|
|
|
|
if (obj == ui->listView) {
|
|
|
|
if (ev->type() == QEvent::KeyPress) {
|
|
|
|
QKeyEvent* keyEvent = static_cast<QKeyEvent*>(ev);
|
|
|
|
switch (keyEvent->key()) {
|
|
|
|
case Qt::Key_Delete:
|
|
|
|
on_action_Delete_Skin_triggered(false);
|
|
|
|
return true;
|
|
|
|
case Qt::Key_F2:
|
|
|
|
on_action_Rename_Skin_triggered(false);
|
|
|
|
return true;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return QDialog::eventFilter(obj, ev);
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinManageDialog::on_action_Rename_Skin_triggered(bool checked)
|
|
|
|
{
|
|
|
|
if (!m_selected_skin.isEmpty()) {
|
|
|
|
ui->listView->edit(ui->listView->currentIndex());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinManageDialog::on_action_Delete_Skin_triggered(bool checked)
|
|
|
|
{
|
|
|
|
if (m_selected_skin.isEmpty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (m_list.getSkinIndex(m_selected_skin) == m_list.getSelectedAccountSkin()) {
|
|
|
|
CustomMessageBox::selectable(this, tr("Delete error"), tr("Can not delete skin that is in use."), QMessageBox::Warning);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto skin = m_list.skin(m_selected_skin);
|
|
|
|
if (!skin)
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto response = CustomMessageBox::selectable(this, tr("Confirm Deletion"),
|
|
|
|
tr("You are about to delete \"%1\".\n"
|
|
|
|
"Are you sure?")
|
|
|
|
.arg(skin->name()),
|
|
|
|
QMessageBox::Warning, QMessageBox::Yes | QMessageBox::No, QMessageBox::No)
|
|
|
|
->exec();
|
|
|
|
|
|
|
|
if (response == QMessageBox::Yes) {
|
|
|
|
if (!m_list.deleteSkin(m_selected_skin, true)) {
|
|
|
|
m_list.deleteSkin(m_selected_skin, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-09-05 20:13:16 +03:00
|
|
|
|
|
|
|
void SkinManageDialog::on_urlBtn_clicked()
|
|
|
|
{
|
|
|
|
auto url = QUrl(ui->urlLine->text());
|
|
|
|
if (!url.isValid()) {
|
|
|
|
CustomMessageBox::selectable(this, tr("Invalid url"), tr("Invalid url"), QMessageBox::Critical)->show();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ui->urlLine->setText("");
|
|
|
|
|
|
|
|
NetJob::Ptr job{ new NetJob(tr("Download skin"), APPLICATION->network()) };
|
|
|
|
|
|
|
|
auto path = FS::PathCombine(m_list.getDir(), url.fileName());
|
|
|
|
job->addNetAction(Net::Download::makeFile(url, path));
|
|
|
|
ProgressDialog dlg(this);
|
|
|
|
dlg.execWithTask(job.get());
|
|
|
|
SkinModel s(path);
|
|
|
|
if (!s.isValid()) {
|
|
|
|
CustomMessageBox::selectable(this, tr("URL is not a valid skin"), tr("Skin images must be 64x64 or 64x32 pixel PNG files."),
|
|
|
|
QMessageBox::Critical)
|
|
|
|
->show();
|
|
|
|
QFile::remove(path);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (QFileInfo(path).suffix().isEmpty()) {
|
|
|
|
QFile::rename(path, path + ".png");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class WaitTask : public Task {
|
|
|
|
public:
|
|
|
|
WaitTask() : m_loop(), m_done(false){};
|
|
|
|
virtual ~WaitTask() = default;
|
|
|
|
|
|
|
|
public slots:
|
|
|
|
void quit()
|
|
|
|
{
|
|
|
|
m_done = true;
|
|
|
|
m_loop.quit();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected:
|
|
|
|
virtual void executeTask()
|
|
|
|
{
|
|
|
|
if (!m_done)
|
|
|
|
m_loop.exec();
|
|
|
|
emitSucceeded();
|
|
|
|
};
|
|
|
|
|
|
|
|
private:
|
|
|
|
QEventLoop m_loop;
|
|
|
|
bool m_done;
|
|
|
|
};
|
|
|
|
|
|
|
|
void SkinManageDialog::on_userBtn_clicked()
|
|
|
|
{
|
|
|
|
auto user = ui->urlLine->text();
|
|
|
|
if (user.isEmpty()) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
ui->urlLine->setText("");
|
|
|
|
MinecraftProfile mcProfile;
|
|
|
|
auto path = FS::PathCombine(m_list.getDir(), user + ".png");
|
|
|
|
|
|
|
|
NetJob::Ptr job{ new NetJob(tr("Download user skin"), APPLICATION->network(), 1) };
|
|
|
|
|
|
|
|
auto uuidOut = std::make_shared<QByteArray>();
|
|
|
|
auto profileOut = std::make_shared<QByteArray>();
|
|
|
|
|
|
|
|
auto uuidLoop = makeShared<WaitTask>();
|
|
|
|
auto profileLoop = makeShared<WaitTask>();
|
|
|
|
|
|
|
|
auto getUUID = Net::Download::makeByteArray("https://api.mojang.com/users/profiles/minecraft/" + user, uuidOut);
|
|
|
|
auto getProfile = Net::Download::makeByteArray(QUrl(), profileOut);
|
|
|
|
auto downloadSkin = Net::Download::makeFile(QUrl(), path);
|
|
|
|
|
|
|
|
connect(getUUID.get(), &Task::aborted, uuidLoop.get(), &WaitTask::quit);
|
|
|
|
connect(getUUID.get(), &Task::failed, uuidLoop.get(), &WaitTask::quit);
|
|
|
|
connect(getProfile.get(), &Task::aborted, profileLoop.get(), &WaitTask::quit);
|
|
|
|
connect(getProfile.get(), &Task::failed, profileLoop.get(), &WaitTask::quit);
|
|
|
|
|
|
|
|
connect(getUUID.get(), &Task::succeeded, this, [uuidLoop, uuidOut, job, getProfile] {
|
|
|
|
try {
|
|
|
|
QJsonParseError parse_error{};
|
|
|
|
QJsonDocument doc = QJsonDocument::fromJson(*uuidOut, &parse_error);
|
|
|
|
if (parse_error.error != QJsonParseError::NoError) {
|
|
|
|
qWarning() << "Error while parsing JSON response from Minecraft skin service at " << parse_error.offset
|
|
|
|
<< " reason: " << parse_error.errorString();
|
|
|
|
uuidLoop->quit();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
const auto root = doc.object();
|
|
|
|
auto id = Json::ensureString(root, "id");
|
|
|
|
if (!id.isEmpty()) {
|
|
|
|
getProfile->setUrl("https://sessionserver.mojang.com/session/minecraft/profile/" + id);
|
|
|
|
} else {
|
|
|
|
job->abort();
|
|
|
|
}
|
|
|
|
} catch (const Exception& e) {
|
|
|
|
qCritical() << "Couldn't load skin json:" << e.cause();
|
|
|
|
}
|
|
|
|
uuidLoop->quit();
|
|
|
|
});
|
|
|
|
|
|
|
|
connect(getProfile.get(), &Task::succeeded, this, [profileLoop, profileOut, job, getProfile, &mcProfile, downloadSkin] {
|
|
|
|
if (Parsers::parseMinecraftProfileMojang(*profileOut, mcProfile)) {
|
|
|
|
downloadSkin->setUrl(mcProfile.skin.url);
|
|
|
|
} else {
|
|
|
|
job->abort();
|
|
|
|
}
|
|
|
|
profileLoop->quit();
|
|
|
|
});
|
|
|
|
|
|
|
|
job->addNetAction(getUUID);
|
|
|
|
job->addTask(uuidLoop);
|
|
|
|
job->addNetAction(getProfile);
|
|
|
|
job->addTask(profileLoop);
|
|
|
|
job->addNetAction(downloadSkin);
|
|
|
|
ProgressDialog dlg(this);
|
|
|
|
dlg.execWithTask(job.get());
|
|
|
|
|
|
|
|
SkinModel s(path);
|
|
|
|
s.setModel(mcProfile.skin.variant == "slim" ? SkinModel::SLIM : SkinModel::CLASSIC);
|
|
|
|
s.setURL(mcProfile.skin.url);
|
|
|
|
if (m_capes.contains(mcProfile.currentCape)) {
|
|
|
|
s.setCapeId(mcProfile.currentCape);
|
|
|
|
}
|
2023-09-05 23:45:32 +03:00
|
|
|
m_list.updateSkin(&s);
|
2023-09-05 20:13:16 +03:00
|
|
|
}
|