Browse Source
Merge pull request #1781 from DarkLordZach/applet-profile-select
Merge pull request #1781 from DarkLordZach/applet-profile-select
am: Implement HLE profile selector appletnce_cpp
committed by
GitHub
13 changed files with 466 additions and 0 deletions
-
4src/core/CMakeLists.txt
-
11src/core/core.cpp
-
5src/core/core.h
-
19src/core/frontend/applets/profile_select.cpp
-
27src/core/frontend/applets/profile_select.h
-
4src/core/hle/service/am/am.cpp
-
77src/core/hle/service/am/applets/profile_select.cpp
-
50src/core/hle/service/am/applets/profile_select.h
-
2src/yuzu/CMakeLists.txt
-
168src/yuzu/applets/profile_select.cpp
-
73src/yuzu/applets/profile_select.h
-
24src/yuzu/main.cpp
-
2src/yuzu/main.h
@ -0,0 +1,19 @@ |
|||||
|
// Copyright 2018 yuzu emulator team
|
||||
|
// Licensed under GPLv2 or any later version
|
||||
|
// Refer to the license.txt file included.
|
||||
|
|
||||
|
#include "core/frontend/applets/profile_select.h"
|
||||
|
#include "core/settings.h"
|
||||
|
|
||||
|
namespace Core::Frontend { |
||||
|
|
||||
|
ProfileSelectApplet::~ProfileSelectApplet() = default; |
||||
|
|
||||
|
void DefaultProfileSelectApplet::SelectProfile( |
||||
|
std::function<void(std::optional<Service::Account::UUID>)> callback) const { |
||||
|
Service::Account::ProfileManager manager; |
||||
|
callback(manager.GetUser(Settings::values.current_user).value_or(Service::Account::UUID{})); |
||||
|
LOG_INFO(Service_ACC, "called, selecting current user instead of prompting..."); |
||||
|
} |
||||
|
|
||||
|
} // namespace Core::Frontend
|
||||
@ -0,0 +1,27 @@ |
|||||
|
// Copyright 2018 yuzu emulator team |
||||
|
// Licensed under GPLv2 or any later version |
||||
|
// Refer to the license.txt file included. |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include <functional> |
||||
|
#include <optional> |
||||
|
#include "core/hle/service/acc/profile_manager.h" |
||||
|
|
||||
|
namespace Core::Frontend { |
||||
|
|
||||
|
class ProfileSelectApplet { |
||||
|
public: |
||||
|
virtual ~ProfileSelectApplet(); |
||||
|
|
||||
|
virtual void SelectProfile( |
||||
|
std::function<void(std::optional<Service::Account::UUID>)> callback) const = 0; |
||||
|
}; |
||||
|
|
||||
|
class DefaultProfileSelectApplet final : public ProfileSelectApplet { |
||||
|
public: |
||||
|
void SelectProfile( |
||||
|
std::function<void(std::optional<Service::Account::UUID>)> callback) const override; |
||||
|
}; |
||||
|
|
||||
|
} // namespace Core::Frontend |
||||
@ -0,0 +1,77 @@ |
|||||
|
// Copyright 2018 yuzu emulator team
|
||||
|
// Licensed under GPLv2 or any later version
|
||||
|
// Refer to the license.txt file included.
|
||||
|
|
||||
|
#include <cstring>
|
||||
|
|
||||
|
#include "common/assert.h"
|
||||
|
#include "common/string_util.h"
|
||||
|
#include "core/core.h"
|
||||
|
#include "core/frontend/applets/software_keyboard.h"
|
||||
|
#include "core/hle/service/am/am.h"
|
||||
|
#include "core/hle/service/am/applets/profile_select.h"
|
||||
|
|
||||
|
namespace Service::AM::Applets { |
||||
|
|
||||
|
constexpr ResultCode ERR_USER_CANCELLED_SELECTION{ErrorModule::Account, 1}; |
||||
|
|
||||
|
ProfileSelect::ProfileSelect() = default; |
||||
|
ProfileSelect::~ProfileSelect() = default; |
||||
|
|
||||
|
void ProfileSelect::Initialize() { |
||||
|
complete = false; |
||||
|
status = RESULT_SUCCESS; |
||||
|
final_data.clear(); |
||||
|
|
||||
|
Applet::Initialize(); |
||||
|
|
||||
|
const auto user_config_storage = broker.PopNormalDataToApplet(); |
||||
|
ASSERT(user_config_storage != nullptr); |
||||
|
const auto& user_config = user_config_storage->GetData(); |
||||
|
|
||||
|
ASSERT(user_config.size() >= sizeof(UserSelectionConfig)); |
||||
|
std::memcpy(&config, user_config.data(), sizeof(UserSelectionConfig)); |
||||
|
} |
||||
|
|
||||
|
bool ProfileSelect::TransactionComplete() const { |
||||
|
return complete; |
||||
|
} |
||||
|
|
||||
|
ResultCode ProfileSelect::GetStatus() const { |
||||
|
return status; |
||||
|
} |
||||
|
|
||||
|
void ProfileSelect::ExecuteInteractive() { |
||||
|
UNREACHABLE_MSG("Attempted to call interactive execution on non-interactive applet."); |
||||
|
} |
||||
|
|
||||
|
void ProfileSelect::Execute() { |
||||
|
if (complete) { |
||||
|
broker.PushNormalDataFromApplet(IStorage{final_data}); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
const auto& frontend{Core::System::GetInstance().GetProfileSelector()}; |
||||
|
|
||||
|
frontend.SelectProfile([this](std::optional<Account::UUID> uuid) { SelectionComplete(uuid); }); |
||||
|
} |
||||
|
|
||||
|
void ProfileSelect::SelectionComplete(std::optional<Account::UUID> uuid) { |
||||
|
UserSelectionOutput output{}; |
||||
|
|
||||
|
if (uuid.has_value() && uuid->uuid != Account::INVALID_UUID) { |
||||
|
output.result = 0; |
||||
|
output.uuid_selected = uuid->uuid; |
||||
|
} else { |
||||
|
status = ERR_USER_CANCELLED_SELECTION; |
||||
|
output.result = ERR_USER_CANCELLED_SELECTION.raw; |
||||
|
output.uuid_selected = Account::INVALID_UUID; |
||||
|
} |
||||
|
|
||||
|
final_data = std::vector<u8>(sizeof(UserSelectionOutput)); |
||||
|
std::memcpy(final_data.data(), &output, final_data.size()); |
||||
|
broker.PushNormalDataFromApplet(IStorage{final_data}); |
||||
|
broker.SignalStateChanged(); |
||||
|
} |
||||
|
|
||||
|
} // namespace Service::AM::Applets
|
||||
@ -0,0 +1,50 @@ |
|||||
|
// Copyright 2018 yuzu emulator team |
||||
|
// Licensed under GPLv2 or any later version |
||||
|
// Refer to the license.txt file included. |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include <vector> |
||||
|
|
||||
|
#include "common/common_funcs.h" |
||||
|
#include "core/hle/service/acc/profile_manager.h" |
||||
|
#include "core/hle/service/am/applets/applets.h" |
||||
|
|
||||
|
namespace Service::AM::Applets { |
||||
|
|
||||
|
struct UserSelectionConfig { |
||||
|
// TODO(DarkLordZach): RE this structure |
||||
|
// It seems to be flags and the like that determine the UI of the applet on the switch... from |
||||
|
// my research this is safe to ignore for now. |
||||
|
INSERT_PADDING_BYTES(0xA0); |
||||
|
}; |
||||
|
static_assert(sizeof(UserSelectionConfig) == 0xA0, "UserSelectionConfig has incorrect size."); |
||||
|
|
||||
|
struct UserSelectionOutput { |
||||
|
u64 result; |
||||
|
u128 uuid_selected; |
||||
|
}; |
||||
|
static_assert(sizeof(UserSelectionOutput) == 0x18, "UserSelectionOutput has incorrect size."); |
||||
|
|
||||
|
class ProfileSelect final : public Applet { |
||||
|
public: |
||||
|
ProfileSelect(); |
||||
|
~ProfileSelect() override; |
||||
|
|
||||
|
void Initialize() override; |
||||
|
|
||||
|
bool TransactionComplete() const override; |
||||
|
ResultCode GetStatus() const override; |
||||
|
void ExecuteInteractive() override; |
||||
|
void Execute() override; |
||||
|
|
||||
|
void SelectionComplete(std::optional<Account::UUID> uuid); |
||||
|
|
||||
|
private: |
||||
|
UserSelectionConfig config; |
||||
|
bool complete = false; |
||||
|
ResultCode status = RESULT_SUCCESS; |
||||
|
std::vector<u8> final_data; |
||||
|
}; |
||||
|
|
||||
|
} // namespace Service::AM::Applets |
||||
@ -0,0 +1,168 @@ |
|||||
|
// Copyright 2018 yuzu Emulator Project
|
||||
|
// Licensed under GPLv2 or any later version
|
||||
|
// Refer to the license.txt file included.
|
||||
|
|
||||
|
#include <mutex>
|
||||
|
#include <QDialogButtonBox>
|
||||
|
#include <QLabel>
|
||||
|
#include <QLineEdit>
|
||||
|
#include <QScrollArea>
|
||||
|
#include <QStandardItemModel>
|
||||
|
#include <QVBoxLayout>
|
||||
|
#include "common/file_util.h"
|
||||
|
#include "common/string_util.h"
|
||||
|
#include "core/hle/lock.h"
|
||||
|
#include "yuzu/applets/profile_select.h"
|
||||
|
#include "yuzu/main.h"
|
||||
|
|
||||
|
// Same backup JPEG used by acc IProfile::GetImage if no jpeg found
|
||||
|
constexpr std::array<u8, 107> backup_jpeg{ |
||||
|
0xff, 0xd8, 0xff, 0xdb, 0x00, 0x43, 0x00, 0x03, 0x02, 0x02, 0x02, 0x02, 0x02, 0x03, 0x02, 0x02, |
||||
|
0x02, 0x03, 0x03, 0x03, 0x03, 0x04, 0x06, 0x04, 0x04, 0x04, 0x04, 0x04, 0x08, 0x06, 0x06, 0x05, |
||||
|
0x06, 0x09, 0x08, 0x0a, 0x0a, 0x09, 0x08, 0x09, 0x09, 0x0a, 0x0c, 0x0f, 0x0c, 0x0a, 0x0b, 0x0e, |
||||
|
0x0b, 0x09, 0x09, 0x0d, 0x11, 0x0d, 0x0e, 0x0f, 0x10, 0x10, 0x11, 0x10, 0x0a, 0x0c, 0x12, 0x13, |
||||
|
0x12, 0x10, 0x13, 0x0f, 0x10, 0x10, 0x10, 0xff, 0xc9, 0x00, 0x0b, 0x08, 0x00, 0x01, 0x00, 0x01, |
||||
|
0x01, 0x01, 0x11, 0x00, 0xff, 0xcc, 0x00, 0x06, 0x00, 0x10, 0x10, 0x05, 0xff, 0xda, 0x00, 0x08, |
||||
|
0x01, 0x01, 0x00, 0x00, 0x3f, 0x00, 0xd2, 0xcf, 0x20, 0xff, 0xd9, |
||||
|
}; |
||||
|
|
||||
|
QString FormatUserEntryText(const QString& username, Service::Account::UUID uuid) { |
||||
|
return QtProfileSelectionDialog::tr( |
||||
|
"%1\n%2", "%1 is the profile username, %2 is the formatted UUID (e.g. " |
||||
|
"00112233-4455-6677-8899-AABBCCDDEEFF))") |
||||
|
.arg(username, QString::fromStdString(uuid.FormatSwitch())); |
||||
|
} |
||||
|
|
||||
|
QString GetImagePath(Service::Account::UUID uuid) { |
||||
|
const auto path = FileUtil::GetUserPath(FileUtil::UserPath::NANDDir) + |
||||
|
"/system/save/8000000000000010/su/avators/" + uuid.FormatSwitch() + ".jpg"; |
||||
|
return QString::fromStdString(path); |
||||
|
} |
||||
|
|
||||
|
QPixmap GetIcon(Service::Account::UUID uuid) { |
||||
|
QPixmap icon{GetImagePath(uuid)}; |
||||
|
|
||||
|
if (!icon) { |
||||
|
icon.fill(Qt::black); |
||||
|
icon.loadFromData(backup_jpeg.data(), static_cast<u32>(backup_jpeg.size())); |
||||
|
} |
||||
|
|
||||
|
return icon.scaled(64, 64, Qt::IgnoreAspectRatio, Qt::SmoothTransformation); |
||||
|
} |
||||
|
|
||||
|
QtProfileSelectionDialog::QtProfileSelectionDialog(QWidget* parent) |
||||
|
: QDialog(parent), profile_manager(std::make_unique<Service::Account::ProfileManager>()) { |
||||
|
outer_layout = new QVBoxLayout; |
||||
|
|
||||
|
instruction_label = new QLabel(tr("Select a user:")); |
||||
|
|
||||
|
scroll_area = new QScrollArea; |
||||
|
|
||||
|
buttons = new QDialogButtonBox; |
||||
|
buttons->addButton(tr("Cancel"), QDialogButtonBox::RejectRole); |
||||
|
buttons->addButton(tr("OK"), QDialogButtonBox::AcceptRole); |
||||
|
|
||||
|
connect(buttons, &QDialogButtonBox::accepted, this, &QtProfileSelectionDialog::accept); |
||||
|
connect(buttons, &QDialogButtonBox::rejected, this, &QtProfileSelectionDialog::reject); |
||||
|
|
||||
|
outer_layout->addWidget(instruction_label); |
||||
|
outer_layout->addWidget(scroll_area); |
||||
|
outer_layout->addWidget(buttons); |
||||
|
|
||||
|
layout = new QVBoxLayout; |
||||
|
tree_view = new QTreeView; |
||||
|
item_model = new QStandardItemModel(tree_view); |
||||
|
tree_view->setModel(item_model); |
||||
|
|
||||
|
tree_view->setAlternatingRowColors(true); |
||||
|
tree_view->setSelectionMode(QHeaderView::SingleSelection); |
||||
|
tree_view->setSelectionBehavior(QHeaderView::SelectRows); |
||||
|
tree_view->setVerticalScrollMode(QHeaderView::ScrollPerPixel); |
||||
|
tree_view->setHorizontalScrollMode(QHeaderView::ScrollPerPixel); |
||||
|
tree_view->setSortingEnabled(true); |
||||
|
tree_view->setEditTriggers(QHeaderView::NoEditTriggers); |
||||
|
tree_view->setUniformRowHeights(true); |
||||
|
tree_view->setIconSize({64, 64}); |
||||
|
tree_view->setContextMenuPolicy(Qt::NoContextMenu); |
||||
|
|
||||
|
item_model->insertColumns(0, 1); |
||||
|
item_model->setHeaderData(0, Qt::Horizontal, "Users"); |
||||
|
|
||||
|
// We must register all custom types with the Qt Automoc system so that we are able to use it
|
||||
|
// with signals/slots. In this case, QList falls under the umbrells of custom types.
|
||||
|
qRegisterMetaType<QList<QStandardItem*>>("QList<QStandardItem*>"); |
||||
|
|
||||
|
layout->setContentsMargins(0, 0, 0, 0); |
||||
|
layout->setSpacing(0); |
||||
|
layout->addWidget(tree_view); |
||||
|
|
||||
|
scroll_area->setLayout(layout); |
||||
|
|
||||
|
connect(tree_view, &QTreeView::clicked, this, &QtProfileSelectionDialog::SelectUser); |
||||
|
|
||||
|
const auto& profiles = profile_manager->GetAllUsers(); |
||||
|
for (const auto& user : profiles) { |
||||
|
Service::Account::ProfileBase profile; |
||||
|
if (!profile_manager->GetProfileBase(user, profile)) |
||||
|
continue; |
||||
|
|
||||
|
const auto username = Common::StringFromFixedZeroTerminatedBuffer( |
||||
|
reinterpret_cast<const char*>(profile.username.data()), profile.username.size()); |
||||
|
|
||||
|
list_items.push_back(QList<QStandardItem*>{new QStandardItem{ |
||||
|
GetIcon(user), FormatUserEntryText(QString::fromStdString(username), user)}}); |
||||
|
} |
||||
|
|
||||
|
for (const auto& item : list_items) |
||||
|
item_model->appendRow(item); |
||||
|
|
||||
|
setLayout(outer_layout); |
||||
|
setWindowTitle(tr("Profile Selector")); |
||||
|
resize(550, 400); |
||||
|
} |
||||
|
|
||||
|
QtProfileSelectionDialog::~QtProfileSelectionDialog() = default; |
||||
|
|
||||
|
void QtProfileSelectionDialog::accept() { |
||||
|
ok = true; |
||||
|
QDialog::accept(); |
||||
|
} |
||||
|
|
||||
|
void QtProfileSelectionDialog::reject() { |
||||
|
ok = false; |
||||
|
user_index = 0; |
||||
|
QDialog::reject(); |
||||
|
} |
||||
|
|
||||
|
bool QtProfileSelectionDialog::GetStatus() const { |
||||
|
return ok; |
||||
|
} |
||||
|
|
||||
|
u32 QtProfileSelectionDialog::GetIndex() const { |
||||
|
return user_index; |
||||
|
} |
||||
|
|
||||
|
void QtProfileSelectionDialog::SelectUser(const QModelIndex& index) { |
||||
|
user_index = index.row(); |
||||
|
} |
||||
|
|
||||
|
QtProfileSelector::QtProfileSelector(GMainWindow& parent) { |
||||
|
connect(this, &QtProfileSelector::MainWindowSelectProfile, &parent, |
||||
|
&GMainWindow::ProfileSelectorSelectProfile, Qt::QueuedConnection); |
||||
|
connect(&parent, &GMainWindow::ProfileSelectorFinishedSelection, this, |
||||
|
&QtProfileSelector::MainWindowFinishedSelection, Qt::DirectConnection); |
||||
|
} |
||||
|
|
||||
|
QtProfileSelector::~QtProfileSelector() = default; |
||||
|
|
||||
|
void QtProfileSelector::SelectProfile( |
||||
|
std::function<void(std::optional<Service::Account::UUID>)> callback) const { |
||||
|
this->callback = std::move(callback); |
||||
|
emit MainWindowSelectProfile(); |
||||
|
} |
||||
|
|
||||
|
void QtProfileSelector::MainWindowFinishedSelection(std::optional<Service::Account::UUID> uuid) { |
||||
|
// Acquire the HLE mutex
|
||||
|
std::lock_guard<std::recursive_mutex> lock(HLE::g_hle_lock); |
||||
|
callback(uuid); |
||||
|
} |
||||
@ -0,0 +1,73 @@ |
|||||
|
// Copyright 2018 yuzu Emulator Project |
||||
|
// Licensed under GPLv2 or any later version |
||||
|
// Refer to the license.txt file included. |
||||
|
|
||||
|
#pragma once |
||||
|
|
||||
|
#include <vector> |
||||
|
#include <QDialog> |
||||
|
#include <QList> |
||||
|
#include "core/frontend/applets/profile_select.h" |
||||
|
|
||||
|
class GMainWindow; |
||||
|
class QDialogButtonBox; |
||||
|
class QGraphicsScene; |
||||
|
class QLabel; |
||||
|
class QScrollArea; |
||||
|
class QStandardItem; |
||||
|
class QStandardItemModel; |
||||
|
class QTreeView; |
||||
|
class QVBoxLayout; |
||||
|
|
||||
|
class QtProfileSelectionDialog final : public QDialog { |
||||
|
Q_OBJECT |
||||
|
|
||||
|
public: |
||||
|
explicit QtProfileSelectionDialog(QWidget* parent); |
||||
|
~QtProfileSelectionDialog() override; |
||||
|
|
||||
|
void accept() override; |
||||
|
void reject() override; |
||||
|
|
||||
|
bool GetStatus() const; |
||||
|
u32 GetIndex() const; |
||||
|
|
||||
|
private: |
||||
|
bool ok = false; |
||||
|
u32 user_index = 0; |
||||
|
|
||||
|
void SelectUser(const QModelIndex& index); |
||||
|
|
||||
|
QVBoxLayout* layout; |
||||
|
QTreeView* tree_view; |
||||
|
QStandardItemModel* item_model; |
||||
|
QGraphicsScene* scene; |
||||
|
|
||||
|
std::vector<QList<QStandardItem*>> list_items; |
||||
|
|
||||
|
QVBoxLayout* outer_layout; |
||||
|
QLabel* instruction_label; |
||||
|
QScrollArea* scroll_area; |
||||
|
QDialogButtonBox* buttons; |
||||
|
|
||||
|
std::unique_ptr<Service::Account::ProfileManager> profile_manager; |
||||
|
}; |
||||
|
|
||||
|
class QtProfileSelector final : public QObject, public Core::Frontend::ProfileSelectApplet { |
||||
|
Q_OBJECT |
||||
|
|
||||
|
public: |
||||
|
explicit QtProfileSelector(GMainWindow& parent); |
||||
|
~QtProfileSelector() override; |
||||
|
|
||||
|
void SelectProfile( |
||||
|
std::function<void(std::optional<Service::Account::UUID>)> callback) const override; |
||||
|
|
||||
|
signals: |
||||
|
void MainWindowSelectProfile() const; |
||||
|
|
||||
|
private: |
||||
|
void MainWindowFinishedSelection(std::optional<Service::Account::UUID> uuid); |
||||
|
|
||||
|
mutable std::function<void(std::optional<Service::Account::UUID>)> callback; |
||||
|
}; |
||||
Write
Preview
Loading…
Cancel
Save
Reference in new issue