2023-12-16 17:36:57 +01:00
|
|
|
#include "ConfigUtils.h"
|
|
|
|
#include "ConfigRenderer.h"
|
|
|
|
#include "config/WUPSConfigAPI.h"
|
|
|
|
#include "hooks.h"
|
2024-08-04 20:59:07 +02:00
|
|
|
#include "plugin/PluginLoadWrapper.h"
|
2024-11-27 20:44:36 +01:00
|
|
|
#include "utils/DrawUtils.h"
|
2024-08-04 20:59:07 +02:00
|
|
|
#include "utils/WUPSBackendSettings.h"
|
2024-11-27 20:44:36 +01:00
|
|
|
#include "utils/dc.h"
|
2023-12-16 17:36:57 +01:00
|
|
|
#include "utils/input/CombinedInput.h"
|
2024-11-27 20:44:36 +01:00
|
|
|
#include "utils/input/Input.h"
|
2023-12-16 17:36:57 +01:00
|
|
|
#include "utils/input/VPADInput.h"
|
|
|
|
#include "utils/input/WPADInput.h"
|
2024-11-27 20:44:36 +01:00
|
|
|
#include "utils/logger.h"
|
|
|
|
#include "utils/utils.h"
|
|
|
|
#include "version.h"
|
2023-12-16 17:36:57 +01:00
|
|
|
|
2024-11-27 20:44:36 +01:00
|
|
|
#include <algorithm>
|
2024-08-04 20:59:07 +02:00
|
|
|
#include <coreinit/title.h>
|
2024-11-27 20:44:36 +01:00
|
|
|
#include <globals.h>
|
|
|
|
#include <memory>
|
2024-08-04 20:59:07 +02:00
|
|
|
#include <sysapp/launch.h>
|
2023-12-16 17:36:57 +01:00
|
|
|
#include <vector>
|
2024-11-27 20:44:36 +01:00
|
|
|
#include <wups/config.h>
|
2023-12-16 17:36:57 +01:00
|
|
|
|
2024-11-27 20:44:36 +01:00
|
|
|
WUPS_CONFIG_SIMPLE_INPUT ConfigUtils::convertInputs(const uint32_t buttons) {
|
2023-12-16 17:36:57 +01:00
|
|
|
WUPSConfigButtons pressedButtons = WUPS_CONFIG_BUTTON_NONE;
|
|
|
|
if (buttons & Input::eButtons::BUTTON_A) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_A;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_LEFT) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_LEFT;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_RIGHT) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_RIGHT;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_L) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_L;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_R) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_R;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_ZL) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_ZL;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_ZR) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_ZR;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_X) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_X;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_Y) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_Y;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_STICK_L) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_STICK_L;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_STICK_R) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_STICK_R;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_PLUS) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_PLUS;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_MINUS) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_MINUS;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_B) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_B;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_UP) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_UP;
|
|
|
|
}
|
|
|
|
if (buttons & Input::eButtons::BUTTON_DOWN) {
|
|
|
|
pressedButtons |= WUPS_CONFIG_BUTTON_DOWN;
|
|
|
|
}
|
2024-11-27 20:44:36 +01:00
|
|
|
return static_cast<WUPS_CONFIG_SIMPLE_INPUT>(pressedButtons);
|
2023-12-16 17:36:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigUtils::displayMenu() {
|
|
|
|
renderBasicScreen("Loading configs...");
|
|
|
|
|
|
|
|
std::vector<ConfigDisplayItem> configs;
|
2024-03-24 07:40:58 +01:00
|
|
|
for (const auto &plugin : gLoadedPlugins) {
|
2023-12-16 17:36:57 +01:00
|
|
|
GeneralConfigInformation info;
|
2024-08-04 15:07:30 +02:00
|
|
|
info.name = plugin.getMetaInformation().getName();
|
|
|
|
info.author = plugin.getMetaInformation().getAuthor();
|
|
|
|
info.version = plugin.getMetaInformation().getVersion();
|
|
|
|
info.pluginData = plugin.getPluginDataCopy();
|
2023-12-16 17:36:57 +01:00
|
|
|
|
|
|
|
std::unique_ptr<WUPSConfigAPIBackend::WUPSConfig> config;
|
2024-08-04 15:41:10 +02:00
|
|
|
|
|
|
|
if (plugin.isLinkedAndLoaded()) {
|
|
|
|
if (const auto configData = plugin.getConfigData()) {
|
|
|
|
if (const auto configHandleOpt = configData->createConfig()) {
|
|
|
|
WUPSConfigAPIStatus callbackResult = configData->CallMenuOpenedCallback(configHandleOpt.value());
|
|
|
|
config = WUPSConfigAPIBackend::Intern::PopConfigByHandle(configHandleOpt.value());
|
|
|
|
if (!config) {
|
|
|
|
DEBUG_FUNCTION_LINE_ERR("Failed to get config for handle: %08X", configHandleOpt.value().handle);
|
|
|
|
} else if (callbackResult != WUPSCONFIG_API_RESULT_SUCCESS) {
|
|
|
|
DEBUG_FUNCTION_LINE_ERR("Callback failed for %s: %s", info.name.c_str(), WUPSConfigAPI_GetStatusStr(callbackResult));
|
|
|
|
config.reset();
|
|
|
|
} else {
|
|
|
|
info.name = config->getName();
|
|
|
|
}
|
2024-03-02 10:02:27 +01:00
|
|
|
} else {
|
2024-08-04 15:41:10 +02:00
|
|
|
DEBUG_FUNCTION_LINE_ERR("Failed to create config for plugin: \"%s\"", info.name.c_str());
|
2023-12-16 17:36:57 +01:00
|
|
|
}
|
|
|
|
} else {
|
2024-08-04 15:41:10 +02:00
|
|
|
for (const auto &hook : plugin.getPluginLinkInformation().getHookDataList()) {
|
|
|
|
if (hook.getType() == WUPS_LOADER_HOOK_GET_CONFIG_DEPRECATED) {
|
|
|
|
if (hook.getFunctionPointer() == nullptr) {
|
|
|
|
DEBUG_FUNCTION_LINE_ERR("Hook had invalid ptr");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
auto cur_config_handle = ((void *(*) ())((uint32_t *) hook.getFunctionPointer()))();
|
|
|
|
if (cur_config_handle == nullptr) {
|
|
|
|
DEBUG_FUNCTION_LINE_WARN("Hook returned empty handle");
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
config = WUPSConfigAPIBackend::Intern::PopConfigByHandle(WUPSConfigHandle(cur_config_handle));
|
|
|
|
if (!config) {
|
|
|
|
DEBUG_FUNCTION_LINE_ERR("Failed to find config for handle: %08X", cur_config_handle);
|
|
|
|
}
|
2023-12-16 17:36:57 +01:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (!config) {
|
2024-03-02 10:02:27 +01:00
|
|
|
config = make_unique_nothrow<WUPSConfigAPIBackend::WUPSConfig>(info.name);
|
2023-12-16 17:36:57 +01:00
|
|
|
}
|
2024-08-04 16:42:24 +02:00
|
|
|
|
2024-08-04 15:07:30 +02:00
|
|
|
configs.emplace_back(info, std::move(config), plugin.isLinkedAndLoaded());
|
2023-12-16 17:36:57 +01:00
|
|
|
}
|
|
|
|
|
2024-05-07 14:22:56 +02:00
|
|
|
// Sort Configs by name
|
2024-11-27 20:44:36 +01:00
|
|
|
std::ranges::sort(configs,
|
|
|
|
[](const ConfigDisplayItem &lhs, const ConfigDisplayItem &rhs) {
|
|
|
|
auto &str1 = lhs.getConfigInformation().name;
|
|
|
|
auto &str2 = rhs.getConfigInformation().name;
|
|
|
|
return std::ranges::lexicographical_compare(str1, str2,
|
|
|
|
[](const char &char1, const char &char2) {
|
|
|
|
return tolower(char1) < tolower(char2);
|
|
|
|
});
|
|
|
|
});
|
2024-05-07 14:22:56 +02:00
|
|
|
|
2023-12-16 17:36:57 +01:00
|
|
|
ConfigRenderer renderer(std::move(configs));
|
|
|
|
configs.clear();
|
|
|
|
|
|
|
|
CombinedInput baseInput;
|
|
|
|
VPadInput vpadInput;
|
2024-05-05 14:46:35 +02:00
|
|
|
WPADInput wpadInputs[7] = {
|
2023-12-16 17:36:57 +01:00
|
|
|
WPAD_CHAN_0,
|
|
|
|
WPAD_CHAN_1,
|
|
|
|
WPAD_CHAN_2,
|
2024-05-05 14:46:35 +02:00
|
|
|
WPAD_CHAN_3,
|
|
|
|
WPAD_CHAN_4,
|
|
|
|
WPAD_CHAN_5,
|
|
|
|
WPAD_CHAN_6,
|
|
|
|
};
|
2023-12-16 17:36:57 +01:00
|
|
|
|
2024-11-27 20:44:36 +01:00
|
|
|
OSTime startTime;
|
2024-03-23 08:29:03 +01:00
|
|
|
bool skipFirstInput = true;
|
2024-05-03 13:00:18 +02:00
|
|
|
|
2024-08-10 09:42:13 +02:00
|
|
|
gOnlyAcceptFromThread = OSGetCurrentThread();
|
|
|
|
ConfigSubState subStateReturnValue = SUB_STATE_ERROR;
|
2023-12-16 17:36:57 +01:00
|
|
|
while (true) {
|
2024-05-10 15:41:40 +02:00
|
|
|
startTime = OSGetTime();
|
2024-05-03 15:08:24 +02:00
|
|
|
if (gConfigMenuShouldClose) {
|
|
|
|
gConfigMenuShouldClose = false;
|
|
|
|
break;
|
|
|
|
}
|
2023-12-16 17:36:57 +01:00
|
|
|
baseInput.reset();
|
|
|
|
if (vpadInput.update(1280, 720)) {
|
|
|
|
baseInput.combine(vpadInput);
|
|
|
|
}
|
|
|
|
for (auto &wpadInput : wpadInputs) {
|
|
|
|
if (wpadInput.update(1280, 720)) {
|
|
|
|
baseInput.combine(wpadInput);
|
|
|
|
}
|
|
|
|
}
|
2024-03-23 08:29:03 +01:00
|
|
|
|
|
|
|
if (skipFirstInput) {
|
|
|
|
skipFirstInput = false;
|
|
|
|
baseInput.lastData = baseInput.data;
|
|
|
|
}
|
|
|
|
|
2023-12-16 17:36:57 +01:00
|
|
|
WUPSConfigSimplePadData simpleData;
|
|
|
|
simpleData.buttons_d = convertInputs(baseInput.data.buttons_d);
|
|
|
|
simpleData.buttons_r = convertInputs(baseInput.data.buttons_r);
|
|
|
|
simpleData.buttons_h = convertInputs(baseInput.data.buttons_h);
|
|
|
|
simpleData.x = baseInput.data.x;
|
|
|
|
simpleData.y = baseInput.data.y;
|
|
|
|
simpleData.touched = baseInput.data.touched;
|
|
|
|
simpleData.validPointer = baseInput.data.validPointer;
|
|
|
|
|
|
|
|
WUPSConfigComplexPadData complexData;
|
|
|
|
complexData.vpad.data = vpadInput.vpad;
|
|
|
|
complexData.vpad.tpCalib = vpadInput.tpCalib;
|
|
|
|
complexData.vpad.vpadError = vpadInput.vpadError;
|
2024-05-05 14:46:35 +02:00
|
|
|
for (int i = 0; i < 7; i++) {
|
2023-12-16 17:36:57 +01:00
|
|
|
complexData.kpad.kpadError[i] = wpadInputs[i].kpadError;
|
|
|
|
complexData.kpad.data[i] = wpadInputs[i].kpad;
|
|
|
|
}
|
|
|
|
|
2024-08-10 09:42:13 +02:00
|
|
|
subStateReturnValue = renderer.Update(baseInput, simpleData, complexData);
|
|
|
|
if (subStateReturnValue != SUB_STATE_RUNNING) {
|
2023-12-16 17:36:57 +01:00
|
|
|
break;
|
|
|
|
}
|
2024-05-10 15:55:10 +02:00
|
|
|
if (renderer.NeedsRedraw() || baseInput.data.buttons_d || baseInput.data.buttons_r) {
|
2024-05-03 11:55:27 +02:00
|
|
|
renderer.Render();
|
|
|
|
}
|
|
|
|
renderer.ResetNeedsRedraw();
|
|
|
|
|
2023-12-16 17:36:57 +01:00
|
|
|
auto diffTime = OSTicksToMicroseconds(OSGetTime() - startTime);
|
|
|
|
if (diffTime < 16000) {
|
|
|
|
OSSleepTicks(OSMicrosecondsToTicks(16000 - diffTime));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-10 15:41:40 +02:00
|
|
|
startTime = OSGetTime();
|
|
|
|
|
2024-08-04 20:59:07 +02:00
|
|
|
std::vector<PluginLoadWrapper> newActivePluginsList;
|
|
|
|
|
2024-08-10 09:42:13 +02:00
|
|
|
if (subStateReturnValue == SUB_STATE_RETURN_WITH_PLUGIN_RELOAD && renderer.GetActivePluginsIfChanged(newActivePluginsList)) {
|
2024-08-04 20:59:07 +02:00
|
|
|
startTime = OSGetTime();
|
|
|
|
renderBasicScreen("Applying changes, app will now restart...");
|
|
|
|
|
|
|
|
// Get list of inactive plugins to save them in the config
|
|
|
|
// Note: this does only consider plugin loaded from the sd card.
|
|
|
|
std::vector<std::string> newInactivePluginsList;
|
|
|
|
for (const auto &cur : newActivePluginsList) {
|
|
|
|
if (!cur.isLoadAndLink()) {
|
|
|
|
auto &source = cur.getPluginData()->getSource();
|
2024-08-10 09:42:13 +02:00
|
|
|
if (source.starts_with(getPluginPath()) && source.ends_with(".wps")) {
|
2024-08-04 20:59:07 +02:00
|
|
|
std::size_t found = source.find_last_of("/\\");
|
|
|
|
std::string filename = source.substr(found + 1);
|
|
|
|
newInactivePluginsList.push_back(filename);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
gLoadOnNextLaunch = newActivePluginsList;
|
|
|
|
WUPSBackendSettings::SetInactivePluginFilenames(newInactivePluginsList);
|
2024-08-10 09:42:13 +02:00
|
|
|
if (!WUPSBackendSettings::SaveSettings()) {
|
|
|
|
DEBUG_FUNCTION_LINE_WARN("Failed to save WUPSBackendSettings");
|
|
|
|
}
|
2024-08-04 20:59:07 +02:00
|
|
|
|
|
|
|
_SYSLaunchTitleWithStdArgsInNoSplash(OSGetTitleID(), nullptr);
|
|
|
|
// Make sure to wait at least 2 seconds so user can read the screen and
|
|
|
|
// are aware the app will restart now.
|
|
|
|
auto diffTime = OSTicksToMilliseconds(OSGetTime() - startTime);
|
|
|
|
if (diffTime < 2000) {
|
|
|
|
OSSleepTicks(OSTicksToMilliseconds(2000 - diffTime));
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
renderBasicScreen("Saving configs...");
|
2024-08-10 09:42:13 +02:00
|
|
|
}
|
|
|
|
for (const auto &plugin : gLoadedPlugins) {
|
|
|
|
const auto configData = plugin.getConfigData();
|
|
|
|
if (configData) {
|
|
|
|
if (configData->CallMenuClosedCallback() == WUPSCONFIG_API_RESULT_MISSING_CALLBACK) {
|
|
|
|
DEBUG_FUNCTION_LINE_WARN("CallMenuClosedCallback is missing for %s", plugin.getMetaInformation().getName().c_str());
|
2024-03-24 07:40:58 +01:00
|
|
|
}
|
2024-08-10 09:42:13 +02:00
|
|
|
} else {
|
|
|
|
CallHook(plugin, WUPS_LOADER_HOOK_CONFIG_CLOSED_DEPRECATED);
|
2023-12-16 17:36:57 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-10 09:42:13 +02:00
|
|
|
|
2023-12-16 17:36:57 +01:00
|
|
|
WUPSConfigAPIBackend::Intern::CleanAllHandles();
|
2024-05-10 15:41:40 +02:00
|
|
|
|
|
|
|
// we want wait at least 300ms to avoid leaking inputs from the config menu to the application
|
|
|
|
auto diffTime = OSTicksToMilliseconds(OSGetTime() - startTime);
|
|
|
|
if (diffTime < 300) {
|
|
|
|
OSSleepTicks(OSMillisecondsToTicks(300 - diffTime));
|
|
|
|
}
|
2023-12-16 17:36:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigUtils::openConfigMenu() {
|
2024-08-09 19:43:11 +02:00
|
|
|
DrawUtils::RenderScreen(&displayMenu);
|
2023-12-16 17:36:57 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void ConfigUtils::renderBasicScreen(std::string_view text) {
|
|
|
|
DrawUtils::beginDraw();
|
|
|
|
DrawUtils::clear(COLOR_BACKGROUND);
|
|
|
|
DrawUtils::setFontColor(COLOR_TEXT);
|
|
|
|
|
|
|
|
// draw top bar
|
|
|
|
DrawUtils::setFontSize(24);
|
|
|
|
DrawUtils::print(16, 6 + 24, "Wii U Plugin System Config Menu");
|
|
|
|
DrawUtils::setFontSize(18);
|
2024-11-27 20:44:36 +01:00
|
|
|
DrawUtils::print(SCREEN_WIDTH - 16, 8 + 24, MODULE_VERSION_FULL, true);
|
2023-12-16 17:36:57 +01:00
|
|
|
DrawUtils::drawRectFilled(8, 8 + 24 + 4, SCREEN_WIDTH - 8 * 2, 3, COLOR_BLACK);
|
|
|
|
|
|
|
|
// draw bottom bar
|
|
|
|
DrawUtils::drawRectFilled(8, SCREEN_HEIGHT - 24 - 8 - 4, SCREEN_WIDTH - 8 * 2, 3, COLOR_BLACK);
|
|
|
|
DrawUtils::setFontSize(18);
|
|
|
|
DrawUtils::print(16, SCREEN_HEIGHT - 10, "\ue07d Navigate ");
|
|
|
|
DrawUtils::print(SCREEN_WIDTH - 16, SCREEN_HEIGHT - 10, "\ue000 Select", true);
|
|
|
|
|
|
|
|
DrawUtils::setFontSize(24);
|
2024-11-27 20:44:36 +01:00
|
|
|
const uint32_t sz = DrawUtils::getTextWidth(text.data());
|
2023-12-16 17:36:57 +01:00
|
|
|
|
|
|
|
DrawUtils::print((SCREEN_WIDTH / 2) - (sz / 2), (SCREEN_HEIGHT / 2), text.data());
|
|
|
|
|
|
|
|
// draw home button
|
|
|
|
DrawUtils::setFontSize(18);
|
2024-11-27 20:44:36 +01:00
|
|
|
const auto exitHint = "\ue044 Exit";
|
2023-12-16 17:36:57 +01:00
|
|
|
DrawUtils::print(SCREEN_WIDTH / 2 + DrawUtils::getTextWidth(exitHint) / 2, SCREEN_HEIGHT - 10, exitHint, true);
|
|
|
|
|
|
|
|
DrawUtils::endDraw();
|
|
|
|
}
|