13617 lines
633 KiB
C++
13617 lines
633 KiB
C++
/********************************************************************************
|
||
* Custom Fork Information
|
||
*
|
||
* File: tesla.hpp
|
||
* Author: ppkantorski
|
||
* Description:
|
||
* This file serves as the core logic for the Ultrahand Overlay project's custom fork
|
||
* of libtesla, an overlay executor. Within this file, you will find a collection of
|
||
* functions, menu structures, and interaction logic designed to facilitate the
|
||
* smooth execution and flexible customization of overlays within the project.
|
||
*
|
||
* For the latest updates and contributions, visit the project's GitHub repository.
|
||
* (GitHub Repository: https://github.com/ppkantorski/Ultrahand-Overlay)
|
||
*
|
||
* Note: Please be aware that this notice cannot be altered or removed. It is a part
|
||
* of the project's documentation and must remain intact.
|
||
*
|
||
* Copyright (c) 2024 ppkantorski
|
||
********************************************************************************/
|
||
|
||
/**
|
||
* Copyright (C) 2020 werwolv
|
||
*
|
||
* This file is part of libtesla.
|
||
*
|
||
* libtesla 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, either version 2 of the License, or
|
||
* (at your option) any later version.
|
||
*
|
||
* libtesla 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 libtesla. If not, see <http://www.gnu.org/licenses/>.
|
||
*/
|
||
|
||
|
||
#pragma once
|
||
|
||
|
||
#include <ultra.hpp>
|
||
#include <switch.h>
|
||
#include <arm_neon.h>
|
||
|
||
#include <strings.h>
|
||
#include <math.h>
|
||
|
||
#include <filesystem> // unused, but preserved for projects that might need it
|
||
#include <algorithm>
|
||
#include <cstring>
|
||
#include <cwctype>
|
||
#include <string>
|
||
#include <functional>
|
||
#include <type_traits>
|
||
#include <mutex>
|
||
#include <shared_mutex>
|
||
#include <memory>
|
||
//#include <chrono> // despite being commented out, it must still be being imported via other libs
|
||
#include <list>
|
||
#include <stack>
|
||
#include <map>
|
||
//#include <barrier>
|
||
|
||
// Define this makro before including tesla.hpp in your main file. If you intend
|
||
// to use the tesla.hpp header in more than one source file, only define it once!
|
||
// #define TESLA_INIT_IMPL
|
||
|
||
#pragma GCC diagnostic push
|
||
#pragma GCC diagnostic ignored "-Wunused-function"
|
||
|
||
#ifdef TESLA_INIT_IMPL
|
||
#define STB_TRUETYPE_IMPLEMENTATION
|
||
#endif
|
||
#include "stb_truetype.h"
|
||
|
||
#pragma GCC diagnostic pop
|
||
|
||
#define ELEMENT_BOUNDS(elem) elem->getX(), elem->getY(), elem->getWidth(), elem->getHeight()
|
||
|
||
#define ASSERT_EXIT(x) if (R_FAILED(x)) std::exit(1)
|
||
#define ASSERT_FATAL(x) if (Result res = x; R_FAILED(res)) fatalThrow(res)
|
||
|
||
#define PACKED __attribute__((packed))
|
||
#define ALWAYS_INLINE inline __attribute__((always_inline))
|
||
|
||
/// Evaluates an expression that returns a result, and returns the result if it would fail.
|
||
#define TSL_R_TRY(resultExpr) \
|
||
({ \
|
||
const auto result = resultExpr; \
|
||
if (R_FAILED(result)) { \
|
||
return result; \
|
||
} \
|
||
})
|
||
|
||
using namespace std::literals::string_literals;
|
||
using namespace std::literals::chrono_literals; // potentially unused, restored for softare compatibility
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
//struct GlyphInfo {
|
||
// u8* pointer;
|
||
// int width;
|
||
// int height;
|
||
//};
|
||
|
||
struct KeyPairHash {
|
||
std::size_t operator()(const std::pair<int, float>& key) const {
|
||
// Combine hashes of both components
|
||
union returnValue {
|
||
char c[8];
|
||
std::size_t s;
|
||
} value;
|
||
memcpy(&value.c[0], &key.first, 4);
|
||
memcpy(&value.c[4], &key.second, 4);
|
||
return value.s;
|
||
}
|
||
};
|
||
|
||
// Custom equality comparison for int-float pairs
|
||
struct KeyPairEqual {
|
||
bool operator()(const std::pair<int, float>& lhs, const std::pair<int, float>& rhs) const {
|
||
//static constexpr float epsilon = 0.00001f;
|
||
return lhs.first == rhs.first &&
|
||
std::abs(lhs.second - rhs.second) < 0.00001f;
|
||
}
|
||
};
|
||
|
||
//std::unordered_map<std::pair<s32, float>, GlyphInfo, KeyPairHash, KeyPairEqual> cache;
|
||
|
||
u8 TeslaFPS = 60;
|
||
//u8 alphabackground = 0xD;
|
||
volatile bool triggerExitNow = false;
|
||
volatile bool isRendering = false;
|
||
volatile bool delayUpdate = false;
|
||
volatile bool pendingExit = false;
|
||
volatile bool wasRendering = false;
|
||
|
||
LEvent renderingStopEvent = {0};
|
||
bool FullMode = true;
|
||
bool deactivateOriginalFooter = false;
|
||
//bool fontCache = true;
|
||
bool disableJumpTo = false;
|
||
|
||
// Check for mini/micro mode flags
|
||
bool isMiniOrMicroMode = false;
|
||
|
||
#endif
|
||
|
||
#if USING_FPS_INDICATOR_DIRECTIVE
|
||
float fps = 0.0;
|
||
int frameCount = 0;
|
||
double elapsedTime;
|
||
#endif
|
||
|
||
|
||
// Custom variables
|
||
//static bool jumpToListItem = false;
|
||
inline std::atomic<bool> jumpToTop{false};
|
||
inline std::atomic<bool> jumpToBottom{false};
|
||
inline std::atomic<bool> skipUp{false};
|
||
inline std::atomic<bool> skipDown{false};
|
||
inline u32 offsetWidthVar = 112;
|
||
inline std::string g_overlayFilename;;
|
||
inline std::string lastOverlayFilename;
|
||
inline std::string lastOverlayMode;
|
||
|
||
inline std::mutex jumpItemMutex;
|
||
inline std::string jumpItemName;
|
||
inline std::string jumpItemValue;
|
||
inline std::atomic<bool> jumpItemExactMatch{true};
|
||
|
||
inline std::atomic<bool> s_onLeftPage{false};
|
||
inline std::atomic<bool> s_onRightPage{false};
|
||
inline bool screenshotsAreDisabled = false;
|
||
|
||
//#if IS_LAUNCHER_DIRECTIVE
|
||
inline bool hideHidden = false;
|
||
//#endif
|
||
|
||
//inline std::atomic<bool> isLaunchingNextOverlay{false};
|
||
inline std::atomic<bool> mainComboHasTriggered{false};
|
||
inline std::atomic<bool> launchComboHasTriggered{false};
|
||
|
||
namespace tsl {
|
||
|
||
// Booleans
|
||
inline std::atomic<bool> clearGlyphCacheNow(false);
|
||
|
||
// Constants
|
||
|
||
namespace cfg {
|
||
|
||
constexpr u32 ScreenWidth = 1920; ///< Width of the Screen
|
||
constexpr u32 ScreenHeight = 1080; ///< Height of the Screen
|
||
constexpr u32 LayerMaxWidth = 1280;
|
||
constexpr u32 LayerMaxHeight = 720;
|
||
|
||
extern u16 LayerWidth; ///< Width of the Tesla layer
|
||
extern u16 LayerHeight; ///< Height of the Tesla layer
|
||
extern u16 LayerPosX; ///< X position of the Tesla layer
|
||
extern u16 LayerPosY; ///< Y position of the Tesla layer
|
||
extern u16 FramebufferWidth; ///< Width of the framebuffer
|
||
extern u16 FramebufferHeight; ///< Height of the framebuffer
|
||
extern u64 launchCombo; ///< Overlay activation key combo
|
||
extern u64 launchCombo2; ///< Overlay activation key combo
|
||
|
||
}
|
||
|
||
/**
|
||
* @brief RGBA4444 Color structure
|
||
*/
|
||
struct Color {
|
||
|
||
union {
|
||
struct {
|
||
u16 r: 4, g: 4, b: 4, a: 4;
|
||
} PACKED;
|
||
u16 rgba;
|
||
};
|
||
|
||
constexpr inline Color(u16 raw): rgba(raw) {}
|
||
constexpr inline Color(u8 r, u8 g, u8 b, u8 a): r(r), g(g), b(b), a(a) {}
|
||
|
||
};
|
||
|
||
//#if USING_WIDGET_DIRECTIVE
|
||
// Ultra-fast version - zero variables, optimized calculations
|
||
inline constexpr Color GradientColor(float temperature) {
|
||
if (temperature <= 35.0f) return Color(7, 7, 15, 0xFF);
|
||
if (temperature >= 65.0f) return Color(15, 0, 0, 0xFF);
|
||
|
||
if (temperature < 45.0f) {
|
||
// Single calculation, avoid repetition
|
||
const float factor = (temperature - 35.0f) * 0.1f;
|
||
return Color(7 - 7 * factor, 7 + 8 * factor, 15 - 15 * factor, 0xFF);
|
||
}
|
||
|
||
if (temperature < 55.0f) {
|
||
return Color(15 * (temperature - 45.0f) * 0.1f, 15, 0, 0xFF);
|
||
}
|
||
|
||
return Color(15, 15 - 15 * (temperature - 55.0f) * 0.1f, 0, 0xFF);
|
||
}
|
||
//#endif
|
||
|
||
|
||
// Ultra-fast version - single variable, minimal branching
|
||
inline Color RGB888(const std::string& hexColor, size_t alpha = 15, const std::string& defaultHexColor = ult::whiteColor) {
|
||
const char* h = hexColor.size() == 6 ? hexColor.data() :
|
||
hexColor.size() == 7 && hexColor[0] == '#' ? hexColor.data() + 1 :
|
||
defaultHexColor.data();
|
||
|
||
return Color(
|
||
(ult::hexMap[h[0]] << 4 | ult::hexMap[h[1]]) >> 4,
|
||
(ult::hexMap[h[2]] << 4 | ult::hexMap[h[3]]) >> 4,
|
||
(ult::hexMap[h[4]] << 4 | ult::hexMap[h[5]]) >> 4,
|
||
alpha
|
||
);
|
||
}
|
||
|
||
|
||
namespace style {
|
||
constexpr u32 ListItemDefaultHeight = 70; ///< Standard list item height
|
||
constexpr u32 MiniListItemDefaultHeight = 40; ///< Mini list item height
|
||
constexpr u32 TrackBarDefaultHeight = 83; ///< Standard track bar height
|
||
constexpr u8 ListItemHighlightSaturation = 7; ///< Maximum saturation of Listitem highlights
|
||
constexpr u8 ListItemHighlightLength = 22; ///< Maximum length of Listitem highlights
|
||
|
||
namespace color {
|
||
constexpr Color ColorFrameBackground = { 0x0, 0x0, 0x0, 0xD }; ///< Overlay frame background color
|
||
constexpr Color ColorTransparent = { 0x0, 0x0, 0x0, 0x0 }; ///< Transparent color
|
||
constexpr Color ColorHighlight = { 0x0, 0xF, 0xD, 0xF }; ///< Greenish highlight color
|
||
constexpr Color ColorFrame = { 0x7, 0x7, 0x7, 0x7 }; ///< Outer boarder color // CUSTOM MODIFICATION
|
||
constexpr Color ColorHandle = { 0x5, 0x5, 0x5, 0xF }; ///< Track bar handle color
|
||
constexpr Color ColorText = { 0xF, 0xF, 0xF, 0xF }; ///< Standard text color
|
||
constexpr Color ColorDescription = { 0xA, 0xA, 0xA, 0xF }; ///< Description text color
|
||
constexpr Color ColorHeaderBar = { 0xC, 0xC, 0xC, 0xF }; ///< Category header rectangle color
|
||
constexpr Color ColorClickAnimation = { 0x0, 0x2, 0x2, 0xF }; ///< Element click animation color
|
||
}
|
||
}
|
||
|
||
static bool overrideBackButton = false; // for properly overriding the automatic "go back" functionality of KEY_B button presses
|
||
|
||
// Theme color variable definitions
|
||
//static bool disableColorfulLogo = false;
|
||
|
||
|
||
static Color logoColor1 = RGB888(ult::whiteColor);
|
||
static Color logoColor2 = RGB888("F7253E");
|
||
|
||
|
||
static size_t defaultBackgroundAlpha = 13;
|
||
|
||
static Color defaultBackgroundColor = RGB888(ult::blackColor, defaultBackgroundAlpha);
|
||
static Color defaultTextColor = RGB888(ult::whiteColor);
|
||
static Color headerTextColor = RGB888(ult::whiteColor);
|
||
static Color headerSeparatorColor = RGB888(ult::whiteColor);
|
||
static Color starColor = RGB888(ult::whiteColor);
|
||
static Color selectionStarColor = RGB888(ult::whiteColor);
|
||
static Color buttonColor = RGB888(ult::whiteColor);
|
||
static Color bottomTextColor = RGB888(ult::whiteColor);
|
||
static Color bottomSeparatorColor = RGB888(ult::whiteColor);
|
||
static Color topSeparatorColor = RGB888("404040");
|
||
|
||
static Color defaultOverlayColor = RGB888(ult::whiteColor);
|
||
static Color defaultPackageColor = RGB888(ult::whiteColor);//RGB888("#00FF00");
|
||
static Color defaultScriptColor = RGB888("FF33FF");
|
||
static Color clockColor = RGB888(ult::whiteColor);
|
||
static Color temperatureColor = RGB888(ult::whiteColor);
|
||
static Color batteryColor = RGB888("ffff45");
|
||
static Color batteryChargingColor = RGB888("00FF00");
|
||
static Color batteryLowColor = RGB888("FF0000");
|
||
static size_t widgetBackdropAlpha = 15;
|
||
static Color widgetBackdropColor = RGB888(ult::blackColor, widgetBackdropAlpha);
|
||
|
||
static Color overlayTextColor = RGB888(ult::whiteColor);
|
||
static Color ultOverlayTextColor = RGB888("9ed0ff");
|
||
static Color packageTextColor = RGB888(ult::whiteColor);
|
||
static Color ultPackageTextColor = RGB888("9ed0ff");
|
||
|
||
static Color bannerVersionTextColor = RGB888("AAAAAA");
|
||
static Color overlayVersionTextColor = RGB888("AAAAAA");
|
||
static Color ultOverlayVersionTextColor = RGB888("00FFDD");
|
||
static Color packageVersionTextColor = RGB888("AAAAAA");
|
||
static Color ultPackageVersionTextColor = RGB888("00FFDD");
|
||
static Color onTextColor = RGB888("00FFDD");
|
||
static Color offTextColor = RGB888("AAAAAA");
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
static Color dynamicLogoRGB1 = RGB888("00E669");
|
||
static Color dynamicLogoRGB2 = RGB888("8080EA");
|
||
#endif
|
||
|
||
//static bool disableSelectionBG = false;
|
||
//static bool disableSelectionValueColor = false;
|
||
static bool invertBGClickColor = false;
|
||
|
||
static size_t selectionBGAlpha = 11;
|
||
static Color selectionBGColor = RGB888(ult::blackColor, selectionBGAlpha);
|
||
|
||
static Color highlightColor1 = RGB888("2288CC");
|
||
static Color highlightColor2 = RGB888("88FFFF");
|
||
static Color highlightColor3 = RGB888("FFFF45");
|
||
static Color highlightColor4 = RGB888("F7253E");
|
||
|
||
static Color highlightColor = tsl::style::color::ColorHighlight;
|
||
|
||
static size_t clickAlpha = 7;
|
||
static Color clickColor = RGB888("3E25F7", clickAlpha);
|
||
|
||
static size_t progressAlpha = 7;
|
||
static Color progressColor = RGB888("253EF7", progressAlpha);
|
||
|
||
static Color trackBarColor = RGB888("555555");
|
||
|
||
static size_t separatorAlpha = 15;
|
||
static Color separatorColor = RGB888("404040", separatorAlpha);
|
||
static Color edgeSeparatorColor = RGB888("303030");
|
||
|
||
static Color textSeparatorColor = RGB888("404040");
|
||
|
||
static Color selectedTextColor = RGB888("9ed0ff");
|
||
static Color selectedValueTextColor = RGB888("FF7777");
|
||
static Color inprogressTextColor = RGB888(ult::whiteColor);
|
||
static Color invalidTextColor = RGB888("FF0000");
|
||
static Color clickTextColor = RGB888(ult::whiteColor);
|
||
|
||
static size_t tableBGAlpha = 14;
|
||
static Color tableBGColor = RGB888("2C2C2C", tableBGAlpha); //RGB888("303030", tableBGAlpha);
|
||
static Color sectionTextColor = RGB888(ult::whiteColor);
|
||
//static Color infoTextColor = RGB888("00FFDD");
|
||
static Color infoTextColor =RGB888("9ed0ff");
|
||
static Color warningTextColor = RGB888("FF7777");
|
||
|
||
static Color healthyRamTextColor = RGB888("00FF00");
|
||
static Color neutralRamTextColor = RGB888("FFAA00");
|
||
static Color badRamTextColor = RGB888("FF0000");
|
||
|
||
static Color trackBarSliderColor = RGB888("606060");
|
||
static Color trackBarSliderBorderColor = RGB888("505050");
|
||
static Color trackBarSliderMalleableColor = RGB888("A0A0A0");
|
||
static Color trackBarFullColor = RGB888("00FFDD");
|
||
static Color trackBarEmptyColor = RGB888("404040");
|
||
|
||
static void initializeThemeVars() { // NOTE: This needs to be called once in your application.
|
||
// Fetch all theme settings at once from the INI file
|
||
auto themeData = ult::getParsedDataFromIniFile(ult::THEME_CONFIG_INI_PATH);
|
||
if (themeData.count(ult::THEME_STR) > 0) {
|
||
auto& themeSection = themeData[ult::THEME_STR];
|
||
|
||
// Fetch and process each theme setting using a helper to simplify fetching and fallback
|
||
auto getValue = [&](const std::string& key) {
|
||
return themeSection.count(key) ? themeSection[key] : ult::defaultThemeSettingsMap[key];
|
||
};
|
||
|
||
// Convert hex color to Color and manage default values and conversion
|
||
auto getColor = [&](const std::string& key, size_t alpha = 15) {
|
||
//std::string hexColor = getValue(key);
|
||
return RGB888(getValue(key), alpha);
|
||
};
|
||
|
||
auto getAlpha = [&](const std::string& key) {
|
||
std::string alphaStr = getValue(key);
|
||
return !alphaStr.empty() ? ult::stoi(alphaStr) : ult::stoi(ult::defaultThemeSettingsMap[key]);
|
||
};
|
||
|
||
//disableColorfulLogo = (getValue("disable_colorful_logo") == ult::TRUE_STR);
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
logoColor1 = getColor("logo_color_1");
|
||
logoColor2 = getColor("logo_color_2");
|
||
#endif
|
||
|
||
defaultBackgroundAlpha = getAlpha("bg_alpha");
|
||
defaultBackgroundColor = getColor("bg_color", defaultBackgroundAlpha);
|
||
defaultTextColor = getColor("text_color");
|
||
headerTextColor = getColor("header_text_color");
|
||
headerSeparatorColor = getColor("header_separator_color");
|
||
starColor = getColor("star_color");
|
||
selectionStarColor = getColor("selection_star_color");
|
||
buttonColor = getColor("bottom_button_color");
|
||
bottomTextColor = getColor("bottom_text_color");
|
||
bottomSeparatorColor = getColor("bottom_separator_color");
|
||
topSeparatorColor = getColor("top_separator_color");
|
||
defaultOverlayColor = getColor("default_overlay_color");
|
||
defaultPackageColor = getColor("default_package_color");
|
||
defaultScriptColor = getColor("default_script_color");
|
||
|
||
clockColor = getColor("clock_color");
|
||
temperatureColor = getColor("temperature_color");
|
||
batteryColor = getColor("battery_color");
|
||
batteryChargingColor = getColor("battery_charging_color");
|
||
batteryLowColor = getColor("battery_low_color");
|
||
widgetBackdropAlpha = getAlpha("widget_backdrop_alpha");
|
||
widgetBackdropColor = getColor("widget_backdrop_color", widgetBackdropAlpha);
|
||
|
||
overlayTextColor = getColor("overlay_text_color");
|
||
ultOverlayTextColor = getColor("ult_overlay_text_color");
|
||
packageTextColor = getColor("package_text_color");
|
||
ultPackageTextColor = getColor("ult_package_text_color");
|
||
|
||
bannerVersionTextColor = getColor("banner_version_text_color");
|
||
overlayVersionTextColor = getColor("overlay_version_text_color");
|
||
ultOverlayVersionTextColor = getColor("ult_overlay_version_text_color");
|
||
packageVersionTextColor =getColor("package_version_text_color");
|
||
ultPackageVersionTextColor = getColor("ult_package_version_text_color");
|
||
onTextColor = getColor("on_text_color");
|
||
offTextColor = getColor("off_text_color");
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
dynamicLogoRGB1 = getColor("dynamic_logo_color_1");
|
||
dynamicLogoRGB2 = getColor("dynamic_logo_color_2");
|
||
#endif
|
||
|
||
//disableSelectionBG = (getValue("disable_selection_bg") == ult::TRUE_STR);
|
||
//disableSelectionValueColor = (getValue("disable_selection_value_color") == ult::TRUE_STR);
|
||
invertBGClickColor = (getValue("invert_bg_click_color") == ult::TRUE_STR);
|
||
|
||
selectionBGAlpha = getAlpha("selection_bg_alpha");
|
||
selectionBGColor = getColor("selection_bg_color", selectionBGAlpha);
|
||
|
||
highlightColor1 = getColor("highlight_color_1");
|
||
highlightColor2 = getColor("highlight_color_2");
|
||
highlightColor3 = getColor("highlight_color_3");
|
||
highlightColor4 = getColor("highlight_color_4");
|
||
|
||
clickAlpha = getAlpha("click_alpha");
|
||
clickColor = getColor("click_color", clickAlpha);
|
||
|
||
progressAlpha = getAlpha("progress_alpha");
|
||
progressColor = getColor("progress_color", progressAlpha);
|
||
|
||
trackBarColor = getColor("trackbar_color");
|
||
|
||
separatorAlpha = getAlpha("separator_alpha");
|
||
separatorColor = getColor("separator_color", separatorAlpha);
|
||
|
||
textSeparatorColor = getColor("text_separator_color");
|
||
|
||
selectedTextColor = getColor("selection_text_color");
|
||
selectedValueTextColor = getColor("selection_value_text_color");
|
||
inprogressTextColor = getColor("inprogress_text_color");
|
||
invalidTextColor = getColor("invalid_text_color");
|
||
clickTextColor = getColor("click_text_color");
|
||
|
||
tableBGAlpha = getAlpha("table_bg_alpha");
|
||
tableBGColor = getColor("table_bg_color", tableBGAlpha);
|
||
sectionTextColor = getColor("table_section_text_color");
|
||
infoTextColor = getColor("table_info_text_color");
|
||
warningTextColor = getColor("warning_text_color");
|
||
|
||
healthyRamTextColor = getColor("healthy_ram_text_color");
|
||
neutralRamTextColor = getColor("neutral_ram_text_color");
|
||
badRamTextColor = getColor("bad_ram_text_color");
|
||
|
||
trackBarSliderColor = getColor("trackbar_slider_color");
|
||
trackBarSliderBorderColor = getColor("trackbar_slider_border_color");
|
||
trackBarSliderMalleableColor = getColor("trackbar_slider_malleable_color");
|
||
trackBarFullColor = getColor("trackbar_full_color");
|
||
trackBarEmptyColor = getColor("trackbar_empty_color");
|
||
}
|
||
}
|
||
|
||
#if !IS_LAUNCHER_DIRECTIVE
|
||
static void initializeUltrahandSettings() { // only needed for regular overlays
|
||
// Load INI data once instead of 4 separate file reads
|
||
auto ultrahandSection = ult::getKeyValuePairsFromSection(ult::ULTRAHAND_CONFIG_INI_PATH, ult::ULTRAHAND_PROJECT_NAME);
|
||
|
||
// Helper lambda to safely get string values
|
||
auto getStringValue = [&](const std::string& key, const std::string& defaultValue = "") -> std::string {
|
||
if (ultrahandSection.count(key) > 0) {
|
||
return ultrahandSection.at(key);
|
||
}
|
||
return defaultValue;
|
||
};
|
||
|
||
// Helper lambda to safely get boolean values
|
||
auto getBoolValue = [&](const std::string& key, bool defaultValue = false) -> bool {
|
||
if (ultrahandSection.count(key) > 0) {
|
||
return (ultrahandSection.at(key) == ult::TRUE_STR);
|
||
}
|
||
return defaultValue;
|
||
};
|
||
|
||
// Get default language with fallback
|
||
std::string defaultLang = getStringValue(ult::DEFAULT_LANG_STR, "en");
|
||
if (defaultLang.empty()) {
|
||
defaultLang = "en";
|
||
}
|
||
|
||
#ifdef UI_OVERRIDE_PATH
|
||
|
||
std::string UI_PATH = UI_OVERRIDE_PATH;
|
||
ult::preprocessPath(UI_PATH);
|
||
const std::string NEW_THEME_CONFIG_INI_PATH = UI_PATH+"theme.ini";
|
||
const std::string NEW_WALLPAPER_PATH = UI_PATH+"wallpaper.rgba";
|
||
|
||
const std::string TRANSLATION_JSON_PATH = UI_PATH+"lang/"+defaultLang+".json";
|
||
if (ult::isFileOrDirectory(NEW_THEME_CONFIG_INI_PATH))
|
||
ult::THEME_CONFIG_INI_PATH = NEW_THEME_CONFIG_INI_PATH; // Override theme path (optional)
|
||
if (ult::isFileOrDirectory(NEW_WALLPAPER_PATH))
|
||
ult::WALLPAPER_PATH = NEW_WALLPAPER_PATH; // Override wallpaper path (optional)
|
||
if (ult::isFileOrDirectory(TRANSLATION_JSON_PATH))
|
||
ult::loadTranslationsFromJSON(TRANSLATION_JSON_PATH); // load translations (optional)
|
||
#endif
|
||
|
||
// Set Ultrahand Globals using loaded section (defaults match initialization function)
|
||
ult::useLaunchCombos = getBoolValue("launch_combos", true); // TRUE_STR default
|
||
ult::useNotifications = getBoolValue("notifications", true); // TRUE_STR default
|
||
if (ult::useNotifications && !ult::isFile(ult::NOTIFICATIONS_FLAG_FILEPATH)) {
|
||
FILE* file = std::fopen((ult::NOTIFICATIONS_FLAG_FILEPATH).c_str(), "w");
|
||
if (file) {
|
||
std::fclose(file);
|
||
}
|
||
} else {
|
||
ult::deleteFileOrDirectory(ult::NOTIFICATIONS_FLAG_FILEPATH);
|
||
}
|
||
|
||
ult::useSwipeToOpen = getBoolValue("swipe_to_open", true); // TRUE_STR default
|
||
ult::useOpaqueScreenshots = getBoolValue("opaque_screenshots", true); // TRUE_STR default
|
||
|
||
ultrahandSection.clear();
|
||
|
||
const std::string langFile = ult::LANG_PATH+defaultLang+".json";
|
||
if (ult::isFileOrDirectory(langFile))
|
||
ult::parseLanguage(langFile);
|
||
}
|
||
#endif
|
||
|
||
|
||
// Declarations
|
||
|
||
/**
|
||
* @brief Direction in which focus moved before landing on
|
||
* the currently focused element
|
||
*/
|
||
enum class FocusDirection {
|
||
None, ///< Focus was placed on the element programatically without user input
|
||
Up, ///< Focus moved upwards
|
||
Down, ///< Focus moved downwards
|
||
Left, ///< Focus moved from left to rigth
|
||
Right ///< Focus moved from right to left
|
||
};
|
||
|
||
/**
|
||
* @brief Current input controll mode
|
||
*
|
||
*/
|
||
enum class InputMode {
|
||
Controller, ///< Input from controller
|
||
Touch, ///< Touch input
|
||
TouchScroll ///< Moving/scrolling touch input
|
||
};
|
||
|
||
class Overlay;
|
||
namespace elm { class Element; }
|
||
|
||
namespace impl {
|
||
|
||
/**
|
||
* @brief Overlay launch parameters
|
||
*/
|
||
enum class LaunchFlags : u8 {
|
||
None = 0, ///< Do nothing special at launch
|
||
CloseOnExit = BIT(0) ///< Close the overlay the last Gui gets poped from the stack
|
||
};
|
||
|
||
static constexpr LaunchFlags operator|(LaunchFlags lhs, LaunchFlags rhs) {
|
||
return static_cast<LaunchFlags>(u8(lhs) | u8(rhs));
|
||
}
|
||
|
||
|
||
|
||
}
|
||
|
||
static void goBack(u32 count = 1);
|
||
|
||
static void pop(u32 count = 1);
|
||
|
||
static void setNextOverlay(const std::string& ovlPath, std::string args = "");
|
||
|
||
template<typename TOverlay, impl::LaunchFlags launchFlags = impl::LaunchFlags::CloseOnExit>
|
||
int loop(int argc, char** argv);
|
||
|
||
// Helpers
|
||
|
||
namespace hlp {
|
||
|
||
/**
|
||
* @brief Wrapper for service initialization
|
||
*
|
||
* @param f wrapped function
|
||
*/
|
||
template<typename F>
|
||
static inline void doWithSmSession(F f) {
|
||
smInitialize();
|
||
f();
|
||
smExit();
|
||
}
|
||
|
||
/**
|
||
* @brief Wrapper for sd card access using stdio
|
||
* @note Consider using raw fs calls instead as they are faster and need less space
|
||
*
|
||
* @param f wrapped function
|
||
*/
|
||
template<typename F>
|
||
static inline void doWithSDCardHandle(F f) {
|
||
fsdevMountSdmc();
|
||
f();
|
||
fsdevUnmountDevice("sdmc");
|
||
}
|
||
|
||
/**
|
||
* @brief Guard that will execute a passed function at the end of the current scope
|
||
*
|
||
* @param f wrapped function
|
||
*/
|
||
template<typename F>
|
||
class ScopeGuard {
|
||
ScopeGuard(const ScopeGuard&) = delete;
|
||
ScopeGuard& operator=(const ScopeGuard&) = delete;
|
||
private:
|
||
F f;
|
||
bool canceled = false;
|
||
public:
|
||
ALWAYS_INLINE ScopeGuard(F f) : f(std::move(f)) { }
|
||
ALWAYS_INLINE ~ScopeGuard() { if (!canceled) { f(); } }
|
||
void dismiss() { canceled = true; }
|
||
};
|
||
|
||
/**
|
||
* @brief libnx hid:sys shim that gives or takes away frocus to or from the process with the given aruid
|
||
*
|
||
* @param enable Give focus or take focus
|
||
* @param aruid Aruid of the process to focus/unfocus
|
||
* @return Result Result
|
||
*/
|
||
static Result hidsysEnableAppletToGetInput(bool enable, u64 aruid) {
|
||
const struct {
|
||
u8 permitInput;
|
||
u64 appletResourceUserId;
|
||
} in = { enable != 0, aruid };
|
||
|
||
return serviceDispatchIn(hidsysGetServiceSession(), 503, in);
|
||
}
|
||
|
||
static Result viAddToLayerStack(ViLayer *layer, ViLayerStack stack) {
|
||
const struct {
|
||
u32 stack;
|
||
u64 layerId;
|
||
} in = { stack, layer->layer_id };
|
||
|
||
return serviceDispatchIn(viGetSession_IManagerDisplayService(), 6000, in);
|
||
}
|
||
|
||
/**
|
||
* @brief Remove layer from layer stack
|
||
*/
|
||
static Result viRemoveFromLayerStack(ViLayer *layer, ViLayerStack stack) {
|
||
const struct {
|
||
u32 stack;
|
||
u64 layerId;
|
||
} in = { stack, layer->layer_id };
|
||
|
||
// Service command 6001 is commonly used for remove operations
|
||
// If this doesn't work, try 6002, 6010, or other nearby values
|
||
return serviceDispatchIn(viGetSession_IManagerDisplayService(), 6001, in);
|
||
}
|
||
|
||
/**
|
||
* @brief Toggles focus between the Tesla overlay and the rest of the system
|
||
*
|
||
* @param enabled Focus Tesla?
|
||
*/
|
||
static void requestForeground(bool enabled, bool updateGlobalFlag = true) {
|
||
if (updateGlobalFlag)
|
||
ult::currentForeground.store(enabled, std::memory_order_release);
|
||
|
||
u64 applicationAruid = 0, appletAruid = 0;
|
||
|
||
for (u64 programId = 0x0100000000001000UL; programId < 0x0100000000001020UL; programId++) {
|
||
pmdmntGetProcessId(&appletAruid, programId);
|
||
|
||
if (appletAruid != 0)
|
||
hidsysEnableAppletToGetInput(!enabled, appletAruid);
|
||
}
|
||
|
||
|
||
pmdmntGetApplicationProcessId(&applicationAruid);
|
||
hidsysEnableAppletToGetInput(!enabled, applicationAruid);
|
||
|
||
hidsysEnableAppletToGetInput(true, 0);
|
||
}
|
||
|
||
|
||
|
||
namespace ini {
|
||
|
||
/**
|
||
* @brief Ini file type
|
||
*/
|
||
using IniData = std::map<std::string, std::map<std::string, std::string>>;
|
||
|
||
/**
|
||
* @brief Parses a ini string
|
||
*
|
||
* @param str String to parse
|
||
* @return Parsed data
|
||
* // Modified to be "const std" instead of just "std"
|
||
*/
|
||
static IniData parseIni(const std::string &str) {
|
||
//IniData iniData;
|
||
//
|
||
//auto lines = split(str, '\n');
|
||
//
|
||
//std::string lastHeader = "";
|
||
//for (auto& line : lines) {
|
||
// line.erase(std::remove_if(line.begin(), line.end(), ::isspace), line.end());
|
||
//
|
||
// if (line[0] == '[' && line[line.size() - 1] == ']') {
|
||
// lastHeader = line.substr(1, line.size() - 2);
|
||
// iniData.emplace(lastHeader, std::map<std::string, std::string>{});
|
||
// }
|
||
// else if (auto keyValuePair = split(line, '='); keyValuePair.size() == 2) {
|
||
// iniData[lastHeader].emplace(keyValuePair[0], keyValuePair[1]);
|
||
// }
|
||
//}
|
||
|
||
return ult::parseIni(str);
|
||
}
|
||
|
||
/**
|
||
* @brief Unparses ini data into a string
|
||
*
|
||
* @param iniData Ini data
|
||
* @return Ini string
|
||
*/
|
||
static std::string unparseIni(const IniData &iniData) {
|
||
std::string result;
|
||
bool addSectionGap = false;
|
||
|
||
for (const auto §ion : iniData) {
|
||
if (addSectionGap) {
|
||
result += '\n';
|
||
}
|
||
result += '[' + section.first + "]\n";
|
||
for (const auto &keyValue : section.second) {
|
||
result += keyValue.first + '=' + keyValue.second + '\n';
|
||
}
|
||
addSectionGap = true;
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Read Tesla settings file
|
||
*
|
||
* @return Settings data
|
||
*/
|
||
static IniData readOverlaySettings(auto& CONFIG_FILE) {
|
||
/* Open Sd card filesystem. */
|
||
FsFileSystem fsSdmc;
|
||
if (R_FAILED(fsOpenSdCardFileSystem(&fsSdmc)))
|
||
return {};
|
||
hlp::ScopeGuard fsGuard([&] { fsFsClose(&fsSdmc); });
|
||
|
||
/* Open config file. */
|
||
FsFile fileConfig;
|
||
if (R_FAILED(fsFsOpenFile(&fsSdmc, CONFIG_FILE, FsOpenMode_Read, &fileConfig)))
|
||
return {};
|
||
hlp::ScopeGuard fileGuard([&] { fsFileClose(&fileConfig); });
|
||
|
||
/* Get config file size. */
|
||
s64 configFileSize;
|
||
if (R_FAILED(fsFileGetSize(&fileConfig, &configFileSize)))
|
||
return {};
|
||
|
||
/* Read and parse config file. */
|
||
std::string configFileData(configFileSize, '\0');
|
||
u64 readSize;
|
||
Result rc = fsFileRead(&fileConfig, 0, configFileData.data(), configFileSize, FsReadOption_None, &readSize);
|
||
if (R_FAILED(rc) || readSize != static_cast<u64>(configFileSize))
|
||
return {};
|
||
|
||
return ult::parseIni(configFileData);
|
||
}
|
||
|
||
/**
|
||
* @brief Replace Tesla settings file with new data
|
||
*
|
||
* @param iniData new data
|
||
*/
|
||
static void writeOverlaySettings(IniData const &iniData, auto& CONFIG_FILE) {
|
||
/* Open Sd card filesystem. */
|
||
FsFileSystem fsSdmc;
|
||
if (R_FAILED(fsOpenSdCardFileSystem(&fsSdmc)))
|
||
return;
|
||
hlp::ScopeGuard fsGuard([&] { fsFsClose(&fsSdmc); });
|
||
|
||
/* Open config file. */
|
||
FsFile fileConfig;
|
||
if (R_FAILED(fsFsOpenFile(&fsSdmc, CONFIG_FILE, FsOpenMode_Write, &fileConfig)))
|
||
return;
|
||
hlp::ScopeGuard fileGuard([&] { fsFileClose(&fileConfig); });
|
||
|
||
const std::string iniString = unparseIni(iniData);
|
||
|
||
fsFileWrite(&fileConfig, 0, iniString.c_str(), iniString.length(), FsWriteOption_Flush);
|
||
}
|
||
|
||
/**
|
||
* @brief Merge and save changes into Tesla settings file
|
||
*
|
||
* @param changes setting values to add or update
|
||
*/
|
||
static void updateOverlaySettings(IniData const &changes, auto& CONFIG_FILE) {
|
||
hlp::ini::IniData iniData = hlp::ini::readOverlaySettings(CONFIG_FILE);
|
||
for (auto §ion : changes) {
|
||
for (auto &keyValue : section.second) {
|
||
iniData[section.first][keyValue.first] = keyValue.second;
|
||
}
|
||
}
|
||
writeOverlaySettings(iniData, CONFIG_FILE);
|
||
}
|
||
|
||
}
|
||
|
||
/**
|
||
* @brief Decodes a key string into it's key code
|
||
*
|
||
* @param value Key string
|
||
* @return Key code
|
||
*/
|
||
static u64 stringToKeyCode(const std::string& value) {
|
||
for (const auto& keyInfo : ult::KEYS_INFO) {
|
||
if (strcasecmp(value.c_str(), keyInfo.name) == 0)
|
||
return keyInfo.key;
|
||
}
|
||
return 0;
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Decodes a combo string into key codes
|
||
*
|
||
* @param value Combo string
|
||
* @return Key codes
|
||
*/
|
||
static u64 comboStringToKeys(const std::string &value) {
|
||
u64 keyCombo = 0x00;
|
||
for (std::string key : ult::split(ult::removeWhiteSpaces(value), '+')) { // CUSTOM MODIFICATION (bug fix)
|
||
keyCombo |= hlp::stringToKeyCode(key);
|
||
}
|
||
return keyCombo;
|
||
}
|
||
|
||
/**
|
||
* @brief Encodes key codes into a combo string
|
||
*
|
||
* @param keys Key codes
|
||
* @return Combo string
|
||
*/
|
||
static std::string keysToComboString(u64 keys) {
|
||
if (keys == 0) return ""; // Early return for empty input
|
||
|
||
std::string result;
|
||
bool first = true;
|
||
|
||
for (const auto &keyInfo : ult::KEYS_INFO) {
|
||
if (keys & keyInfo.key) {
|
||
if (!first) {
|
||
result += "+";
|
||
}
|
||
result += keyInfo.name;
|
||
first = false;
|
||
}
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
// Function to load key combo mappings from both overlays.ini and packages.ini
|
||
static void loadEntryKeyCombos() {
|
||
ult::g_entryCombos.clear();
|
||
|
||
// Load overlay combos from overlays.ini
|
||
auto overlayData = ult::getParsedDataFromIniFile(ult::OVERLAYS_INI_FILEPATH);
|
||
std::string fullPath;
|
||
u64 keys;
|
||
|
||
std::vector<std::string> modeList, comboList;
|
||
for (auto& [fileName, settings] : overlayData) {
|
||
fullPath = ult::OVERLAY_PATH + fileName;
|
||
|
||
// 1) main key_combo
|
||
if (auto it = settings.find(ult::KEY_COMBO_STR); it != settings.end() && !it->second.empty()) {
|
||
keys = hlp::comboStringToKeys(it->second);
|
||
if (keys) ult::g_entryCombos[keys] = { fullPath, "" };
|
||
}
|
||
|
||
// 2) per-mode combos
|
||
auto modesIt = settings.find("mode_args");
|
||
auto argsIt = settings.find("mode_combos");
|
||
if (modesIt != settings.end()) {
|
||
modeList = ult::splitIniList(modesIt->second);
|
||
comboList = (argsIt != settings.end())
|
||
? ult::splitIniList(argsIt->second)
|
||
: std::vector<std::string>();
|
||
if (comboList.size() < modeList.size())
|
||
comboList.resize(modeList.size());
|
||
|
||
for (size_t i = 0; i < modeList.size(); ++i) {
|
||
const std::string& comboStr = comboList[i];
|
||
if (comboStr.empty()) continue;
|
||
keys = hlp::comboStringToKeys(comboStr);
|
||
if (!keys) continue;
|
||
// launchArg is the *mode* (i.e. modeList[i])
|
||
ult::g_entryCombos[keys] = { fullPath, modeList[i] };
|
||
}
|
||
}
|
||
}
|
||
|
||
// Load package combos from packages.ini
|
||
auto packageData = ult::getParsedDataFromIniFile(ult::PACKAGES_INI_FILEPATH);
|
||
for (auto& [packageName, settings] : packageData) {
|
||
// Only handle main key_combo for packages (no modes for packages)
|
||
if (auto it = settings.find(ult::KEY_COMBO_STR); it != settings.end() && !it->second.empty()) {
|
||
keys = hlp::comboStringToKeys(it->second);
|
||
//std::string tmpPackageName = packageName;
|
||
//ult::removeQuotes(packageName);
|
||
if (keys) ult::g_entryCombos[keys] = { ult::OVERLAY_PATH + "ovlmenu.ovl", "--package " + packageName};
|
||
}
|
||
}
|
||
}
|
||
|
||
// Function to check if a key combination matches any overlay key combo
|
||
static OverlayCombo getEntryForKeyCombo(u64 keys) {
|
||
if (auto it = ult::g_entryCombos.find(keys); it != ult::g_entryCombos.end())
|
||
return it->second;
|
||
return { "", "" };
|
||
}
|
||
|
||
}
|
||
|
||
|
||
|
||
// Renderer
|
||
|
||
namespace gfx {
|
||
|
||
extern "C" u64 __nx_vi_layer_id;
|
||
|
||
|
||
struct ScissoringConfig {
|
||
u32 x, y, w, h, x_max, y_max;
|
||
};
|
||
|
||
|
||
// Forward declarations
|
||
class Renderer;
|
||
|
||
|
||
#ifdef UI_OVERRIDE_PATH
|
||
inline static std::shared_mutex s_translationCacheMutex;
|
||
#endif
|
||
class FontManager {
|
||
public:
|
||
struct Glyph {
|
||
stbtt_fontinfo *currFont;
|
||
float currFontSize;
|
||
int bounds[4];
|
||
int xAdvance;
|
||
u8 *glyphBmp;
|
||
int width, height;
|
||
|
||
// Add destructor to ensure cleanup
|
||
~Glyph() {
|
||
if (glyphBmp) {
|
||
stbtt_FreeBitmap(glyphBmp, nullptr);
|
||
glyphBmp = nullptr;
|
||
}
|
||
}
|
||
|
||
// Prevent copying to avoid double-free
|
||
Glyph(const Glyph&) = delete;
|
||
Glyph& operator=(const Glyph&) = delete;
|
||
|
||
// Allow moving
|
||
Glyph(Glyph&& other) noexcept
|
||
: currFont(other.currFont), currFontSize(other.currFontSize)
|
||
, xAdvance(other.xAdvance), glyphBmp(other.glyphBmp)
|
||
, width(other.width), height(other.height) {
|
||
memcpy(bounds, other.bounds, sizeof(bounds));
|
||
other.glyphBmp = nullptr; // Prevent double-free
|
||
}
|
||
|
||
Glyph& operator=(Glyph&& other) noexcept {
|
||
if (this != &other) {
|
||
if (glyphBmp) {
|
||
stbtt_FreeBitmap(glyphBmp, nullptr);
|
||
}
|
||
currFont = other.currFont;
|
||
currFontSize = other.currFontSize;
|
||
xAdvance = other.xAdvance;
|
||
glyphBmp = other.glyphBmp;
|
||
width = other.width;
|
||
height = other.height;
|
||
memcpy(bounds, other.bounds, sizeof(bounds));
|
||
other.glyphBmp = nullptr;
|
||
}
|
||
return *this;
|
||
}
|
||
|
||
Glyph() : currFont(nullptr), currFontSize(0.0f), xAdvance(0),
|
||
glyphBmp(nullptr), width(0), height(0) {
|
||
std::memset(bounds, 0, sizeof(bounds));
|
||
}
|
||
};
|
||
|
||
struct FontMetrics {
|
||
int ascent, descent, lineGap;
|
||
int lineHeight; // ascent - descent + lineGap
|
||
stbtt_fontinfo* font;
|
||
float fontSize;
|
||
|
||
FontMetrics() : ascent(0), descent(0), lineGap(0), lineHeight(0), font(nullptr), fontSize(0.0f) {}
|
||
|
||
FontMetrics(stbtt_fontinfo* f, float size) : font(f), fontSize(size) {
|
||
if (font) {
|
||
stbtt_GetFontVMetrics(font, &ascent, &descent, &lineGap);
|
||
const float scale = stbtt_ScaleForPixelHeight(font, fontSize);
|
||
ascent = static_cast<int>(ascent * scale);
|
||
descent = static_cast<int>(descent * scale);
|
||
lineGap = static_cast<int>(lineGap * scale);
|
||
lineHeight = ascent - descent + lineGap;
|
||
} else {
|
||
ascent = descent = lineGap = lineHeight = 0;
|
||
}
|
||
}
|
||
};
|
||
|
||
enum class CacheType {
|
||
Regular,
|
||
Notification,
|
||
Persistent
|
||
};
|
||
|
||
private:
|
||
inline static std::shared_mutex s_cacheMutex;
|
||
inline static std::mutex s_initMutex;
|
||
|
||
// Existing caches
|
||
inline static std::unordered_map<u64, std::shared_ptr<Glyph>> s_sharedGlyphCache;
|
||
inline static std::unordered_map<u64, std::shared_ptr<Glyph>> s_persistentGlyphCache;
|
||
|
||
// NEW: Notification-specific cache
|
||
inline static std::unordered_map<u64, std::shared_ptr<Glyph>> s_notificationGlyphCache;
|
||
|
||
// Font metrics cache
|
||
inline static std::unordered_map<u64, FontMetrics> s_fontMetricsCache;
|
||
|
||
// Add cache size limits
|
||
static constexpr size_t MAX_CACHE_SIZE = 600;
|
||
static constexpr size_t CLEANUP_THRESHOLD = 500;
|
||
static constexpr size_t MAX_NOTIFICATION_CACHE_SIZE = 200; // Separate limit for notifications
|
||
|
||
// font handles & state
|
||
inline static stbtt_fontinfo* s_stdFont = nullptr;
|
||
inline static stbtt_fontinfo* s_localFont = nullptr;
|
||
inline static stbtt_fontinfo* s_extFont = nullptr;
|
||
inline static bool s_hasLocalFont = false;
|
||
inline static bool s_initialized = false;
|
||
|
||
// Fix cache key generation to prevent collisions
|
||
static u64 generateCacheKey(u32 character, bool monospace, u32 fontSize) {
|
||
// Use more bits for fontSize and separate monospace bit
|
||
u64 key = static_cast<u64>(character);
|
||
key = (key << 32) | static_cast<u64>(fontSize);
|
||
if (monospace) {
|
||
key |= (1ULL << 63); // Use the highest bit for monospace
|
||
}
|
||
return key;
|
||
}
|
||
|
||
// Generate cache key for font metrics
|
||
static u64 generateFontMetricsCacheKey(stbtt_fontinfo* font, u32 fontSize) {
|
||
// Use pointer address as font identifier and fontSize
|
||
const u64 fontKey = reinterpret_cast<uintptr_t>(font);
|
||
return (fontKey << 32) | static_cast<u64>(fontSize);
|
||
}
|
||
|
||
// Cleanup old entries when cache gets too large
|
||
static void cleanupOldEntries() {
|
||
if (s_sharedGlyphCache.size() <= CLEANUP_THRESHOLD) return;
|
||
|
||
// Simple cleanup: remove oldest entries
|
||
// In a real implementation, you might want LRU or other strategies
|
||
const size_t toRemove = s_sharedGlyphCache.size() - CLEANUP_THRESHOLD;
|
||
auto it = s_sharedGlyphCache.begin();
|
||
for (size_t i = 0; i < toRemove && it != s_sharedGlyphCache.end(); ++i) {
|
||
it = s_sharedGlyphCache.erase(it);
|
||
}
|
||
}
|
||
|
||
// NEW: Cleanup notification cache when it gets too large
|
||
static void cleanupNotificationCache() {
|
||
if (s_notificationGlyphCache.size() <= MAX_NOTIFICATION_CACHE_SIZE) return;
|
||
|
||
const size_t toRemove = s_notificationGlyphCache.size() - (MAX_NOTIFICATION_CACHE_SIZE / 2);
|
||
auto it = s_notificationGlyphCache.begin();
|
||
for (size_t i = 0; i < toRemove && it != s_notificationGlyphCache.end(); ++i) {
|
||
it = s_notificationGlyphCache.erase(it);
|
||
}
|
||
}
|
||
|
||
// NEW: Internal unified glyph creation method
|
||
static std::shared_ptr<Glyph> getOrCreateGlyphInternal(u32 character, bool monospace, u32 fontSize, CacheType cacheType) {
|
||
const u64 key = generateCacheKey(character, monospace, fontSize);
|
||
|
||
// Select target cache based on type
|
||
std::unordered_map<u64, std::shared_ptr<Glyph>>* targetCache;
|
||
switch (cacheType) {
|
||
case CacheType::Notification:
|
||
targetCache = &s_notificationGlyphCache;
|
||
break;
|
||
case CacheType::Persistent:
|
||
targetCache = &s_persistentGlyphCache;
|
||
break;
|
||
default:
|
||
targetCache = &s_sharedGlyphCache;
|
||
break;
|
||
}
|
||
|
||
// First, try to find in target cache with shared lock
|
||
{
|
||
std::shared_lock<std::shared_mutex> readLock(s_cacheMutex);
|
||
|
||
if (!s_initialized) return nullptr;
|
||
|
||
// Check target cache first
|
||
auto it = targetCache->find(key);
|
||
if (it != targetCache->end()) {
|
||
return it->second;
|
||
}
|
||
|
||
// For notification cache, also check persistent cache (but not regular cache)
|
||
// For regular cache, also check persistent cache (existing behavior)
|
||
if (cacheType != CacheType::Persistent) {
|
||
auto persistentIt = s_persistentGlyphCache.find(key);
|
||
if (persistentIt != s_persistentGlyphCache.end()) {
|
||
return persistentIt->second;
|
||
}
|
||
}
|
||
}
|
||
|
||
// Glyph not found, need to create it with exclusive lock
|
||
std::unique_lock<std::shared_mutex> writeLock(s_cacheMutex);
|
||
|
||
if (!s_initialized) return nullptr;
|
||
|
||
// Double-check pattern for target cache
|
||
auto it = targetCache->find(key);
|
||
if (it != targetCache->end()) {
|
||
return it->second;
|
||
}
|
||
|
||
// Double-check persistent cache
|
||
if (cacheType != CacheType::Persistent) {
|
||
auto persistentIt = s_persistentGlyphCache.find(key);
|
||
if (persistentIt != s_persistentGlyphCache.end()) {
|
||
return persistentIt->second;
|
||
}
|
||
}
|
||
|
||
// Check cache size and cleanup if needed
|
||
if (cacheType == CacheType::Regular && s_sharedGlyphCache.size() >= MAX_CACHE_SIZE) {
|
||
cleanupOldEntries();
|
||
} else if (cacheType == CacheType::Notification && s_notificationGlyphCache.size() >= MAX_NOTIFICATION_CACHE_SIZE) {
|
||
cleanupNotificationCache();
|
||
}
|
||
|
||
// Create new glyph
|
||
auto glyph = std::make_shared<Glyph>();
|
||
glyph->currFont = selectFontForCharacterUnsafe(character);
|
||
if (!glyph->currFont) {
|
||
return nullptr;
|
||
}
|
||
|
||
glyph->currFontSize = stbtt_ScaleForPixelHeight(glyph->currFont, fontSize);
|
||
|
||
stbtt_GetCodepointBitmapBoxSubpixel(glyph->currFont, character,
|
||
glyph->currFontSize, glyph->currFontSize, 0, 0,
|
||
&glyph->bounds[0], &glyph->bounds[1], &glyph->bounds[2], &glyph->bounds[3]);
|
||
|
||
s32 yAdvance = 0;
|
||
stbtt_GetCodepointHMetrics(glyph->currFont, monospace ? 'W' : character,
|
||
&glyph->xAdvance, &yAdvance);
|
||
|
||
glyph->glyphBmp = stbtt_GetCodepointBitmap(glyph->currFont,
|
||
glyph->currFontSize, glyph->currFontSize, character,
|
||
&glyph->width, &glyph->height, nullptr, nullptr);
|
||
|
||
// Store in target cache
|
||
(*targetCache)[key] = glyph;
|
||
|
||
return glyph;
|
||
}
|
||
|
||
public:
|
||
// NEW: Preload and persist specific characters
|
||
static void preloadPersistentGlyphs(const std::string& characters, u32 fontSize, bool monospace = false) {
|
||
std::unique_lock<std::shared_mutex> writeLock(s_cacheMutex);
|
||
|
||
if (!s_initialized) return;
|
||
|
||
// Convert UTF-8 string to UTF-32 codepoints
|
||
#pragma GCC diagnostic push
|
||
#pragma GCC diagnostic ignored "-Wdeprecated-declarations"
|
||
|
||
std::wstring_convert<std::codecvt_utf8<char32_t>, char32_t> converter;
|
||
const std::u32string codepoints = converter.from_bytes(characters);
|
||
|
||
#pragma GCC diagnostic pop
|
||
|
||
s32 yAdvance;
|
||
for (char32_t character : codepoints) {
|
||
const u64 key = generateCacheKey(character, monospace, fontSize);
|
||
|
||
if (s_persistentGlyphCache.find(key) != s_persistentGlyphCache.end()) {
|
||
continue;
|
||
}
|
||
|
||
auto glyph = std::make_shared<Glyph>();
|
||
glyph->currFont = selectFontForCharacterUnsafe(character);
|
||
if (!glyph->currFont) continue;
|
||
|
||
glyph->currFontSize = stbtt_ScaleForPixelHeight(glyph->currFont, fontSize);
|
||
|
||
stbtt_GetCodepointBitmapBoxSubpixel(glyph->currFont, character,
|
||
glyph->currFontSize, glyph->currFontSize, 0, 0,
|
||
&glyph->bounds[0], &glyph->bounds[1], &glyph->bounds[2], &glyph->bounds[3]);
|
||
|
||
yAdvance = 0;
|
||
stbtt_GetCodepointHMetrics(glyph->currFont, monospace ? 'W' : character,
|
||
&glyph->xAdvance, &yAdvance);
|
||
|
||
glyph->glyphBmp = stbtt_GetCodepointBitmap(glyph->currFont,
|
||
glyph->currFontSize, glyph->currFontSize, character,
|
||
&glyph->width, &glyph->height, nullptr, nullptr);
|
||
|
||
s_persistentGlyphCache[key] = glyph;
|
||
}
|
||
}
|
||
|
||
|
||
static void initializeFonts(stbtt_fontinfo* stdFont, stbtt_fontinfo* localFont,
|
||
stbtt_fontinfo* extFont, bool hasLocalFont) {
|
||
std::lock_guard<std::mutex> initLock(s_initMutex);
|
||
std::unique_lock<std::shared_mutex> cacheLock(s_cacheMutex);
|
||
|
||
s_stdFont = stdFont;
|
||
s_localFont = localFont;
|
||
s_extFont = extFont;
|
||
s_hasLocalFont = hasLocalFont;
|
||
s_initialized = true;
|
||
}
|
||
|
||
static stbtt_fontinfo* selectFontForCharacter(u32 character) {
|
||
std::shared_lock<std::shared_mutex> lock(s_cacheMutex);
|
||
|
||
if (!s_initialized) return nullptr;
|
||
|
||
if (stbtt_FindGlyphIndex(s_extFont, character)) {
|
||
return s_extFont;
|
||
} else if (s_hasLocalFont && stbtt_FindGlyphIndex(s_localFont, character) != 0) {
|
||
return s_localFont;
|
||
}
|
||
return s_stdFont;
|
||
}
|
||
|
||
// Get font metrics with caching
|
||
static FontMetrics getFontMetrics(stbtt_fontinfo* font, u32 fontSize) {
|
||
if (!font) return FontMetrics();
|
||
|
||
const u64 key = generateFontMetricsCacheKey(font, fontSize);
|
||
|
||
// First, try to find existing metrics with shared lock
|
||
{
|
||
std::shared_lock<std::shared_mutex> readLock(s_cacheMutex);
|
||
auto it = s_fontMetricsCache.find(key);
|
||
if (it != s_fontMetricsCache.end()) {
|
||
return it->second;
|
||
}
|
||
}
|
||
|
||
// Metrics not found, need to create them with exclusive lock
|
||
std::unique_lock<std::shared_mutex> writeLock(s_cacheMutex);
|
||
|
||
// Double-check pattern
|
||
auto it = s_fontMetricsCache.find(key);
|
||
if (it != s_fontMetricsCache.end()) {
|
||
return it->second;
|
||
}
|
||
|
||
// Create new font metrics
|
||
FontMetrics metrics(font, static_cast<float>(fontSize));
|
||
s_fontMetricsCache[key] = metrics;
|
||
|
||
return metrics;
|
||
}
|
||
|
||
// Convenience method to get font metrics for a character (selects appropriate font)
|
||
static FontMetrics getFontMetricsForCharacter(u32 character, u32 fontSize) {
|
||
stbtt_fontinfo* font = selectFontForCharacter(character);
|
||
return getFontMetrics(font, fontSize);
|
||
}
|
||
|
||
// UPDATED: Regular glyph method - now uses internal method
|
||
static std::shared_ptr<Glyph> getOrCreateGlyph(u32 character, bool monospace, u32 fontSize) {
|
||
return getOrCreateGlyphInternal(character, monospace, fontSize, CacheType::Regular);
|
||
}
|
||
|
||
// NEW: Notification-specific glyph method
|
||
static std::shared_ptr<Glyph> getOrCreateNotificationGlyph(u32 character, bool monospace, u32 fontSize) {
|
||
return getOrCreateGlyphInternal(character, monospace, fontSize, CacheType::Notification);
|
||
}
|
||
|
||
// NEW: Clear only the notification cache
|
||
static void clearNotificationCache() {
|
||
std::unique_lock<std::shared_mutex> cacheLock(s_cacheMutex);
|
||
s_notificationGlyphCache.clear();
|
||
s_notificationGlyphCache.rehash(0);
|
||
}
|
||
|
||
static void clearCache() {
|
||
// Note: This is now safe because any code holding a shared_ptr
|
||
// will keep the Glyph alive even after the cache is cleared
|
||
std::unique_lock<std::shared_mutex> cacheLock(s_cacheMutex);
|
||
s_sharedGlyphCache.clear();
|
||
s_sharedGlyphCache.rehash(0);
|
||
s_fontMetricsCache.clear(); // Also clear font metrics cache
|
||
s_fontMetricsCache.rehash(0);
|
||
}
|
||
|
||
static void clearAllCaches() {
|
||
std::unique_lock<std::shared_mutex> cacheLock(s_cacheMutex);
|
||
s_sharedGlyphCache.clear();
|
||
s_sharedGlyphCache.rehash(0);
|
||
s_persistentGlyphCache.clear();
|
||
s_persistentGlyphCache.rehash(0);
|
||
s_notificationGlyphCache.clear();
|
||
s_notificationGlyphCache.rehash(0);
|
||
s_fontMetricsCache.clear();
|
||
s_fontMetricsCache.rehash(0);
|
||
}
|
||
|
||
static void cleanup() {
|
||
std::lock_guard<std::mutex> initLock(s_initMutex);
|
||
std::unique_lock<std::shared_mutex> cacheLock(s_cacheMutex);
|
||
|
||
s_sharedGlyphCache.clear();
|
||
s_sharedGlyphCache.rehash(0);
|
||
s_persistentGlyphCache.clear();
|
||
s_persistentGlyphCache.rehash(0);
|
||
s_notificationGlyphCache.clear();
|
||
s_notificationGlyphCache.rehash(0);
|
||
s_fontMetricsCache.clear();
|
||
s_initialized = false;
|
||
s_stdFont = nullptr;
|
||
s_localFont = nullptr;
|
||
s_extFont = nullptr;
|
||
s_hasLocalFont = false;
|
||
}
|
||
|
||
static size_t getCacheSize() {
|
||
std::shared_lock<std::shared_mutex> lock(s_cacheMutex);
|
||
return s_sharedGlyphCache.size();
|
||
}
|
||
|
||
static size_t getFontMetricsCacheSize() {
|
||
std::shared_lock<std::shared_mutex> lock(s_cacheMutex);
|
||
return s_fontMetricsCache.size();
|
||
}
|
||
|
||
static bool isInitialized() {
|
||
std::shared_lock<std::shared_mutex> lock(s_cacheMutex);
|
||
return s_initialized;
|
||
}
|
||
|
||
static size_t getPersistentCacheSize() {
|
||
std::shared_lock<std::shared_mutex> lock(s_cacheMutex);
|
||
return s_persistentGlyphCache.size();
|
||
}
|
||
|
||
// NEW: Get notification cache size
|
||
static size_t getNotificationCacheSize() {
|
||
std::shared_lock<std::shared_mutex> lock(s_cacheMutex);
|
||
return s_notificationGlyphCache.size();
|
||
}
|
||
|
||
// Add memory usage monitoring
|
||
static size_t getMemoryUsage() {
|
||
std::shared_lock<std::shared_mutex> lock(s_cacheMutex);
|
||
size_t totalMemory = 0;
|
||
|
||
// Regular cache
|
||
for (const auto& pair : s_sharedGlyphCache) {
|
||
const auto& glyph = pair.second;
|
||
if (glyph && glyph->glyphBmp) {
|
||
totalMemory += glyph->width * glyph->height;
|
||
}
|
||
}
|
||
|
||
// Persistent cache
|
||
for (const auto& pair : s_persistentGlyphCache) {
|
||
const auto& glyph = pair.second;
|
||
if (glyph && glyph->glyphBmp) {
|
||
totalMemory += glyph->width * glyph->height;
|
||
}
|
||
}
|
||
|
||
// Notification cache
|
||
for (const auto& pair : s_notificationGlyphCache) {
|
||
const auto& glyph = pair.second;
|
||
if (glyph && glyph->glyphBmp) {
|
||
totalMemory += glyph->width * glyph->height;
|
||
}
|
||
}
|
||
|
||
return totalMemory;
|
||
}
|
||
|
||
private:
|
||
static stbtt_fontinfo* selectFontForCharacterUnsafe(u32 character) {
|
||
if (!s_initialized) return nullptr;
|
||
|
||
if (stbtt_FindGlyphIndex(s_extFont, character)) {
|
||
return s_extFont;
|
||
} else if (s_hasLocalFont && stbtt_FindGlyphIndex(s_localFont, character) != 0) {
|
||
return s_localFont;
|
||
}
|
||
return s_stdFont;
|
||
}
|
||
};
|
||
|
||
// Static member definitions
|
||
//std::shared_mutex FontManager::s_cacheMutex;
|
||
//std::mutex FontManager::s_initMutex;
|
||
//std::unordered_map<u64, std::unique_ptr<FontManager::Glyph>> FontManager::s_sharedGlyphCache;
|
||
//stbtt_fontinfo* FontManager::s_stdFont = nullptr;
|
||
//stbtt_fontinfo* FontManager::s_localFont = nullptr;
|
||
//stbtt_fontinfo* FontManager::s_extFont = nullptr;
|
||
//bool FontManager::s_hasLocalFont = false;
|
||
//bool FontManager::s_initialized = false;
|
||
|
||
// Updated thread-safe calculateStringWidth function
|
||
static float calculateStringWidth(const std::string& originalString, const float fontSize, const bool monospace = false) {
|
||
if (originalString.empty() || !FontManager::isInitialized()) {
|
||
return 0.0f;
|
||
}
|
||
|
||
// Thread-safe translation cache access
|
||
std::string text;
|
||
#ifdef UI_OVERRIDE_PATH
|
||
{
|
||
std::shared_lock<std::shared_mutex> readLock(s_translationCacheMutex);
|
||
auto translatedIt = ult::translationCache.find(originalString);
|
||
if (translatedIt != ult::translationCache.end()) {
|
||
text = translatedIt->second;
|
||
} else {
|
||
// Need to upgrade to write lock
|
||
readLock.unlock();
|
||
std::unique_lock<std::shared_mutex> writeLock(s_translationCacheMutex);
|
||
|
||
// Double-check pattern
|
||
translatedIt = ult::translationCache.find(originalString);
|
||
if (translatedIt != ult::translationCache.end()) {
|
||
text = translatedIt->second;
|
||
} else {
|
||
ult::translationCache[originalString] = originalString;
|
||
text = originalString;
|
||
}
|
||
}
|
||
}
|
||
#else
|
||
text = originalString;
|
||
#endif
|
||
|
||
// CRITICAL: Use the same data types as drawString
|
||
s32 maxWidth = 0;
|
||
s32 currentLineWidth = 0;
|
||
ssize_t codepointWidth;
|
||
u32 currCharacter = 0;
|
||
|
||
// Convert fontSize to u32 to match drawString behavior
|
||
const u32 fontSizeInt = static_cast<u32>(fontSize);
|
||
|
||
auto itStrEnd = text.cend();
|
||
auto itStr = text.cbegin();
|
||
|
||
// Fast ASCII check
|
||
bool isAsciiOnly = true;
|
||
for (unsigned char c : text) {
|
||
if (c > 127) {
|
||
isAsciiOnly = false;
|
||
break;
|
||
}
|
||
}
|
||
|
||
while (itStr != itStrEnd) {
|
||
// Decode UTF-8 codepoint
|
||
if (isAsciiOnly) {
|
||
currCharacter = static_cast<u32>(*itStr);
|
||
codepointWidth = 1;
|
||
} else {
|
||
codepointWidth = decode_utf8(&currCharacter, reinterpret_cast<const u8*>(&(*itStr)));
|
||
if (codepointWidth <= 0) break;
|
||
}
|
||
|
||
itStr += codepointWidth;
|
||
|
||
// Handle newlines
|
||
if (currCharacter == '\n') {
|
||
maxWidth = std::max(currentLineWidth, maxWidth);
|
||
currentLineWidth = 0;
|
||
continue;
|
||
}
|
||
|
||
// Use u32 fontSize to match drawString - now thread-safe
|
||
std::shared_ptr<FontManager::Glyph> glyph = FontManager::getOrCreateGlyph(currCharacter, monospace, fontSizeInt);
|
||
if (!glyph) continue;
|
||
|
||
// CRITICAL: Use the same calculation as drawString
|
||
currentLineWidth += static_cast<s32>(glyph->xAdvance * glyph->currFontSize);
|
||
}
|
||
|
||
// Final width calculation
|
||
maxWidth = std::max(currentLineWidth, maxWidth);
|
||
return static_cast<float>(maxWidth);
|
||
}
|
||
|
||
static std::pair<int, int> getUnderscanPixels();
|
||
|
||
/**
|
||
* @brief Manages the Tesla layer and draws raw data to the screen
|
||
*/
|
||
class Renderer final {
|
||
public:
|
||
|
||
using Glyph = FontManager::Glyph;
|
||
|
||
Renderer& operator=(Renderer&) = delete;
|
||
|
||
friend class tsl::Overlay;
|
||
|
||
/**
|
||
* @brief Gets the renderer instance
|
||
*
|
||
* @return Renderer
|
||
*/
|
||
inline static Renderer& get() {
|
||
static Renderer renderer;
|
||
|
||
return renderer;
|
||
}
|
||
|
||
stbtt_fontinfo m_stdFont, m_localFont, m_extFont;
|
||
bool m_hasLocalFont = false;
|
||
|
||
/**
|
||
* @brief Handles opacity of drawn colors for fadeout. Pass all colors through this function in order to apply opacity properly
|
||
*
|
||
* @param c Original color
|
||
* @return Color with applied opacity
|
||
*/
|
||
static inline Color a(const Color& c) {
|
||
const u8 opacity_limit = static_cast<u8>(0xF * Renderer::s_opacity);
|
||
return (c.rgba & 0x0FFF) | (static_cast<u16>(
|
||
ult::disableTransparency
|
||
? (ult::useOpaqueScreenshots
|
||
? 0xF // fully opaque when both flags on
|
||
: (c.a > 0xE ? c.a : 0xE)) // clamp to 14, keep lower values
|
||
: (c.a < opacity_limit ? c.a : opacity_limit) // normal fade logic
|
||
) << 12);
|
||
}
|
||
|
||
static inline Color aWithOpacity(const Color& c) {
|
||
const u8 opacity_limit = static_cast<u8>(0xF * Renderer::s_opacity);
|
||
return (c.rgba & 0x0FFF) | (static_cast<u16>(
|
||
ult::disableTransparency
|
||
? 0xF // fully opaque when both flags on
|
||
: (c.a < opacity_limit ? c.a : opacity_limit) // normal fade logic
|
||
) << 12);
|
||
}
|
||
|
||
static inline Color a2(const Color& c) {
|
||
const u8 opacity_limit = static_cast<u8>(0xF);
|
||
return (c.rgba & 0x0FFF) | (static_cast<u16>(
|
||
ult::disableTransparency
|
||
? (ult::useOpaqueScreenshots
|
||
? 0xF // fully opaque when both flags on
|
||
: (c.a > 0xE ? c.a : 0xE)) // clamp to 14, keep lower values
|
||
: (c.a < opacity_limit ? c.a : opacity_limit) // normal fade logic
|
||
) << 12);
|
||
}
|
||
|
||
/**
|
||
* @brief Enables scissoring, discarding of any draw outside the given boundaries
|
||
*
|
||
* @param x x pos
|
||
* @param y y pos
|
||
* @param w Width
|
||
* @param h Height
|
||
*/
|
||
inline void enableScissoring(const u32 x, const u32 y, const u32 w, const u32 h) {
|
||
this->m_scissoringStack.emplace(x, y, w, h, x+w, y+h);
|
||
}
|
||
|
||
/**
|
||
* @brief Disables scissoring
|
||
*/
|
||
inline void disableScissoring() {
|
||
this->m_scissoringStack.pop();
|
||
}
|
||
|
||
|
||
// Drawing functions
|
||
|
||
/**
|
||
* @brief Draw a single pixel onto the screen
|
||
*
|
||
* @param x X pos
|
||
* @param y Y pos
|
||
* @param color Color
|
||
*/
|
||
inline void setPixel(const u32 x, const u32 y, const Color& color) {
|
||
const u32 offset = this->getPixelOffset(x, y);
|
||
if (offset != UINT32_MAX) [[likely]] {
|
||
Color* framebuffer = static_cast<Color*>(this->getCurrentFramebuffer());
|
||
framebuffer[offset] = color;
|
||
}
|
||
}
|
||
|
||
inline void setPixelAtOffset(const u32 offset, const Color& color) {
|
||
Color* framebuffer = static_cast<Color*>(this->getCurrentFramebuffer());
|
||
framebuffer[offset] = color;
|
||
}
|
||
|
||
|
||
|
||
/**
|
||
* @brief Blends two colors
|
||
*
|
||
* @param src Source color
|
||
* @param dst Destination color
|
||
* @param alpha Opacity
|
||
* @return Blended color
|
||
*/
|
||
static constexpr u8 inv_alpha_table[16] = {15,14,13,12,11,10,9,8,7,6,5,4,3,2,1,0};
|
||
|
||
inline u8 __attribute__((always_inline)) blendColor(const u8 src, const u8 dst, const u8 alpha) {
|
||
return ((src * inv_alpha_table[alpha]) + (dst * alpha)) >> 4;
|
||
}
|
||
|
||
/**
|
||
* @brief Draws a single source blended pixel onto the screen
|
||
*
|
||
* @param x X pos
|
||
* @param y Y pos
|
||
* @param color Color
|
||
*/
|
||
inline void setPixelBlendSrc(const u32 x, const u32 y, const Color& color) {
|
||
const u32 offset = this->getPixelOffset(x, y);
|
||
if (offset == UINT32_MAX) [[unlikely]]
|
||
return;
|
||
|
||
Color* framebuffer = static_cast<Color*>(this->getCurrentFramebuffer());
|
||
const Color src = framebuffer[offset];
|
||
|
||
// Direct write instead of calling setPixel
|
||
framebuffer[offset] = Color(
|
||
blendColor(src.r, color.r, color.a),
|
||
blendColor(src.g, color.g, color.a),
|
||
blendColor(src.b, color.b, color.a),
|
||
src.a
|
||
);
|
||
}
|
||
|
||
|
||
// Compromise version - keep framebuffer lookup but inline the rest
|
||
inline void setPixelBlendDst(const u32 x, const u32 y, const Color& color) {
|
||
const u32 offset = this->getPixelOffset(x, y);
|
||
if (offset == UINT32_MAX) [[unlikely]]
|
||
return;
|
||
|
||
Color* framebuffer = static_cast<Color*>(this->getCurrentFramebuffer());
|
||
const Color src = framebuffer[offset];
|
||
|
||
// Direct write instead of calling setPixel
|
||
framebuffer[offset] = Color(
|
||
blendColor(src.r, color.r, color.a),
|
||
blendColor(src.g, color.g, color.a),
|
||
blendColor(src.b, color.b, color.a),
|
||
(color.a + (src.a * (0xF - color.a) >> 4))
|
||
);
|
||
}
|
||
|
||
// Batch version for setPixelBlendDst
|
||
inline void setPixelBlendDstBatch(const u32 baseX, const u32 baseY,
|
||
const u8 red[16], const u8 green[16],
|
||
const u8 blue[16], const u8 alpha[16],
|
||
const s32 count) {
|
||
// All variables moved outside the loop
|
||
const u16* framebuffer = static_cast<const u16*>(this->getCurrentFramebuffer());
|
||
u32 offset;
|
||
u8 currentAlpha;
|
||
u8 invAlpha;
|
||
Color src = {0}, end = {0};
|
||
u32 currentX;
|
||
|
||
for (s32 i = 0; i < count; ++i) {
|
||
// Early exit for transparent pixels
|
||
currentAlpha = alpha[i];
|
||
if (currentAlpha == 0)
|
||
continue;
|
||
|
||
currentX = baseX + i;
|
||
offset = this->getPixelOffset(currentX, baseY);
|
||
if (offset == UINT32_MAX) [[unlikely]]
|
||
continue;
|
||
|
||
// Direct framebuffer access and color construction
|
||
src = framebuffer[offset];
|
||
invAlpha = 0xF - currentAlpha;
|
||
|
||
// Direct member assignment instead of constructor
|
||
end.r = blendColor(src.r, red[i], currentAlpha);
|
||
end.g = blendColor(src.g, green[i], currentAlpha);
|
||
end.b = blendColor(src.b, blue[i], currentAlpha);
|
||
end.a = (currentAlpha + (src.a * invAlpha >> 4));
|
||
|
||
this->setPixelAtOffset(offset, end);
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Draws a rectangle of given sizes
|
||
*
|
||
* @param x X pos
|
||
* @param y Y pos
|
||
* @param w Width
|
||
* @param h Height
|
||
* @param color Color
|
||
*/
|
||
inline void drawRect(const s32 x, const s32 y, const s32 w, const s32 h, const Color& color) {
|
||
// Early exit for invalid dimensions
|
||
//if (w <= 0 || h <= 0) return;
|
||
|
||
// Calculate clipped bounds
|
||
const s32 x_start = x < 0 ? 0 : x;
|
||
const s32 y_start = y < 0 ? 0 : y;
|
||
const s32 x_end = (x + w > cfg::FramebufferWidth) ? cfg::FramebufferWidth : x + w;
|
||
const s32 y_end = (y + h > cfg::FramebufferHeight) ? cfg::FramebufferHeight : y + h;
|
||
|
||
// Early exit if completely outside bounds
|
||
if (x_start >= x_end || y_start >= y_end) [[unlikely]] return;
|
||
|
||
|
||
// Draw row by row for better cache locality
|
||
for (s32 yi = y_start; yi < y_end; ++yi) {
|
||
for (s32 xi = x_start; xi < x_end; ++xi) {
|
||
this->setPixelBlendDst(xi, yi, color);
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief Worker function for multithreaded rectangle drawing
|
||
* @param x_start Start X coordinate
|
||
* @param x_end End X coordinate
|
||
* @param y_start Start Y coordinate for this thread
|
||
* @param y_end End Y coordinate for this thread
|
||
* @param color Color to draw
|
||
*/
|
||
inline void processRectChunk(const s32 x_start, const s32 x_end, const s32 y_start, const s32 y_end, const Color& color) {
|
||
for (s32 yi = y_start; yi < y_end; ++yi) {
|
||
for (s32 xi = x_start; xi < x_end; ++xi) {
|
||
this->setPixelBlendDst(xi, yi, color);
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Draws a rectangle of given sizes (Multi-threaded)
|
||
*
|
||
* @param x X pos
|
||
* @param y Y pos
|
||
* @param w Width
|
||
* @param h Height
|
||
* @param color Color
|
||
*/
|
||
inline void drawRectMultiThreaded(const s32 x, const s32 y, const s32 w, const s32 h, const Color& color) {
|
||
// Early exit for invalid dimensions
|
||
if (w <= 0 || h <= 0) return;
|
||
|
||
// Calculate clipped bounds
|
||
const s32 x_start = x < 0 ? 0 : x;
|
||
const s32 y_start = y < 0 ? 0 : y;
|
||
const s32 x_end = (x + w > cfg::FramebufferWidth) ? cfg::FramebufferWidth : x + w;
|
||
const s32 y_end = (y + h > cfg::FramebufferHeight) ? cfg::FramebufferHeight : y + h;
|
||
|
||
// Early exit if completely outside bounds
|
||
if (x_start >= x_end || y_start >= y_end) return;
|
||
|
||
// Calculate visible dimensions
|
||
const s32 visibleHeight = y_end - y_start;
|
||
|
||
// Calculate chunk size - divide rows among threads
|
||
const s32 chunkSize = std::max(1, visibleHeight / static_cast<s32>(ult::numThreads));
|
||
|
||
// Launch threads using ult::renderThreads array
|
||
for (unsigned i = 0; i < static_cast<unsigned>(ult::numThreads); ++i) {
|
||
const s32 startRow = y_start + (i * chunkSize);
|
||
const s32 endRow = (i == static_cast<unsigned>(ult::numThreads) - 1) ?
|
||
y_end :
|
||
std::min(startRow + chunkSize, y_end);
|
||
|
||
// Skip threads that have no work
|
||
if (startRow >= endRow) {
|
||
ult::renderThreads[i] = std::thread([](){}); // Empty thread (still needed for joining)
|
||
continue;
|
||
}
|
||
|
||
// Use member function instead of lambda - much faster
|
||
ult::renderThreads[i] = std::thread(&Renderer::processRectChunk, this,
|
||
x_start, x_end, startRow, endRow, color);
|
||
}
|
||
|
||
// Join all ult::renderThreads
|
||
for (auto& t : ult::renderThreads) {
|
||
t.join();
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Draws a rectangle of given sizes with empty filling
|
||
*
|
||
* @param x X pos
|
||
* @param y Y pos
|
||
* @param w Width
|
||
* @param h Height
|
||
* @param color Color
|
||
*/
|
||
inline void drawEmptyRect(s32 x, s32 y, s32 w, s32 h, Color color) {
|
||
// Only precompute values that are actually reused
|
||
const s32 x_end = x + w - 1;
|
||
const s32 y_end = y + h - 1;
|
||
|
||
// Early exit for completely out-of-bounds rectangles
|
||
if (x_end < 0 || y_end < 0 || x >= cfg::FramebufferWidth || y >= cfg::FramebufferHeight) [[unlikely]] {
|
||
return;
|
||
}
|
||
|
||
// Early exit for degenerate rectangles
|
||
//if (w <= 0 || h <= 0) {
|
||
// return;
|
||
//}
|
||
|
||
// These are reused for both horizontal lines
|
||
const s32 line_x_start = x < 0 ? 0 : x;
|
||
const s32 line_x_end = x_end >= cfg::FramebufferWidth ? cfg::FramebufferWidth - 1 : x_end;
|
||
|
||
// Draw top horizontal line
|
||
if (y >= 0 && y < cfg::FramebufferHeight) {
|
||
for (s32 xi = line_x_start; xi <= line_x_end; ++xi) {
|
||
this->setPixelBlendDst(xi, y, color);
|
||
}
|
||
}
|
||
|
||
// Draw bottom horizontal line (only if different from top)
|
||
if (h > 1 && y_end >= 0 && y_end < cfg::FramebufferHeight) {
|
||
for (s32 xi = line_x_start; xi <= line_x_end; ++xi) {
|
||
this->setPixelBlendDst(xi, y_end, color);
|
||
}
|
||
}
|
||
|
||
// Draw vertical lines only if there's space between horizontal lines
|
||
if (h > 2) {
|
||
// These are reused for both vertical lines
|
||
const s32 line_y_start = (y + 1) < 0 ? 0 : (y + 1);
|
||
const s32 line_y_end = (y_end - 1) >= cfg::FramebufferHeight ? cfg::FramebufferHeight - 1 : (y_end - 1);
|
||
|
||
// Only proceed if there are actually vertical pixels to draw
|
||
if (line_y_start <= line_y_end) {
|
||
// Left vertical line
|
||
if (x >= 0 && x < cfg::FramebufferWidth) {
|
||
for (s32 yi = line_y_start; yi <= line_y_end; ++yi) {
|
||
this->setPixelBlendDst(x, yi, color);
|
||
}
|
||
}
|
||
|
||
// Right vertical line (only if different from left)
|
||
if (w > 1 && x_end >= 0 && x_end < cfg::FramebufferWidth) {
|
||
for (s32 yi = line_y_start; yi <= line_y_end; ++yi) {
|
||
this->setPixelBlendDst(x_end, yi, color);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief Draws a line
|
||
*
|
||
* @param x0 Start X pos
|
||
* @param y0 Start Y pos
|
||
* @param x1 End X pos
|
||
* @param y1 End Y pos
|
||
* @param color Color
|
||
*/
|
||
inline void drawLine(s32 x0, s32 y0, s32 x1, s32 y1, Color color) {
|
||
// Early exit for single point
|
||
if (x0 == x1 && y0 == y1) {
|
||
if (x0 >= 0 && y0 >= 0 && x0 < cfg::FramebufferWidth && y0 < cfg::FramebufferHeight) {
|
||
this->setPixelBlendDst(x0, y0, color);
|
||
}
|
||
return;
|
||
}
|
||
|
||
// Calculate deltas
|
||
const s32 dx = x1 - x0;
|
||
const s32 dy = y1 - y0;
|
||
|
||
// Calculate absolute deltas and steps
|
||
const s32 abs_dx = dx < 0 ? -dx : dx;
|
||
const s32 abs_dy = dy < 0 ? -dy : dy;
|
||
const s32 step_x = dx < 0 ? -1 : 1;
|
||
const s32 step_y = dy < 0 ? -1 : 1;
|
||
|
||
// Bresenham's algorithm
|
||
s32 x = x0, y = y0;
|
||
s32 error = abs_dx - abs_dy;
|
||
s32 error2;
|
||
|
||
while (true) {
|
||
// Bounds check and draw pixel
|
||
if (x >= 0 && y >= 0 && x < cfg::FramebufferWidth && y < cfg::FramebufferHeight) {
|
||
this->setPixelBlendDst(x, y, color);
|
||
}
|
||
|
||
// Check if we've reached the end point
|
||
if (x == x1 && y == y1) break;
|
||
|
||
// Calculate error and step
|
||
error2 = error << 1; // error * 2
|
||
|
||
if (error2 > -abs_dy) {
|
||
error -= abs_dy;
|
||
x += step_x;
|
||
}
|
||
if (error2 < abs_dx) {
|
||
error += abs_dx;
|
||
y += step_y;
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief Draws a dashed line
|
||
*
|
||
* @param x0 Start X pos
|
||
* @param y0 Start Y pos
|
||
* @param x1 End X pos
|
||
* @param y1 End Y pos
|
||
* @param line_width How long one line can be
|
||
* @param color Color
|
||
*/
|
||
inline void drawDashedLine(s32 x0, s32 y0, s32 x1, s32 y1, s32 line_width, Color color) {
|
||
// Source of formula: https://www.cc.gatech.edu/grads/m/Aaron.E.McClennen/Bresenham/code.html
|
||
|
||
const s32 x_min = std::min(x0, x1);
|
||
const s32 x_max = std::max(x0, x1);
|
||
const s32 y_min = std::min(y0, y1);
|
||
const s32 y_max = std::max(y0, y1);
|
||
|
||
if (x_min < 0 || y_min < 0 || x_min >= cfg::FramebufferWidth || y_min >= cfg::FramebufferHeight)
|
||
return;
|
||
|
||
const s32 dx = x_max - x_min;
|
||
const s32 dy = y_max - y_min;
|
||
s32 d = 2 * dy - dx;
|
||
|
||
const s32 incrE = 2*dy;
|
||
const s32 incrNE = 2*(dy - dx);
|
||
|
||
this->setPixelBlendDst(x_min, y_min, color);
|
||
|
||
s32 x = x_min;
|
||
s32 y = y_min;
|
||
s32 rendered = 0;
|
||
|
||
while(x < x1) {
|
||
if (d <= 0) {
|
||
d += incrE;
|
||
x++;
|
||
}
|
||
else {
|
||
d += incrNE;
|
||
x++;
|
||
y++;
|
||
}
|
||
rendered++;
|
||
if (x < 0 || y < 0 || x >= cfg::FramebufferWidth || y >= cfg::FramebufferHeight)
|
||
continue;
|
||
if (x <= x_max && y <= y_max) {
|
||
if (rendered > 0 && rendered < line_width) {
|
||
this->setPixelBlendDst(x, y, color);
|
||
}
|
||
else if (rendered > 0 && rendered >= line_width) {
|
||
rendered *= -1;
|
||
}
|
||
}
|
||
}
|
||
|
||
}
|
||
|
||
inline void drawCircle(const s32 centerX, const s32 centerY, const u16 radius, const bool filled, const Color& color) {
|
||
s32 x = radius;
|
||
s32 y = 0;
|
||
s32 radiusError = 0;
|
||
s32 xChange = 1 - (radius << 1);
|
||
s32 yChange = 0;
|
||
|
||
while (x >= y) {
|
||
if (filled) {
|
||
for (s32 i = centerX - x; i <= centerX + x; i++) {
|
||
this->setPixelBlendDst(i, centerY + y, color);
|
||
this->setPixelBlendDst(i, centerY - y, color);
|
||
}
|
||
|
||
for (s32 i = centerX - y; i <= centerX + y; i++) {
|
||
this->setPixelBlendDst(i, centerY + x, color);
|
||
this->setPixelBlendDst(i, centerY - x, color);
|
||
}
|
||
} else {
|
||
this->setPixelBlendDst(centerX + x, centerY + y, color);
|
||
this->setPixelBlendDst(centerX + y, centerY + x, color);
|
||
this->setPixelBlendDst(centerX - y, centerY + x, color);
|
||
this->setPixelBlendDst(centerX - x, centerY + y, color);
|
||
this->setPixelBlendDst(centerX - x, centerY - y, color);
|
||
this->setPixelBlendDst(centerX - y, centerY - x, color);
|
||
this->setPixelBlendDst(centerX + y, centerY - x, color);
|
||
this->setPixelBlendDst(centerX + x, centerY - y, color);
|
||
}
|
||
|
||
y++;
|
||
radiusError += yChange;
|
||
yChange += 2;
|
||
|
||
if (((radiusError << 1) + xChange) > 0) {
|
||
x--;
|
||
radiusError += xChange;
|
||
xChange += 2;
|
||
}
|
||
}
|
||
}
|
||
|
||
inline void drawBorderedRoundedRect(const s32 x, const s32 y, const s32 width, const s32 height, const s32 thickness, const s32 radius, const Color& highlightColor) {
|
||
const s32 startX = x + 4;
|
||
const s32 startY = y;
|
||
const s32 adjustedWidth = width - 12;
|
||
const s32 adjustedHeight = height + 1;
|
||
|
||
// Pre-calculate corner positions
|
||
const s32 leftCornerX = startX;
|
||
const s32 rightCornerX = x + width - 9;
|
||
const s32 topCornerY = startY;
|
||
const s32 bottomCornerY = startY + height;
|
||
|
||
// Draw borders (unchanged for exact visual match)
|
||
this->drawRect(startX, startY - thickness, adjustedWidth, thickness, highlightColor); // Top border
|
||
this->drawRect(startX, startY + adjustedHeight, adjustedWidth, thickness, highlightColor); // Bottom border
|
||
this->drawRect(startX - thickness, startY, thickness, adjustedHeight, highlightColor); // Left border
|
||
this->drawRect(startX + adjustedWidth, startY, thickness, adjustedHeight, highlightColor); // Right border
|
||
|
||
// Optimized filled quarter circle drawing - all 4 corners in one pass
|
||
s32 cx = radius;
|
||
s32 cy = 0;
|
||
s32 radiusError = 0;
|
||
s32 xChange = 1 - (radius << 1);
|
||
s32 yChange = 0;
|
||
|
||
while (cx >= cy) {
|
||
// Draw horizontal spans for all 4 corners simultaneously
|
||
// Upper-left corner (quadrant 2) - two horizontal lines
|
||
for (s32 i = leftCornerX - cx; i <= leftCornerX; i++) {
|
||
this->setPixelBlendDst(i, topCornerY - cy, highlightColor);
|
||
}
|
||
for (s32 i = leftCornerX - cy; i <= leftCornerX; i++) {
|
||
this->setPixelBlendDst(i, topCornerY - cx, highlightColor);
|
||
}
|
||
|
||
// Lower-left corner (quadrant 3) - two horizontal lines
|
||
for (s32 i = leftCornerX - cx; i <= leftCornerX; i++) {
|
||
this->setPixelBlendDst(i, bottomCornerY + cy, highlightColor);
|
||
}
|
||
for (s32 i = leftCornerX - cy; i <= leftCornerX; i++) {
|
||
this->setPixelBlendDst(i, bottomCornerY + cx, highlightColor);
|
||
}
|
||
|
||
// Upper-right corner (quadrant 1) - two horizontal lines
|
||
for (s32 i = rightCornerX; i <= rightCornerX + cx; i++) {
|
||
this->setPixelBlendDst(i, topCornerY - cy, highlightColor);
|
||
}
|
||
for (s32 i = rightCornerX; i <= rightCornerX + cy; i++) {
|
||
this->setPixelBlendDst(i, topCornerY - cx, highlightColor);
|
||
}
|
||
|
||
// Lower-right corner (quadrant 4) - two horizontal lines
|
||
for (s32 i = rightCornerX; i <= rightCornerX + cx; i++) {
|
||
this->setPixelBlendDst(i, bottomCornerY + cy, highlightColor);
|
||
}
|
||
for (s32 i = rightCornerX; i <= rightCornerX + cy; i++) {
|
||
this->setPixelBlendDst(i, bottomCornerY + cx, highlightColor);
|
||
}
|
||
|
||
// Bresenham circle algorithm step
|
||
cy++;
|
||
radiusError += yChange;
|
||
yChange += 2;
|
||
|
||
if (((radiusError << 1) + xChange) > 0) {
|
||
cx--;
|
||
radiusError += xChange;
|
||
xChange += 2;
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
// Pre-compute all horizontal spans for the entire shape
|
||
struct HorizontalSpan {
|
||
s32 start_x, end_x;
|
||
};
|
||
|
||
// Define processChunk as a static member function
|
||
// Optimized processRoundedRectChunk - assumes bounds checking done by caller
|
||
static void processRoundedRectChunk(Renderer* self, const s32 x, const s32 y, const s32 w, const s32 h,
|
||
const s32 radius, const Color& color, const s32 startRow, const s32 endRow) {
|
||
// Original rectangle bounds
|
||
const s32 orig_x = x, orig_y = y;
|
||
const s32 orig_x_end = x + w, orig_y_end = y + h;
|
||
|
||
// Calculate clipping bounds
|
||
const s32 clip_x = std::max(0, x);
|
||
const s32 clip_x_end = std::min(static_cast<s32>(cfg::FramebufferWidth), x + w);
|
||
|
||
// Use ORIGINAL coordinates to determine corner regions
|
||
const s32 orig_x_left = orig_x + radius, orig_x_right = orig_x_end - radius;
|
||
const s32 orig_y_top = orig_y + radius, orig_y_bottom = orig_y_end - radius;
|
||
const s32 r2 = radius * radius;
|
||
const u8 red = color.r, green = color.g, blue = color.b, alpha = color.a;
|
||
|
||
alignas(64) u8 redArray[512], greenArray[512], blueArray[512], alphaArray[512];
|
||
for (s32 i = 0; i < 512; i += 8) {
|
||
redArray[i] = redArray[i+1] = redArray[i+2] = redArray[i+3] =
|
||
redArray[i+4] = redArray[i+5] = redArray[i+6] = redArray[i+7] = red;
|
||
greenArray[i] = greenArray[i+1] = greenArray[i+2] = greenArray[i+3] =
|
||
greenArray[i+4] = greenArray[i+5] = greenArray[i+6] = greenArray[i+7] = green;
|
||
blueArray[i] = blueArray[i+1] = blueArray[i+2] = blueArray[i+3] =
|
||
blueArray[i+4] = blueArray[i+5] = blueArray[i+6] = blueArray[i+7] = blue;
|
||
alphaArray[i] = alphaArray[i+1] = alphaArray[i+2] = alphaArray[i+3] =
|
||
alphaArray[i+4] = alphaArray[i+5] = alphaArray[i+6] = alphaArray[i+7] = alpha;
|
||
}
|
||
|
||
s32 orig_span_start, orig_span_end;
|
||
s32 dx;
|
||
for (s32 y_current = startRow; y_current < endRow; ++y_current) {
|
||
// Skip if outside original rectangle bounds
|
||
if (y_current < orig_y || y_current >= orig_y_end) continue;
|
||
|
||
|
||
|
||
if (y_current >= orig_y_top && y_current < orig_y_bottom) {
|
||
// Middle section - full width
|
||
orig_span_start = orig_x;
|
||
orig_span_end = orig_x_end;
|
||
} else {
|
||
// Corner section
|
||
const s32 dy_abs = (y_current < orig_y_top) ? (orig_y_top - y_current) : (y_current - orig_y_bottom);
|
||
const s32 dy2 = dy_abs * dy_abs;
|
||
if (dy2 > r2) continue;
|
||
|
||
// Compute dx using integer square root approximation
|
||
dx = 0;
|
||
const s32 t = r2 - dy2;
|
||
while (dx * dx <= t) {
|
||
dx++;
|
||
}
|
||
dx--; // Get the largest dx where dx^2 + dy2 <= r2
|
||
|
||
// Calculate the span for this row in the original rectangle
|
||
orig_span_start = std::max(orig_x_left - dx, orig_x);
|
||
orig_span_end = std::min(orig_x_right + dx, orig_x_end);
|
||
}
|
||
|
||
// Clip the original span to visible bounds
|
||
const s32 span_start = std::max(orig_span_start, clip_x);
|
||
const s32 span_end = std::min(orig_span_end, clip_x_end);
|
||
|
||
if (span_start >= span_end) continue;
|
||
|
||
// Batch rendering
|
||
for (s32 x_pos = span_start; x_pos < span_end; x_pos += 512) {
|
||
self->setPixelBlendDstBatch(x_pos, y_current, redArray, greenArray, blueArray, alphaArray, std::min(512, span_end - x_pos));
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
|
||
/**
|
||
* @brief Draws a rounded rectangle of given sizes and corner radius (Multi-threaded)
|
||
*
|
||
* @param x X pos
|
||
* @param y Y pos
|
||
* @param w Width
|
||
* @param h Height
|
||
* @param radius Corner radius
|
||
* @param color Color
|
||
*/
|
||
inline void drawRoundedRectMultiThreaded(const s32 x, const s32 y, const s32 w, const s32 h, const s32 radius, const Color& color) {
|
||
if (w <= 0 || h <= 0) return;
|
||
|
||
// Get framebuffer bounds for early exit check
|
||
const s32 fb_width = static_cast<s32>(cfg::FramebufferWidth);
|
||
const s32 fb_height = static_cast<s32>(cfg::FramebufferHeight);
|
||
|
||
// Calculate clipped bounds for early exit check
|
||
const s32 clampedX = std::max(0, x);
|
||
const s32 clampedY = std::max(0, y);
|
||
const s32 clampedXEnd = std::min(fb_width, x + w);
|
||
const s32 clampedYEnd = std::min(fb_height, y + h);
|
||
|
||
// Early exit if nothing to draw after clamping
|
||
if (clampedX >= clampedXEnd || clampedY >= clampedYEnd) return;
|
||
|
||
// Calculate visible dimensions
|
||
const s32 visibleHeight = clampedYEnd - clampedY;
|
||
|
||
// Dynamic chunk size based on visible rectangle height
|
||
const s32 chunkSize = std::max(1, visibleHeight / (static_cast<s32>(ult::numThreads) * 2));
|
||
std::atomic<s32> currentRow(clampedY);
|
||
|
||
auto threadTask = [&]() {
|
||
s32 startRow, endRow;
|
||
while ((startRow = currentRow.fetch_add(chunkSize)) < clampedYEnd) {
|
||
endRow = std::min(startRow + chunkSize, clampedYEnd);
|
||
processRoundedRectChunk(this, x, y, w, h, radius, color, startRow, endRow);
|
||
}
|
||
};
|
||
|
||
// Launch threads using ult::renderThreads array
|
||
for (unsigned i = 0; i < static_cast<unsigned>(ult::numThreads); ++i) {
|
||
ult::renderThreads[i] = std::thread(threadTask);
|
||
}
|
||
|
||
// Join all ult::renderThreads
|
||
for (auto& t : ult::renderThreads) {
|
||
t.join();
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief Draws a rounded rectangle of given sizes and corner radius (Single-threaded)
|
||
*
|
||
* @param x X pos
|
||
* @param y Y pos
|
||
* @param w Width
|
||
* @param h Height
|
||
* @param radius Corner radius
|
||
* @param color Color
|
||
*/
|
||
inline void drawRoundedRectSingleThreaded(const s32 x, const s32 y, const s32 w, const s32 h, const s32 radius, const Color& color) {
|
||
if (w <= 0 || h <= 0) return;
|
||
|
||
// Get framebuffer bounds for early exit check
|
||
const s32 fb_width = static_cast<s32>(cfg::FramebufferWidth);
|
||
const s32 fb_height = static_cast<s32>(cfg::FramebufferHeight);
|
||
|
||
// Calculate clipped bounds for early exit check
|
||
const s32 clampedX = std::max(0, x);
|
||
const s32 clampedY = std::max(0, y);
|
||
const s32 clampedXEnd = std::min(fb_width, x + w);
|
||
const s32 clampedYEnd = std::min(fb_height, y + h);
|
||
|
||
// Early exit if nothing to draw after clamping
|
||
if (clampedX >= clampedXEnd || clampedY >= clampedYEnd) return;
|
||
|
||
processRoundedRectChunk(this, x, y, w, h, radius, color, clampedY, clampedYEnd);
|
||
}
|
||
|
||
std::function<void(s32, s32, s32, s32, s32, Color)> drawRoundedRect;
|
||
inline void updateDrawFunction() {
|
||
if (ult::expandedMemory) {
|
||
drawRoundedRect = [this](s32 x, s32 y, s32 w, s32 h, s32 radius, Color color) {
|
||
drawRoundedRectMultiThreaded(x, y, w, h, radius, color);
|
||
};
|
||
} else {
|
||
drawRoundedRect = [this](s32 x, s32 y, s32 w, s32 h, s32 radius, Color color) {
|
||
drawRoundedRectSingleThreaded(x, y, w, h, radius, color);
|
||
};
|
||
}
|
||
}
|
||
|
||
|
||
inline void drawUniformRoundedRect(const s32 x, const s32 y, const s32 w, const s32 h, const Color& color) {
|
||
// Early exit for degenerate cases
|
||
//if (w <= 0 || h <= 0) return;
|
||
|
||
// Calculate radius and bounds
|
||
const s32 radius = h >> 1; // h / 2
|
||
//if (radius <= 0) return;
|
||
|
||
// Get framebuffer bounds
|
||
const s32 fb_width = cfg::FramebufferWidth;
|
||
const s32 fb_height = cfg::FramebufferHeight;
|
||
|
||
// Calculate clipped drawing bounds
|
||
const s32 clip_left = std::max(0, x);
|
||
const s32 clip_top = std::max(0, y);
|
||
const s32 clip_right = std::min(fb_width, x + w);
|
||
const s32 clip_bottom = std::min(fb_height, y + h);
|
||
|
||
// Early exit if completely clipped
|
||
if (clip_left >= clip_right || clip_top >= clip_bottom) return;
|
||
|
||
// Shape parameters
|
||
const s32 center_y = y + radius;
|
||
const s32 rect_left = x + radius;
|
||
const s32 rect_right = x + w - radius;
|
||
const s32 radius_sq = radius * radius;
|
||
|
||
// Choose drawing method based on alpha
|
||
const bool fullOpacity = (color.a == 0xF);
|
||
|
||
// Pre-compute variables
|
||
s32 y_curr, x_curr;
|
||
s32 dy, dy_sq, x_offset_sq;
|
||
s32 x_offset, row_start, row_end;
|
||
//u32 pixel_offset;
|
||
|
||
// Main drawing loop
|
||
for (y_curr = clip_top; y_curr < clip_bottom; ++y_curr) {
|
||
dy = y_curr - center_y;
|
||
dy_sq = dy * dy;
|
||
|
||
// Skip rows outside the shape
|
||
if (dy_sq > radius_sq) continue;
|
||
|
||
// Calculate horizontal extent for this row
|
||
x_offset_sq = radius_sq - dy_sq;
|
||
|
||
// Fast integer square root with better rounding
|
||
if (radius <= 32) {
|
||
// Direct calculation for small values
|
||
x_offset = 0;
|
||
while (x_offset * x_offset <= x_offset_sq) {
|
||
x_offset++;
|
||
}
|
||
// More intelligent step-back: only if we're significantly over
|
||
// This reduces the "flat edge" appearance
|
||
if (x_offset > 0) {
|
||
s32 current_sq = x_offset * x_offset;
|
||
s32 prev_sq = (x_offset - 1) * (x_offset - 1);
|
||
// Only step back if we're closer to the previous value
|
||
if (current_sq - x_offset_sq > x_offset_sq - prev_sq) {
|
||
x_offset--;
|
||
}
|
||
}
|
||
} else {
|
||
// Newton's method for larger values (converges in ~4 iterations)
|
||
x_offset = radius; // Initial guess
|
||
for (int i = 0; i < 4; ++i) {
|
||
x_offset = (x_offset + x_offset_sq / x_offset) >> 1;
|
||
}
|
||
// Ensure we're close to the actual value
|
||
while ((x_offset + 1) * (x_offset + 1) <= x_offset_sq) x_offset++;
|
||
while (x_offset * x_offset > x_offset_sq) x_offset--;
|
||
}
|
||
|
||
// Calculate row bounds
|
||
row_start = rect_left - x_offset;
|
||
row_end = rect_right + x_offset;
|
||
|
||
// Clip to visible area
|
||
row_start = std::max(row_start, clip_left);
|
||
row_end = std::min(row_end, clip_right);
|
||
|
||
if (row_start >= row_end) continue;
|
||
|
||
// Draw the row
|
||
if (fullOpacity) {
|
||
// Use setPixel for full opacity
|
||
for (x_curr = row_start; x_curr < row_end; ++x_curr) {
|
||
//this->setPixel(x_curr, y_curr, color);
|
||
//const u32 pixel_offset = this->getPixelOffset(x, y);
|
||
this->setPixelAtOffset(this->getPixelOffset(x_curr, y_curr), color);
|
||
}
|
||
} else {
|
||
// Blended drawing
|
||
for (x_curr = row_start; x_curr < row_end; ++x_curr) {
|
||
this->setPixelBlendDst(x_curr, y_curr, color);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
// Struct for batch pixel processing with better alignment
|
||
struct alignas(64) PixelBatch {
|
||
s32 baseX, baseY;
|
||
u8 red[32], green[32], blue[32], alpha[32]; // Doubled for 32-pixel batches
|
||
s32 count;
|
||
};
|
||
|
||
// Batch pixel setter - process multiple pixels at once if available
|
||
inline void setPixelBatchBlendSrc(const s32 baseX, const s32 baseY, const PixelBatch& batch) {
|
||
// If your graphics system supports batch operations, use them here
|
||
// Otherwise fall back to individual calls
|
||
for (s32 i = 0; i < batch.count; ++i) {
|
||
setPixelBlendSrc(baseX + i, baseY, {
|
||
batch.red[i], batch.green[i], batch.blue[i], batch.alpha[i]
|
||
});
|
||
}
|
||
}
|
||
|
||
// Fixed compilation errors - simplified SIMD version
|
||
static constexpr uint8x16_t lut = {0, 17, 34, 51, 68, 85, 102, 119, 136, 153, 170, 187, 204, 221, 238, 255};
|
||
const uint8x16_t mask_low = vdupq_n_u8(0x0F);
|
||
// Pre-computed lookup table for 4-bit to 8-bit conversion
|
||
static constexpr u8 expand4to8[16] = {
|
||
0, 17, 34, 51, 68, 85, 102, 119, 136, 153, 170, 187, 204, 221, 238, 255
|
||
};
|
||
|
||
// inline void processBMPChunk(const s32 x, const s32 y, const s32 screenW, const u8 *preprocessedData,
|
||
// const s32 startRow, const s32 endRow) {
|
||
// const s32 bytesPerRow = screenW * 2;
|
||
// const s32 endX16 = screenW & ~15;
|
||
//
|
||
// // Pre-declare all variables outside loops to avoid repeated allocations
|
||
// const u8 *rowPtr;
|
||
// s32 baseY;
|
||
// s32 x1;
|
||
// const u8* ptr;
|
||
// uint8x16x2_t packed;
|
||
// uint8x16_t high1, low1, high2, low2;
|
||
// uint8x16_t red, green, blue, alpha;
|
||
// alignas(16) u8 red_vals[16], green_vals[16], blue_vals[16], alpha_vals[16];
|
||
// s32 baseX;
|
||
// s32 pixelX;
|
||
// u32 offset;
|
||
// Color color = {0}, src = {0}, end = {0};
|
||
// const u16* framebuffer;
|
||
// u8 p1, p2;
|
||
//
|
||
// for (s32 y1 = startRow; y1 < endRow; ++y1) {
|
||
// rowPtr = preprocessedData + (y1 * bytesPerRow);
|
||
// baseY = y + y1;
|
||
//
|
||
// x1 = 0;
|
||
//
|
||
// // SIMD processing for 16 pixels at once
|
||
// for (; x1 < endX16; x1 += 16) {
|
||
// ptr = rowPtr + (x1 << 1);
|
||
// packed = vld2q_u8(ptr);
|
||
//
|
||
// // Expand 4-bit to 8-bit values
|
||
// high1 = vshrq_n_u8(packed.val[0], 4);
|
||
// low1 = vandq_u8(packed.val[0], mask_low);
|
||
// high2 = vshrq_n_u8(packed.val[1], 4);
|
||
// low2 = vandq_u8(packed.val[1], mask_low);
|
||
//
|
||
// red = vqtbl1q_u8(lut, high1);
|
||
// green = vqtbl1q_u8(lut, low1);
|
||
// blue = vqtbl1q_u8(lut, high2);
|
||
// alpha = vqtbl1q_u8(lut, low2);
|
||
//
|
||
// // Store to arrays and process individually
|
||
// vst1q_u8(red_vals, red);
|
||
// vst1q_u8(green_vals, green);
|
||
// vst1q_u8(blue_vals, blue);
|
||
// vst1q_u8(alpha_vals, alpha);
|
||
//
|
||
// baseX = x + x1;
|
||
//
|
||
// // Process 16 pixels with minimal function call overhead
|
||
// for (int i = 0; i < 16; ++i) {
|
||
// // Skip transparent pixels
|
||
// if (alpha_vals[i] == 0) continue;
|
||
//
|
||
// pixelX = baseX + i;
|
||
// offset = this->getPixelOffset(pixelX, baseY);
|
||
//
|
||
// if (offset != UINT32_MAX) {
|
||
// color = {red_vals[i], green_vals[i], blue_vals[i], alpha_vals[i]};
|
||
//
|
||
// framebuffer = static_cast<const u16*>(this->getCurrentFramebuffer());
|
||
// src = Color(framebuffer[offset]);
|
||
//
|
||
// end = {
|
||
// blendColor(src.r, color.r, color.a),
|
||
// blendColor(src.g, color.g, color.a),
|
||
// blendColor(src.b, color.b, color.a),
|
||
// src.a
|
||
// };
|
||
//
|
||
// this->setPixelAtOffset(offset, end);
|
||
// }
|
||
// }
|
||
// }
|
||
//
|
||
// // Handle remaining pixels (less than 16)
|
||
// for (; x1 < screenW; ++x1) {
|
||
// p1 = rowPtr[x1 << 1];
|
||
// p2 = rowPtr[(x1 << 1) + 1];
|
||
//
|
||
// setPixelBlendSrc(x + x1, baseY, {
|
||
// expand4to8[p1 >> 4], expand4to8[p1 & 0x0F],
|
||
// expand4to8[p2 >> 4], expand4to8[p2 & 0x0F]
|
||
// });
|
||
// }
|
||
// }
|
||
//
|
||
// ult::inPlotBarrier.arrive_and_wait();
|
||
// }
|
||
|
||
|
||
inline void processBMPChunk(const s32 x, const s32 y, const s32 screenW, const u8 *preprocessedData,
|
||
const s32 startRow, const s32 endRow, const u8 globalAlphaLimit) {
|
||
const s32 bytesPerRow = screenW * 2;
|
||
const s32 endX16 = screenW & ~15;
|
||
|
||
// Create SIMD vector for alpha limit
|
||
const uint8x16_t alpha_limit_vec = vdupq_n_u8(globalAlphaLimit);
|
||
|
||
// Pre-declare all variables outside loops
|
||
const u8 *rowPtr;
|
||
s32 baseY;
|
||
s32 x1;
|
||
const u8* ptr;
|
||
uint8x16x2_t packed;
|
||
uint8x16_t high1, low1, high2, low2;
|
||
uint8x16_t red, green, blue, alpha;
|
||
alignas(16) u8 red_vals[16], green_vals[16], blue_vals[16], alpha_vals[16];
|
||
s32 baseX;
|
||
s32 pixelX;
|
||
u32 offset;
|
||
Color color = {0}, src = {0}, end = {0};
|
||
const u16* framebuffer;
|
||
u8 p1, p2;
|
||
|
||
for (s32 y1 = startRow; y1 < endRow; ++y1) {
|
||
rowPtr = preprocessedData + (y1 * bytesPerRow);
|
||
baseY = y + y1;
|
||
|
||
x1 = 0;
|
||
|
||
// SIMD processing for 16 pixels at once
|
||
for (; x1 < endX16; x1 += 16) {
|
||
ptr = rowPtr + (x1 << 1);
|
||
packed = vld2q_u8(ptr);
|
||
|
||
// Expand 4-bit to 8-bit values
|
||
high1 = vshrq_n_u8(packed.val[0], 4);
|
||
low1 = vandq_u8(packed.val[0], mask_low);
|
||
high2 = vshrq_n_u8(packed.val[1], 4);
|
||
low2 = vandq_u8(packed.val[1], mask_low);
|
||
|
||
red = vqtbl1q_u8(lut, high1);
|
||
green = vqtbl1q_u8(lut, low1);
|
||
blue = vqtbl1q_u8(lut, high2);
|
||
alpha = vqtbl1q_u8(lut, low2);
|
||
|
||
// Apply alpha limit using SIMD min operation
|
||
alpha = vminq_u8(alpha, alpha_limit_vec);
|
||
|
||
// Store to arrays and process individually
|
||
vst1q_u8(red_vals, red);
|
||
vst1q_u8(green_vals, green);
|
||
vst1q_u8(blue_vals, blue);
|
||
vst1q_u8(alpha_vals, alpha);
|
||
|
||
baseX = x + x1;
|
||
|
||
// Process 16 pixels with minimal function call overhead
|
||
for (int i = 0; i < 16; ++i) {
|
||
// Skip transparent pixels
|
||
if (alpha_vals[i] == 0) continue;
|
||
|
||
pixelX = baseX + i;
|
||
offset = this->getPixelOffset(pixelX, baseY);
|
||
|
||
if (offset != UINT32_MAX) {
|
||
color = {red_vals[i], green_vals[i], blue_vals[i], alpha_vals[i]};
|
||
|
||
framebuffer = static_cast<const u16*>(this->getCurrentFramebuffer());
|
||
src = Color(framebuffer[offset]);
|
||
|
||
end = {
|
||
blendColor(src.r, color.r, color.a),
|
||
blendColor(src.g, color.g, color.a),
|
||
blendColor(src.b, color.b, color.a),
|
||
src.a
|
||
};
|
||
|
||
this->setPixelAtOffset(offset, end);
|
||
}
|
||
}
|
||
}
|
||
|
||
// Handle remaining pixels (less than 16) with pre-computed alpha limit
|
||
for (; x1 < screenW; ++x1) {
|
||
p1 = rowPtr[x1 << 1];
|
||
p2 = rowPtr[(x1 << 1) + 1];
|
||
|
||
u8 alpha = expand4to8[p2 & 0x0F];
|
||
alpha = (alpha < globalAlphaLimit) ? alpha : globalAlphaLimit;
|
||
|
||
setPixelBlendSrc(x + x1, baseY, {
|
||
expand4to8[p1 >> 4], expand4to8[p1 & 0x0F],
|
||
expand4to8[p2 >> 4], alpha
|
||
});
|
||
}
|
||
}
|
||
|
||
ult::inPlotBarrier.arrive_and_wait();
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Draws a scaled RGBA8888 bitmap from memory
|
||
*
|
||
* @param x X start position
|
||
* @param y Y start position
|
||
* @param w Bitmap width (original width of the bitmap)
|
||
* @param h Bitmap height (original height of the bitmap)
|
||
* @param bmp Pointer to bitmap data
|
||
* @param screenW Target screen width
|
||
* @param screenH Target screen height
|
||
*/
|
||
|
||
//inline void drawBitmapRGBA4444(const s32 x, const s32 y, const s32 screenW, const s32 screenH, const u8 *preprocessedData) {
|
||
// s32 startRow;
|
||
//
|
||
// // Divide rows among ult::renderThreads
|
||
// //s32 chunkSize = (screenH + ult::numThreads - 1) / ult::numThreads;
|
||
// for (unsigned i = 0; i < ult::numThreads; ++i) {
|
||
// startRow = i * ult::bmpChunkSize;
|
||
// //s32 endRow = std::min(startRow + ult::bmpChunkSize, screenH);
|
||
//
|
||
// // Bind the member function and create the thread
|
||
// ult::renderThreads[i] = std::thread(std::bind(&tsl::gfx::Renderer::processBMPChunk, this, x, y, screenW, preprocessedData, startRow, std::min(startRow + ult::bmpChunkSize, screenH)));
|
||
// }
|
||
//
|
||
// // Join all ult::renderThreads
|
||
// for (auto& t : ult::renderThreads) {
|
||
// t.join();
|
||
// }
|
||
//}
|
||
|
||
inline void drawBitmapRGBA4444(const s32 x, const s32 y, const s32 screenW, const s32 screenH,
|
||
const u8 *preprocessedData, float opacity = 1.0f) {
|
||
// Pre-compute alpha limit once
|
||
const u8 globalAlphaLimit = static_cast<u8>(0xF * opacity);
|
||
|
||
s32 startRow;
|
||
|
||
for (unsigned i = 0; i < ult::numThreads; ++i) {
|
||
startRow = i * ult::bmpChunkSize;
|
||
|
||
// Pass the alpha limit to each thread
|
||
ult::renderThreads[i] = std::thread(std::bind(&tsl::gfx::Renderer::processBMPChunk,
|
||
this, x, y, screenW, preprocessedData, startRow,
|
||
std::min(startRow + ult::bmpChunkSize, screenH), globalAlphaLimit));
|
||
}
|
||
|
||
// Join all threads
|
||
for (auto& t : ult::renderThreads) {
|
||
t.join();
|
||
}
|
||
}
|
||
|
||
|
||
|
||
//inline void drawWallpaper() {
|
||
// if (!ult::expandedMemory || ult::refreshWallpaper.load(std::memory_order_acquire)) {
|
||
// return;
|
||
// }
|
||
//
|
||
// ult::inPlot.store(true, std::memory_order_release);
|
||
//
|
||
// if (!ult::wallpaperData.empty() &&
|
||
// !ult::refreshWallpaper.load(std::memory_order_acquire) &&
|
||
// ult::correctFrameSize) {
|
||
// drawBitmapRGBA4444(0, 0, cfg::FramebufferWidth, cfg::FramebufferHeight, ult::wallpaperData.data());
|
||
// }
|
||
//
|
||
// ult::inPlot.store(false, std::memory_order_release);
|
||
//}
|
||
|
||
inline void drawWallpaper() {
|
||
if (!ult::expandedMemory || ult::refreshWallpaper.load(std::memory_order_acquire)) {
|
||
return;
|
||
}
|
||
|
||
ult::inPlot.store(true, std::memory_order_release);
|
||
|
||
if (!ult::wallpaperData.empty() &&
|
||
!ult::refreshWallpaper.load(std::memory_order_acquire) &&
|
||
ult::correctFrameSize) {
|
||
// Use the renderer's opacity directly
|
||
drawBitmapRGBA4444(0, 0, cfg::FramebufferWidth, cfg::FramebufferHeight,
|
||
ult::wallpaperData.data(), Renderer::s_opacity);
|
||
}
|
||
|
||
ult::inPlot.store(false, std::memory_order_release);
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Draws a RGBA8888 bitmap from memory
|
||
*
|
||
* @param x X start position
|
||
* @param y Y start position
|
||
* @param w Bitmap width
|
||
* @param h Bitmap height
|
||
* @param bmp Pointer to bitmap data
|
||
*/
|
||
inline void drawBitmap(s32 x, s32 y, s32 w, s32 h, const u8 *bmp) {
|
||
if (w <= 0 || h <= 0) [[unlikely]] return;
|
||
|
||
const u8* __restrict__ src = bmp;
|
||
|
||
s32 px;
|
||
|
||
// Completely unroll small bitmaps for maximum speed
|
||
if (w <= 8 && h <= 8) [[likely]] {
|
||
// Specialized path for small bitmaps (icons, etc.)
|
||
for (s32 py = 0; py < h; ++py) {
|
||
const s32 rowY = y + py;
|
||
px = x;
|
||
|
||
// Unroll inner loop completely for small widths
|
||
switch(w) {
|
||
case 8: goto pixel8;
|
||
case 7: goto pixel7;
|
||
case 6: goto pixel6;
|
||
case 5: goto pixel5;
|
||
case 4: goto pixel4;
|
||
case 3: goto pixel3;
|
||
case 2: goto pixel2;
|
||
case 1: goto pixel1;
|
||
default: break;
|
||
}
|
||
|
||
pixel8: {
|
||
const Color c = {static_cast<u8>(src[0] >> 4), static_cast<u8>(src[1] >> 4),
|
||
static_cast<u8>(src[2] >> 4), static_cast<u8>(src[3] >> 4)};
|
||
setPixelBlendSrc(px++, rowY, a(c)); src += 4;
|
||
}
|
||
pixel7: {
|
||
const Color c = {static_cast<u8>(src[0] >> 4), static_cast<u8>(src[1] >> 4),
|
||
static_cast<u8>(src[2] >> 4), static_cast<u8>(src[3] >> 4)};
|
||
setPixelBlendSrc(px++, rowY, a(c)); src += 4;
|
||
}
|
||
pixel6: {
|
||
const Color c = {static_cast<u8>(src[0] >> 4), static_cast<u8>(src[1] >> 4),
|
||
static_cast<u8>(src[2] >> 4), static_cast<u8>(src[3] >> 4)};
|
||
setPixelBlendSrc(px++, rowY, a(c)); src += 4;
|
||
}
|
||
pixel5: {
|
||
const Color c = {static_cast<u8>(src[0] >> 4), static_cast<u8>(src[1] >> 4),
|
||
static_cast<u8>(src[2] >> 4), static_cast<u8>(src[3] >> 4)};
|
||
setPixelBlendSrc(px++, rowY, a(c)); src += 4;
|
||
}
|
||
pixel4: {
|
||
const Color c = {static_cast<u8>(src[0] >> 4), static_cast<u8>(src[1] >> 4),
|
||
static_cast<u8>(src[2] >> 4), static_cast<u8>(src[3] >> 4)};
|
||
setPixelBlendSrc(px++, rowY, a(c)); src += 4;
|
||
}
|
||
pixel3: {
|
||
const Color c = {static_cast<u8>(src[0] >> 4), static_cast<u8>(src[1] >> 4),
|
||
static_cast<u8>(src[2] >> 4), static_cast<u8>(src[3] >> 4)};
|
||
setPixelBlendSrc(px++, rowY, a(c)); src += 4;
|
||
}
|
||
pixel2: {
|
||
const Color c = {static_cast<u8>(src[0] >> 4), static_cast<u8>(src[1] >> 4),
|
||
static_cast<u8>(src[2] >> 4), static_cast<u8>(src[3] >> 4)};
|
||
setPixelBlendSrc(px++, rowY, a(c)); src += 4;
|
||
}
|
||
pixel1: {
|
||
const Color c = {static_cast<u8>(src[0] >> 4), static_cast<u8>(src[1] >> 4),
|
||
static_cast<u8>(src[2] >> 4), static_cast<u8>(src[3] >> 4)};
|
||
setPixelBlendSrc(px, rowY, a(c)); src += 4;
|
||
}
|
||
}
|
||
return;
|
||
}
|
||
|
||
// Fallback to vectorized version for larger bitmaps
|
||
const s32 vectorWidth = w & ~7; // Process 8 pixels at a time
|
||
const s32 remainder = w & 7;
|
||
|
||
for (s32 py = 0; py < h; ++py) {
|
||
const s32 rowY = y + py;
|
||
px = x;
|
||
|
||
// Process 8 pixels at once (cache-friendly)
|
||
for (s32 i = 0; i < vectorWidth; i += 8) {
|
||
// Prefetch next cache line
|
||
__builtin_prefetch(src + 64, 0, 3);
|
||
|
||
// Process 8 pixels with minimal overhead
|
||
for (int j = 0; j < 8; ++j) {
|
||
const Color c = {static_cast<u8>(src[0] >> 4), static_cast<u8>(src[1] >> 4),
|
||
static_cast<u8>(src[2] >> 4), static_cast<u8>(src[3] >> 4)};
|
||
setPixelBlendSrc(px++, rowY, (c));
|
||
src += 4;
|
||
}
|
||
}
|
||
|
||
// Handle remainder
|
||
for (s32 i = 0; i < remainder; ++i) {
|
||
const Color c = {static_cast<u8>(src[0] >> 4), static_cast<u8>(src[1] >> 4),
|
||
static_cast<u8>(src[2] >> 4), static_cast<u8>(src[3] >> 4)};
|
||
setPixelBlendSrc(px++, rowY, (c));
|
||
src += 4;
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief Fills the entire layer with a given color
|
||
*
|
||
* @param color Color
|
||
*/
|
||
inline void fillScreen(const Color& color) {
|
||
std::fill_n(static_cast<Color*>(this->getCurrentFramebuffer()), this->getFramebufferSize() / sizeof(Color), color);
|
||
}
|
||
|
||
/**
|
||
* @brief Clears the layer (With transparency)
|
||
*
|
||
*/
|
||
inline void clearScreen() {
|
||
this->fillScreen(Color(0x0, 0x0, 0x0, 0x0)); // Fully transparent
|
||
}
|
||
|
||
//struct Glyph {
|
||
// stbtt_fontinfo *currFont;
|
||
// float currFontSize;
|
||
// int bounds[4];
|
||
// int xAdvance;
|
||
// u8 *glyphBmp;
|
||
// int width, height;
|
||
//};
|
||
|
||
const stbtt_fontinfo& getStandardFont() const {
|
||
return m_stdFont;
|
||
}
|
||
|
||
|
||
// Optimized unified drawString method with thread safety
|
||
inline std::pair<s32, s32> drawString(const std::string& originalString, bool monospace,
|
||
const s32 x, const s32 y, const u32 fontSize,
|
||
const Color& defaultColor, const ssize_t maxWidth = 0,
|
||
bool draw = true,
|
||
const Color* highlightColor = nullptr,
|
||
const std::vector<std::string>* specialSymbols = nullptr,
|
||
const u32 highlightStartChar = 0,
|
||
const u32 highlightEndChar = 0,
|
||
const bool useNotificationCache = false) { // NEW parameter
|
||
|
||
// Thread-safe translation cache access
|
||
std::string text;
|
||
#ifdef UI_OVERRIDE_PATH
|
||
{
|
||
std::shared_lock<std::shared_mutex> readLock(s_translationCacheMutex);
|
||
auto translatedIt = ult::translationCache.find(originalString);
|
||
if (translatedIt != ult::translationCache.end()) {
|
||
text = translatedIt->second;
|
||
} else {
|
||
// Need to upgrade to write lock
|
||
readLock.unlock();
|
||
std::unique_lock<std::shared_mutex> writeLock(s_translationCacheMutex);
|
||
|
||
// Double-check pattern
|
||
translatedIt = ult::translationCache.find(originalString);
|
||
if (translatedIt != ult::translationCache.end()) {
|
||
text = translatedIt->second;
|
||
} else {
|
||
ult::translationCache[originalString] = originalString;
|
||
text = originalString;
|
||
}
|
||
}
|
||
}
|
||
#else
|
||
text = originalString;
|
||
#endif
|
||
|
||
if (text.empty() || fontSize == 0) return {0, 0};
|
||
|
||
const float maxWidthLimit = maxWidth > 0 ? x + maxWidth : std::numeric_limits<float>::max();
|
||
|
||
// Check if highlighting is enabled (both highlight color and delimiters must be provided)
|
||
const bool highlightingEnabled = highlightColor && highlightStartChar != 0 && highlightEndChar != 0;
|
||
|
||
// Get font metrics for consistent line height using a standard character
|
||
// This ensures consistent line spacing regardless of which specific characters are used
|
||
const auto fontMetrics = FontManager::getFontMetricsForCharacter('A', fontSize);
|
||
const s32 lineHeight = static_cast<s32>(fontMetrics.lineHeight);
|
||
|
||
// Fast ASCII check with early exit
|
||
bool isAsciiOnly = true;
|
||
const char* textPtr = text.data();
|
||
const char* textEnd = textPtr + text.size();
|
||
for (const char* p = textPtr; p < textEnd; ++p) {
|
||
if (static_cast<unsigned char>(*p) > 127) {
|
||
isAsciiOnly = false;
|
||
break;
|
||
}
|
||
}
|
||
|
||
s32 maxX = x, currX = x, currY = y; // Changed to s32 for consistency
|
||
s32 maxY = y + lineHeight; // Initialize with at least one line height
|
||
bool inHighlight = false;
|
||
const Color* currentColor = &defaultColor;
|
||
|
||
// Pre-declare variables used in loops to avoid repeated allocations
|
||
u32 currCharacter;
|
||
ssize_t codepointWidth;
|
||
std::shared_ptr<FontManager::Glyph> glyph;
|
||
bool symbolProcessed;
|
||
size_t remainingLength;
|
||
u32 symChar;
|
||
ssize_t symWidth;
|
||
size_t i;
|
||
|
||
// Main processing loop with pointer arithmetic for ASCII optimization
|
||
if (isAsciiOnly && !specialSymbols) {
|
||
// Fast ASCII-only path
|
||
for (const char* p = textPtr; p < textEnd && currX < maxWidthLimit; ++p) {
|
||
currCharacter = static_cast<u32>(*p);
|
||
|
||
// Handle highlighting with configurable delimiters
|
||
if (highlightingEnabled) {
|
||
if (currCharacter == highlightStartChar) {
|
||
inHighlight = true;
|
||
} else if (currCharacter == highlightEndChar) {
|
||
inHighlight = false;
|
||
}
|
||
currentColor = (currCharacter == highlightStartChar || currCharacter == highlightEndChar) ?
|
||
&defaultColor : (inHighlight ? highlightColor : &defaultColor);
|
||
}
|
||
|
||
// Handle newline
|
||
if (currCharacter == '\n') {
|
||
maxX = std::max(currX, maxX);
|
||
currX = x;
|
||
currY += lineHeight; // Use consistent line height
|
||
maxY = std::max(maxY, currY + lineHeight); // Update maxY for new line
|
||
continue;
|
||
}
|
||
|
||
// Get glyph (now thread-safe)
|
||
// Get glyph - UPDATED to use notification cache when requested
|
||
if (useNotificationCache) {
|
||
glyph = FontManager::getOrCreateNotificationGlyph(currCharacter, monospace, fontSize);
|
||
} else {
|
||
glyph = FontManager::getOrCreateGlyph(currCharacter, monospace, fontSize);
|
||
}
|
||
if (!glyph) continue;
|
||
|
||
// Track maximum Y position reached using consistent line height
|
||
maxY = std::max(maxY, currY + lineHeight);
|
||
|
||
// Render if needed
|
||
if (draw && glyph->glyphBmp && currCharacter > 32) { // Space is 32
|
||
renderGlyph(glyph, currX, currY, *currentColor);
|
||
}
|
||
|
||
currX += static_cast<s32>(glyph->xAdvance * glyph->currFontSize);
|
||
}
|
||
} else {
|
||
// UTF-8 path with special symbols support
|
||
auto itStr = text.cbegin();
|
||
const auto itStrEnd = text.cend();
|
||
|
||
while (itStr != itStrEnd && currX < maxWidthLimit) {
|
||
// Check for special symbols first
|
||
symbolProcessed = false;
|
||
if (specialSymbols) {
|
||
remainingLength = itStrEnd - itStr;
|
||
for (const auto& symbol : *specialSymbols) {
|
||
if (remainingLength >= symbol.length() &&
|
||
std::equal(symbol.begin(), symbol.end(), itStr)) {
|
||
|
||
// Process special symbol
|
||
for (i = 0; i < symbol.length(); ) {
|
||
symWidth = decode_utf8(&symChar,
|
||
reinterpret_cast<const u8*>(&symbol[i]));
|
||
if (symWidth <= 0) break;
|
||
|
||
if (symChar == '\n') {
|
||
maxX = std::max(currX, maxX);
|
||
currX = x;
|
||
currY += lineHeight; // Use consistent line height
|
||
maxY = std::max(maxY, currY + lineHeight); // Update maxY for new line
|
||
} else {
|
||
glyph = FontManager::getOrCreateGlyph(symChar, monospace, fontSize);
|
||
if (glyph) {
|
||
// Track maximum Y position reached using consistent line height
|
||
maxY = std::max(maxY, currY + lineHeight);
|
||
|
||
if (draw && glyph->glyphBmp && symChar > 32) {
|
||
renderGlyph(glyph, currX, currY, *highlightColor);
|
||
}
|
||
currX += static_cast<s32>(glyph->xAdvance * glyph->currFontSize);
|
||
}
|
||
}
|
||
i += symWidth;
|
||
}
|
||
itStr += symbol.length();
|
||
symbolProcessed = true;
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (symbolProcessed) continue;
|
||
|
||
// Decode character
|
||
if (isAsciiOnly) {
|
||
currCharacter = static_cast<u32>(*itStr);
|
||
codepointWidth = 1;
|
||
} else {
|
||
codepointWidth = decode_utf8(&currCharacter, reinterpret_cast<const u8*>(&(*itStr)));
|
||
if (codepointWidth <= 0) break;
|
||
}
|
||
|
||
itStr += codepointWidth;
|
||
|
||
// Handle highlighting with configurable delimiters
|
||
if (highlightingEnabled) {
|
||
if (currCharacter == highlightStartChar) {
|
||
inHighlight = true;
|
||
} else if (currCharacter == highlightEndChar) {
|
||
inHighlight = false;
|
||
}
|
||
currentColor = (currCharacter == highlightStartChar || currCharacter == highlightEndChar) ?
|
||
&defaultColor : (inHighlight ? highlightColor : &defaultColor);
|
||
}
|
||
|
||
// Handle newline
|
||
if (currCharacter == '\n') {
|
||
maxX = std::max(currX, maxX);
|
||
currX = x;
|
||
currY += lineHeight; // Use consistent line height
|
||
maxY = std::max(maxY, currY + lineHeight); // Update maxY for new line
|
||
continue;
|
||
}
|
||
|
||
// Get glyph (now thread-safe)
|
||
glyph = FontManager::getOrCreateGlyph(currCharacter, monospace, fontSize);
|
||
if (!glyph) continue;
|
||
|
||
// Track maximum Y position reached using consistent line height
|
||
maxY = std::max(maxY, currY + lineHeight);
|
||
|
||
// Render if needed
|
||
if (draw && glyph->glyphBmp && currCharacter > 32) {
|
||
renderGlyph(glyph, currX, currY, *currentColor);
|
||
}
|
||
|
||
currX += static_cast<s32>(glyph->xAdvance * glyph->currFontSize);
|
||
}
|
||
}
|
||
|
||
maxX = std::max(currX, maxX);
|
||
// Return consistent height based on proper font metrics
|
||
return {maxX - x, maxY - y};
|
||
}
|
||
|
||
inline std::pair<s32, s32> drawNotificationString(const std::string& text, bool monospace,
|
||
const s32 x, const s32 y, const u32 fontSize,
|
||
const Color& defaultColor, const ssize_t maxWidth = 0,
|
||
bool draw = true,
|
||
const Color* highlightColor = nullptr,
|
||
const std::vector<std::string>* specialSymbols = nullptr,
|
||
const u32 highlightStartChar = 0,
|
||
const u32 highlightEndChar = 0) {
|
||
return drawString(text, monospace, x, y, fontSize, defaultColor, maxWidth, draw,
|
||
highlightColor, specialSymbols, highlightStartChar, highlightEndChar, true);
|
||
}
|
||
|
||
// Convenience wrappers for backward compatibility
|
||
inline std::pair<s32, s32> drawStringWithHighlight(const std::string& text, bool monospace,
|
||
s32 x, s32 y, const u32 fontSize,
|
||
const Color& defaultColor,
|
||
const Color& specialColor,
|
||
const ssize_t maxWidth = 0,
|
||
const u32 startChar = '(',
|
||
const u32 endChar = ')') {
|
||
return drawString(text, monospace, x, y, fontSize, defaultColor, maxWidth, true, &specialColor, nullptr, startChar, endChar);
|
||
}
|
||
|
||
inline std::pair<s32, s32> drawStringWithColoredSections(const std::string& text, bool monospace,
|
||
const std::vector<std::string>& specialSymbols,
|
||
s32 x, const s32 y, const u32 fontSize,
|
||
const Color& defaultColor,
|
||
const Color& specialColor) {
|
||
return drawString(text, monospace, x, y, fontSize, defaultColor, 0, true, &specialColor, &specialSymbols);
|
||
}
|
||
|
||
// Calculate string dimensions without drawing
|
||
inline std::pair<s32, s32> getTextDimensions(const std::string& text, bool monospace,
|
||
const u32 fontSize, const ssize_t maxWidth = 0) {
|
||
return drawString(text, monospace, 0, 0, fontSize, Color{0,0,0,0}, maxWidth, false);
|
||
}
|
||
|
||
inline std::pair<s32, s32> getNotificationTextDimensions(const std::string& text, bool monospace,
|
||
const u32 fontSize, const ssize_t maxWidth = 0) {
|
||
return drawString(text, monospace, 0, 0, fontSize, Color{0,0,0,0}, maxWidth, false,
|
||
nullptr, nullptr, 0, 0, true);
|
||
}
|
||
|
||
// Thread-safe limitStringLength using the unified cache
|
||
inline std::string limitStringLength(const std::string& originalString, const bool monospace,
|
||
const u32 fontSize, const s32 maxLength) { // Changed fontSize to u32
|
||
|
||
// Thread-safe translation cache access
|
||
std::string text;
|
||
#ifdef UI_OVERRIDE_PATH
|
||
{
|
||
std::shared_lock<std::shared_mutex> readLock(s_translationCacheMutex);
|
||
auto translatedIt = ult::translationCache.find(originalString);
|
||
if (translatedIt != ult::translationCache.end()) {
|
||
text = translatedIt->second;
|
||
} else {
|
||
// Need to upgrade to write lock
|
||
readLock.unlock();
|
||
std::unique_lock<std::shared_mutex> writeLock(s_translationCacheMutex);
|
||
|
||
// Double-check pattern
|
||
translatedIt = ult::translationCache.find(originalString);
|
||
if (translatedIt != ult::translationCache.end()) {
|
||
text = translatedIt->second;
|
||
} else {
|
||
ult::translationCache[originalString] = originalString;
|
||
text = originalString;
|
||
}
|
||
}
|
||
}
|
||
#else
|
||
text = originalString;
|
||
#endif
|
||
|
||
if (text.size() < 2) return text;
|
||
|
||
// Get ellipsis width using shared cache (now thread-safe)
|
||
static constexpr u32 ellipsisChar = 0x2026;
|
||
std::shared_ptr<FontManager::Glyph> ellipsisGlyph = FontManager::getOrCreateGlyph(ellipsisChar, monospace, fontSize);
|
||
if (!ellipsisGlyph) return text;
|
||
|
||
// Fixed: Use consistent s32 calculation like other functions
|
||
const s32 ellipsisWidth = static_cast<s32>(ellipsisGlyph->xAdvance * ellipsisGlyph->currFontSize);
|
||
const s32 maxWidthWithoutEllipsis = maxLength - ellipsisWidth;
|
||
|
||
if (maxWidthWithoutEllipsis <= 0) {
|
||
return "…"; // If there's no room for text, just return ellipsis
|
||
}
|
||
|
||
// Calculate width incrementally
|
||
s32 currX = 0;
|
||
auto itStr = text.cbegin();
|
||
const auto itStrEnd = text.cend();
|
||
auto lastValidPos = itStr;
|
||
|
||
// Fast ASCII check
|
||
bool isAsciiOnly = true;
|
||
for (unsigned char c : text) {
|
||
if (c > 127) {
|
||
isAsciiOnly = false;
|
||
break;
|
||
}
|
||
}
|
||
|
||
// Move variable declarations outside the loop
|
||
u32 currCharacter;
|
||
ssize_t codepointWidth;
|
||
s32 charWidth;
|
||
size_t bytePos;
|
||
|
||
while (itStr != itStrEnd) {
|
||
// Decode UTF-8 codepoint
|
||
if (isAsciiOnly) {
|
||
currCharacter = static_cast<u32>(*itStr);
|
||
codepointWidth = 1;
|
||
} else {
|
||
codepointWidth = decode_utf8(&currCharacter, reinterpret_cast<const u8*>(&(*itStr)));
|
||
if (codepointWidth <= 0) break;
|
||
}
|
||
|
||
// FontManager::getOrCreateGlyph is now thread-safe
|
||
std::shared_ptr<FontManager::Glyph> glyph = FontManager::getOrCreateGlyph(currCharacter, monospace, fontSize);
|
||
if (!glyph) {
|
||
itStr += codepointWidth;
|
||
continue;
|
||
}
|
||
|
||
// Fixed: Use consistent s32 calculation
|
||
charWidth = static_cast<s32>(glyph->xAdvance * glyph->currFontSize);
|
||
|
||
if (currX + charWidth > maxWidthWithoutEllipsis) {
|
||
// Calculate the byte position for substring
|
||
bytePos = std::distance(text.cbegin(), lastValidPos);
|
||
return text.substr(0, bytePos) + "…";
|
||
}
|
||
|
||
currX += charWidth;
|
||
itStr += codepointWidth;
|
||
lastValidPos = itStr;
|
||
}
|
||
|
||
return text;
|
||
}
|
||
|
||
inline void setLayerPos(u32 x, u32 y) {
|
||
//const float ratio = 1.5;
|
||
//u32 maxX = cfg::ScreenWidth - (int)(ratio * cfg::FramebufferWidth);
|
||
//u32 maxY = cfg::ScreenHeight - (int)(ratio * cfg::FramebufferHeight);
|
||
if (x > cfg::ScreenWidth - (int)(1.5 * cfg::FramebufferWidth) || y > cfg::ScreenHeight - (int)(1.5 * cfg::FramebufferHeight)) {
|
||
return;
|
||
}
|
||
setLayerPosImpl(x, y);
|
||
}
|
||
|
||
void updateLayerSize() {
|
||
const auto [horizontalUnderscanPixels, verticalUnderscanPixels] = getUnderscanPixels();
|
||
|
||
// Recalculate layer dimensions with new underscan values
|
||
cfg::LayerWidth = cfg::ScreenWidth * (float(cfg::FramebufferWidth) / float(cfg::LayerMaxWidth));
|
||
cfg::LayerHeight = cfg::ScreenHeight * (float(cfg::FramebufferHeight) / float(cfg::LayerMaxHeight));
|
||
|
||
// Apply underscan adjustments
|
||
if (ult::DefaultFramebufferWidth == 1280 && ult::DefaultFramebufferHeight == 28) {
|
||
cfg::LayerHeight += cfg::ScreenHeight/720. * verticalUnderscanPixels;
|
||
} else if (ult::correctFrameSize) {
|
||
cfg::LayerWidth += horizontalUnderscanPixels;
|
||
}
|
||
|
||
// Update position if using right alignment
|
||
if (ult::useRightAlignment && ult::correctFrameSize) {
|
||
ult::layerEdge = (1280 - 448);
|
||
}
|
||
|
||
// Update the existing layer with new dimensions
|
||
viSetLayerSize(&this->m_layer, cfg::LayerWidth, cfg::LayerHeight);
|
||
|
||
// Update position if using right alignment
|
||
if (ult::useRightAlignment && ult::correctFrameSize) {
|
||
viSetLayerPosition(&this->m_layer, 1280-32 - horizontalUnderscanPixels, 0);
|
||
viSetLayerSize(&this->m_layer, cfg::LayerWidth, cfg::LayerHeight);
|
||
viSetLayerPosition(&this->m_layer, 1280-32 - horizontalUnderscanPixels, 0);
|
||
}
|
||
// ADD THIS: Update position for micro mode bottom positioning
|
||
else if (ult::DefaultFramebufferWidth == 1280 && ult::DefaultFramebufferHeight == 28 && cfg::LayerPosY > 500) {
|
||
// Only adjust if already positioned at bottom (LayerPosY > 500 indicates bottom positioning)
|
||
const u32 targetY = !verticalUnderscanPixels ? 1038 : 1038- (cfg::ScreenHeight/720. * verticalUnderscanPixels) +0.5;
|
||
viSetLayerPosition(&this->m_layer, 0, targetY);
|
||
viSetLayerSize(&this->m_layer, cfg::LayerWidth, cfg::LayerHeight);
|
||
viSetLayerPosition(&this->m_layer, 0, targetY);
|
||
}
|
||
}
|
||
|
||
static Renderer& getRenderer() {
|
||
return get();
|
||
}
|
||
|
||
//inline void setLayerPosImpl(u32 x, u32 y) {
|
||
// // Get the underscan pixel values for both horizontal and vertical borders
|
||
// const auto [horizontalUnderscanPixels, verticalUnderscanPixels] = getUnderscanPixels();
|
||
// //int horizontalUnderscanPixels = 0;
|
||
//
|
||
// //ult::useRightAlignment = (ult::parseValueFromIniSection(ult::ULTRAHAND_CONFIG_INI_PATH, ult::ULTRAHAND_PROJECT_NAME, "right_alignment") == ult::TRUE_STR);
|
||
// //cfg::LayerPosX = 1280-32;
|
||
// cfg::LayerPosX = 0;
|
||
// cfg::LayerPosY = 0;
|
||
// cfg::FramebufferWidth = ult::DefaultFramebufferWidth;
|
||
// cfg::FramebufferHeight = ult::DefaultFramebufferHeight;
|
||
//
|
||
// //ult::correctFrameSize = (cfg::FramebufferWidth == 448 && cfg::FramebufferHeight == 720); // for detecting the correct Overlay display size
|
||
// if (ult::useRightAlignment && ult::correctFrameSize) {
|
||
// cfg::LayerPosX = 1280-32 - horizontalUnderscanPixels;
|
||
// ult::layerEdge = (1280-448);
|
||
// }
|
||
//
|
||
// cfg::LayerPosX += x;
|
||
// cfg::LayerPosY += y;
|
||
// ASSERT_FATAL(viSetLayerPosition(&this->m_layer, cfg::LayerPosX, cfg::LayerPosY));
|
||
//}
|
||
|
||
inline void setLayerPosImpl(u32 x, u32 y) {
|
||
// Get the underscan pixel values for both horizontal and vertical borders
|
||
//const auto [horizontalUnderscanPixels, verticalUnderscanPixels] = getUnderscanPixels();
|
||
|
||
// Simply set the position to what was requested - no automatic right alignment
|
||
cfg::LayerPosX = x;
|
||
cfg::LayerPosY = y;
|
||
|
||
ASSERT_FATAL(viSetLayerPosition(&this->m_layer, cfg::LayerPosX, cfg::LayerPosY));
|
||
}
|
||
|
||
|
||
#if USING_WIDGET_DIRECTIVE
|
||
// Method to draw clock, temperatures, and battery percentage
|
||
inline void drawWidget() {
|
||
static time_t lastTimeUpdate = 0;
|
||
static char timeStr[20];
|
||
static char PCB_temperatureStr[10];
|
||
static char SOC_temperatureStr[10];
|
||
static char chargeString[6];
|
||
static time_t lastSensorUpdate = 0;
|
||
|
||
const bool showAnyWidget = !(ult::hideBattery && ult::hidePCBTemp && ult::hideSOCTemp && ult::hideClock);
|
||
|
||
// Draw separator and backdrop if showing any widget
|
||
if (showAnyWidget) {
|
||
drawRect(239, 15 + 2 - 2, 1, 64 + 2, topSeparatorColor);
|
||
if (!ult::hideWidgetBackdrop) {
|
||
drawUniformRoundedRect(
|
||
247, 15 + 2 - 2,
|
||
(ult::extendedWidgetBackdrop
|
||
? tsl::cfg::FramebufferWidth - 255
|
||
: tsl::cfg::FramebufferWidth - 215),
|
||
64 + 2, widgetBackdropColor
|
||
);
|
||
}
|
||
}
|
||
|
||
// Calculate base Y offset
|
||
size_t y_offset = ((ult::hideBattery && ult::hidePCBTemp && ult::hideSOCTemp) || ult::hideClock)
|
||
? (55 + 2 - 1)
|
||
: (44 + 2 - 1);
|
||
|
||
// Constants for centering calculations
|
||
const int backdropCenterX = 247 + ((tsl::cfg::FramebufferWidth - 255) >> 1);
|
||
|
||
time_t currentTime = time(nullptr);
|
||
|
||
// Draw clock
|
||
if (!ult::hideClock) {
|
||
if (currentTime != lastTimeUpdate || ult::languageWasChanged.load(std::memory_order_acquire)) {
|
||
strftime(timeStr, sizeof(timeStr), ult::datetimeFormat.c_str(), localtime(¤tTime));
|
||
ult::localizeTimeStr(timeStr);
|
||
lastTimeUpdate = currentTime;
|
||
}
|
||
|
||
const int timeWidth = getTextDimensions(timeStr, false, 20).first;
|
||
|
||
if (ult::centerWidgetAlignment) {
|
||
// Centered alignment
|
||
drawString(timeStr, false, backdropCenterX - (timeWidth >> 1), y_offset, 20, clockColor);
|
||
} else {
|
||
// Right alignment
|
||
drawString(timeStr, false, tsl::cfg::FramebufferWidth - timeWidth - 25, y_offset, 20, clockColor);
|
||
}
|
||
|
||
y_offset += 22;
|
||
}
|
||
|
||
// Update sensor data every second
|
||
if ((currentTime - lastSensorUpdate) >= 1) {
|
||
if (!ult::hideSOCTemp) {
|
||
float socTemp = 0.0f;
|
||
ult::ReadSocTemperature(&socTemp);
|
||
ult::SOC_temperature.store(socTemp, std::memory_order_release);
|
||
snprintf(
|
||
SOC_temperatureStr, sizeof(SOC_temperatureStr),
|
||
"%d°C",
|
||
static_cast<int>(round(ult::SOC_temperature.load(std::memory_order_acquire)))
|
||
);
|
||
}
|
||
|
||
if (!ult::hidePCBTemp) {
|
||
float pcbTemp = 0.0f;
|
||
ult::ReadPcbTemperature(&pcbTemp);
|
||
ult::PCB_temperature.store(pcbTemp, std::memory_order_release);
|
||
snprintf(
|
||
PCB_temperatureStr, sizeof(PCB_temperatureStr),
|
||
"%d°C",
|
||
static_cast<int>(round(ult::PCB_temperature.load(std::memory_order_acquire)))
|
||
);
|
||
}
|
||
|
||
if (!ult::hideBattery) {
|
||
uint32_t bc = 0;
|
||
bool charging = false;
|
||
ult::powerGetDetails(&bc, &charging);
|
||
bc = std::min(bc, 100U);
|
||
ult::batteryCharge.store(bc, std::memory_order_release);
|
||
ult::isCharging.store(charging, std::memory_order_release);
|
||
snprintf(chargeString, sizeof(chargeString), "%u%%", bc);
|
||
}
|
||
|
||
lastSensorUpdate = currentTime;
|
||
}
|
||
|
||
if (ult::centerWidgetAlignment) {
|
||
// CENTERED ALIGNMENT
|
||
int totalWidth = 0;
|
||
int socWidth = 0, pcbWidth = 0, chargeWidth = 0;
|
||
bool hasMultiple = false;
|
||
|
||
const float socTemp = ult::SOC_temperature.load(std::memory_order_acquire);
|
||
const float pcbTemp = ult::PCB_temperature.load(std::memory_order_acquire);
|
||
const uint32_t batteryCharge = ult::batteryCharge.load(std::memory_order_acquire);
|
||
const bool charging = ult::isCharging.load(std::memory_order_acquire);
|
||
|
||
if (!ult::hideSOCTemp && socTemp > 0.0f) {
|
||
socWidth = getTextDimensions(SOC_temperatureStr, false, 20).first;
|
||
totalWidth += socWidth;
|
||
hasMultiple = true;
|
||
}
|
||
if (!ult::hidePCBTemp && pcbTemp > 0.0f) {
|
||
pcbWidth = getTextDimensions(PCB_temperatureStr, false, 20).first;
|
||
if (hasMultiple) totalWidth += 5;
|
||
totalWidth += pcbWidth;
|
||
hasMultiple = true;
|
||
}
|
||
if (!ult::hideBattery && batteryCharge > 0) {
|
||
chargeWidth = getTextDimensions(chargeString, false, 20).first;
|
||
if (hasMultiple) totalWidth += 5;
|
||
totalWidth += chargeWidth;
|
||
}
|
||
|
||
int currentX = backdropCenterX - (totalWidth >> 1);
|
||
if (socWidth > 0) {
|
||
drawString(
|
||
SOC_temperatureStr, false, currentX, y_offset, 20,
|
||
ult::dynamicWidgetColors
|
||
? tsl::GradientColor(socTemp)
|
||
: temperatureColor
|
||
);
|
||
currentX += socWidth + 5;
|
||
}
|
||
if (pcbWidth > 0) {
|
||
drawString(
|
||
PCB_temperatureStr, false, currentX, y_offset, 20,
|
||
ult::dynamicWidgetColors
|
||
? tsl::GradientColor(pcbTemp)
|
||
: temperatureColor
|
||
);
|
||
currentX += pcbWidth + 5;
|
||
}
|
||
if (chargeWidth > 0) {
|
||
const Color batteryColorToUse = charging
|
||
? batteryChargingColor
|
||
: (batteryCharge < 20 ? batteryLowColor : batteryColor);
|
||
drawString(chargeString, false, currentX, y_offset, 20, batteryColorToUse);
|
||
}
|
||
|
||
} else {
|
||
// RIGHT ALIGNMENT
|
||
int chargeWidth = 0, pcbWidth = 0, socWidth = 0;
|
||
const float pcbTemp = ult::PCB_temperature.load(std::memory_order_acquire);
|
||
const float socTemp = ult::SOC_temperature.load(std::memory_order_acquire);
|
||
const uint32_t batteryCharge = ult::batteryCharge.load(std::memory_order_acquire);
|
||
const bool charging = ult::isCharging.load(std::memory_order_acquire);
|
||
|
||
if (!ult::hideBattery && batteryCharge > 0) {
|
||
const Color batteryColorToUse = charging
|
||
? batteryChargingColor
|
||
: (batteryCharge < 20 ? batteryLowColor : batteryColor);
|
||
chargeWidth = getTextDimensions(chargeString, false, 20).first;
|
||
drawString(
|
||
chargeString, false,
|
||
tsl::cfg::FramebufferWidth - chargeWidth - 25,
|
||
y_offset, 20, batteryColorToUse
|
||
);
|
||
}
|
||
|
||
int offset = 0;
|
||
if (!ult::hidePCBTemp && pcbTemp > 0.0f) {
|
||
if (!ult::hideBattery) offset -= 5;
|
||
pcbWidth = getTextDimensions(PCB_temperatureStr, false, 20).first;
|
||
drawString(
|
||
PCB_temperatureStr, false,
|
||
tsl::cfg::FramebufferWidth + offset - pcbWidth - chargeWidth - 25,
|
||
y_offset, 20,
|
||
ult::dynamicWidgetColors
|
||
? tsl::GradientColor(pcbTemp)
|
||
: defaultTextColor
|
||
);
|
||
}
|
||
if (!ult::hideSOCTemp && socTemp > 0.0f) {
|
||
if (!ult::hidePCBTemp || !ult::hideBattery) offset -= 5;
|
||
socWidth = getTextDimensions(SOC_temperatureStr, false, 20).first;
|
||
drawString(
|
||
SOC_temperatureStr, false,
|
||
tsl::cfg::FramebufferWidth + offset - socWidth - pcbWidth - chargeWidth - 25,
|
||
y_offset, 20,
|
||
ult::dynamicWidgetColors
|
||
? tsl::GradientColor(socTemp)
|
||
: defaultTextColor
|
||
);
|
||
}
|
||
}
|
||
}
|
||
#endif
|
||
|
||
// Single unified glyph cache for all text operations
|
||
//inline static std::unordered_map<u64, Glyph> s_unifiedGlyphCache;
|
||
|
||
// Helper to select appropriate font for a character
|
||
inline std::shared_ptr<FontManager::Glyph> getOrCreateGlyph(u32 character, bool monospace, u32 fontSize) {
|
||
return FontManager::getOrCreateGlyph(character, monospace, fontSize);
|
||
}
|
||
|
||
inline stbtt_fontinfo* selectFontForCharacter(u32 character) {
|
||
return FontManager::selectFontForCharacter(character);
|
||
}
|
||
|
||
// Optimized glyph rendering
|
||
inline void renderGlyph(std::shared_ptr<FontManager::Glyph> glyph, float x, float y, const Color& color) {
|
||
if (!glyph->glyphBmp || color.a == 0) return;
|
||
|
||
const s32 xPos = static_cast<s32>(x + glyph->bounds[0]);
|
||
const s32 yPos = static_cast<s32>(y + glyph->bounds[1]);
|
||
|
||
// Quick bounds check
|
||
if (xPos >= cfg::FramebufferWidth || yPos >= cfg::FramebufferHeight ||
|
||
xPos + glyph->width <= 0 || yPos + glyph->height <= 0) return;
|
||
|
||
// Calculate clipping
|
||
const s32 startX = std::max(0, -xPos);
|
||
const s32 startY = std::max(0, -yPos);
|
||
const s32 endX = std::min(glyph->width, static_cast<s32>(cfg::FramebufferWidth) - xPos);
|
||
const s32 endY = std::min(glyph->height, static_cast<s32>(cfg::FramebufferHeight) - yPos);
|
||
|
||
// Move variable declarations outside loops
|
||
const s32 simdEnd = std::min(endX, (startX + 7) & ~7);
|
||
s32 bmpX;
|
||
uint8_t alpha;
|
||
s32 pixelX;
|
||
//Color tmpColor = {0};
|
||
|
||
// Render with optimized inner loop
|
||
const uint8_t* bmpPtr = glyph->glyphBmp + startY * glyph->width;
|
||
for (s32 bmpY = startY; bmpY < endY; ++bmpY) {
|
||
const s32 pixelY = yPos + bmpY;
|
||
bmpX = startX;
|
||
|
||
// Process 8 pixels at once
|
||
for (; bmpX < simdEnd; ++bmpX) {
|
||
alpha = bmpPtr[bmpX] >> 4;
|
||
if (alpha) {
|
||
pixelX = xPos + bmpX;
|
||
if (alpha == 0xF) {
|
||
this->setPixel(pixelX, pixelY, color);
|
||
} else {
|
||
this->setPixelBlendDst(pixelX, pixelY, Color(color.r, color.g, color.b, alpha));
|
||
}
|
||
}
|
||
}
|
||
|
||
// Process remaining pixels
|
||
for (; bmpX < endX; ++bmpX) {
|
||
alpha = bmpPtr[bmpX] >> 4;
|
||
if (alpha) {
|
||
pixelX = xPos + bmpX;
|
||
if (alpha == 0xF) {
|
||
this->setPixel(pixelX, pixelY, color);
|
||
} else {
|
||
this->setPixelBlendDst(pixelX, pixelY, Color(color.r, color.g, color.b, alpha));
|
||
}
|
||
}
|
||
}
|
||
bmpPtr += glyph->width;
|
||
}
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Adds the layer from screenshot and recording stacks
|
||
*/
|
||
inline void addScreenshotStacks() {
|
||
ASSERT_FATAL(tsl::hlp::viAddToLayerStack(&this->m_layer, ViLayerStack_Screenshot));
|
||
ASSERT_FATAL(tsl::hlp::viAddToLayerStack(&this->m_layer, ViLayerStack_Recording));
|
||
screenshotsAreDisabled = false;
|
||
}
|
||
|
||
/**
|
||
* @brief Removes the layer from screenshot and recording stacks
|
||
*/
|
||
inline void removeScreenshotStacks() {
|
||
ASSERT_FATAL(tsl::hlp::viRemoveFromLayerStack(&this->m_layer, ViLayerStack_Screenshot));
|
||
ASSERT_FATAL(tsl::hlp::viRemoveFromLayerStack(&this->m_layer, ViLayerStack_Recording));
|
||
screenshotsAreDisabled = true;
|
||
}
|
||
|
||
private:
|
||
Renderer() {
|
||
updateDrawFunction();
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the opacity of the layer
|
||
*
|
||
* @param opacity Opacity
|
||
*/
|
||
inline static void setOpacity(float opacity) {
|
||
opacity = std::clamp(opacity, 0.0F, 1.0F);
|
||
|
||
Renderer::s_opacity = opacity;
|
||
}
|
||
|
||
bool m_initialized = false;
|
||
ViDisplay m_display;
|
||
ViLayer m_layer;
|
||
Event m_vsyncEvent;
|
||
|
||
NWindow m_window;
|
||
Framebuffer m_framebuffer;
|
||
void *m_currentFramebuffer = nullptr;
|
||
|
||
std::stack<ScissoringConfig> m_scissoringStack;
|
||
|
||
|
||
static inline float s_opacity = 1.0F;
|
||
|
||
|
||
/**
|
||
* @brief Get the current framebuffer address
|
||
*
|
||
* @return Framebuffer address
|
||
*/
|
||
inline void* getCurrentFramebuffer() {
|
||
return this->m_currentFramebuffer;
|
||
}
|
||
|
||
/**
|
||
* @brief Get the next framebuffer address
|
||
*
|
||
* @return Next framebuffer address
|
||
*/
|
||
inline void* getNextFramebuffer() {
|
||
return static_cast<u8*>(this->m_framebuffer.buf) + this->getNextFramebufferSlot() * this->getFramebufferSize();
|
||
}
|
||
|
||
/**
|
||
* @brief Get the framebuffer size
|
||
*
|
||
* @return Framebuffer size
|
||
*/
|
||
inline size_t getFramebufferSize() {
|
||
return this->m_framebuffer.fb_size;
|
||
}
|
||
|
||
/**
|
||
* @brief Get the number of framebuffers in use
|
||
*
|
||
* @return Number of framebuffers
|
||
*/
|
||
inline size_t getFramebufferCount() {
|
||
return this->m_framebuffer.num_fbs;
|
||
}
|
||
|
||
/**
|
||
* @brief Get the currently used framebuffer's slot
|
||
*
|
||
* @return Slot
|
||
*/
|
||
inline u8 getCurrentFramebufferSlot() {
|
||
return this->m_window.cur_slot;
|
||
}
|
||
|
||
/**
|
||
* @brief Get the next framebuffer's slot
|
||
*
|
||
* @return Next slot
|
||
*/
|
||
inline u8 getNextFramebufferSlot() {
|
||
return (this->getCurrentFramebufferSlot() + 1) % this->getFramebufferCount();
|
||
}
|
||
|
||
/**
|
||
* @brief Waits for the vsync event
|
||
*
|
||
*/
|
||
inline void waitForVSync() {
|
||
eventWait(&this->m_vsyncEvent, UINT64_MAX);
|
||
}
|
||
|
||
/**
|
||
* @brief Decodes a x and y coordinate into a offset into the swizzled framebuffer
|
||
*
|
||
* @param x X pos
|
||
* @param y Y Pos
|
||
* @return Offset
|
||
*/
|
||
|
||
inline u32 __attribute__((always_inline)) getPixelOffset(const u32 x, const u32 y) {
|
||
// Check for scissoring boundaries
|
||
if (!this->m_scissoringStack.empty()) {
|
||
const auto& currScissorConfig = this->m_scissoringStack.top();
|
||
if (x < currScissorConfig.x || y < currScissorConfig.y ||
|
||
x >= currScissorConfig.x_max ||
|
||
y >= currScissorConfig.y_max) {
|
||
return UINT32_MAX;
|
||
}
|
||
}
|
||
|
||
// Replace divisions and modulos with bit operations - EXACT same logic
|
||
//return ((((y & 127) >> 4) + ((x >> 5) << 3) + ((y >> 7) * 112)) << 9) + // *512 = <<9
|
||
// (((y & 15) >> 3) << 8) + // ((y % 16) / 8) * 256
|
||
// (((x & 31) >> 4) << 7) + // ((x % 32) / 16) * 128
|
||
// (((y & 7) >> 1) << 5) + // ((y % 8) / 2) * 32
|
||
// (((x & 15) >> 3) << 4) + // ((x % 16) / 8) * 16
|
||
// ((y & 1) << 3) + // (y % 2) * 8
|
||
// (x & 7); // x % 8
|
||
|
||
//return ((((y & 127) >> 4) + ((x >> 5) << 3) + ((y >> 7) * (((cfg::FramebufferWidth / 2) >> 4) << 3))) << 9) + // *512 = <<9
|
||
// (((y & 15) >> 3) << 8) + // ((y % 16) / 8) * 256
|
||
// (((x & 31) >> 4) << 7) + // ((x % 32) / 16) * 128
|
||
// (((y & 7) >> 1) << 5) + // ((y % 8) / 2) * 32
|
||
// (((x & 15) >> 3) << 4) + // ((x % 16) / 8) * 16
|
||
// ((y & 1) << 3) + // (y % 2) * 8
|
||
// (x & 7); // x % 8
|
||
|
||
//return ((((y & 127) >> 4) + ((x >> 5) << 3) + ((y >> 7) * offsetWidthVar)) << 9) + // *512 = <<9
|
||
// (((y & 15) >> 3) << 8) + // ((y % 16) / 8) * 256
|
||
// (((x & 31) >> 4) << 7) + // ((x % 32) / 16) * 128
|
||
// (((y & 7) >> 1) << 5) + // ((y % 8) / 2) * 32
|
||
// (((x & 15) >> 3) << 4) + // ((x % 16) / 8) * 16
|
||
// ((y & 1) << 3) + // (y % 2) * 8
|
||
// (x & 7); // x % 8
|
||
|
||
return ((((y & 127) >> 4) + ((x >> 5) << 3) + ((y >> 7) * offsetWidthVar)) << 9) +
|
||
((y & 8) << 5) + ((x & 16) << 3) + ((y & 6) << 4) +
|
||
((x & 8) << 1) + ((y & 1) << 3) + (x & 7);
|
||
|
||
//const u32 y_hi = y >> 7;
|
||
//const u32 y_mid = (y >> 4) & 7; // bits 4-6 of y
|
||
//const u32 y_lo = y & 15; // bits 0-3 of y
|
||
//
|
||
//const u32 x_hi = x >> 5; // bits 5+ of x
|
||
//const u32 x_lo = x & 31; // bits 0-4 of x
|
||
//
|
||
//return ((y_mid + (x_hi << 3) + (y_hi * offsetWidthVar)) << 9) +
|
||
// ((y_lo >> 3) << 8) + ((x_lo >> 4) << 7) +
|
||
// (((y_lo & 7) >> 1) << 5) + (((x_lo & 15) >> 3) << 4) +
|
||
// ((y_lo & 1) << 3) + (x_lo & 7);
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Initializes the renderer and layers
|
||
*
|
||
*/
|
||
void init() {
|
||
// Get the underscan pixel values for both horizontal and vertical borders
|
||
const auto [horizontalUnderscanPixels, verticalUnderscanPixels] = getUnderscanPixels();
|
||
//int horizontalUnderscanPixels = 0;
|
||
|
||
|
||
ult::useRightAlignment = (ult::parseValueFromIniSection(ult::ULTRAHAND_CONFIG_INI_PATH, ult::ULTRAHAND_PROJECT_NAME, "right_alignment") == ult::TRUE_STR);
|
||
|
||
//cfg::LayerPosX = 1280-32;
|
||
cfg::LayerPosX = 0;
|
||
cfg::LayerPosY = 0;
|
||
cfg::FramebufferWidth = ult::DefaultFramebufferWidth;
|
||
cfg::FramebufferHeight = ult::DefaultFramebufferHeight;
|
||
|
||
offsetWidthVar = (((cfg::FramebufferWidth / 2) >> 4) << 3);
|
||
|
||
ult::correctFrameSize = (cfg::FramebufferWidth == 448 && cfg::FramebufferHeight == 720); // for detecting the correct Overlay display size
|
||
if (ult::useRightAlignment && ult::correctFrameSize) {
|
||
cfg::LayerPosX = 1280-32 - horizontalUnderscanPixels;
|
||
ult::layerEdge = (1280-448);
|
||
}
|
||
|
||
cfg::LayerWidth = cfg::ScreenWidth * (float(cfg::FramebufferWidth) / float(cfg::LayerMaxWidth));
|
||
cfg::LayerHeight = cfg::ScreenHeight * (float(cfg::FramebufferHeight) / float(cfg::LayerMaxHeight));
|
||
|
||
// Apply underscanning offset
|
||
if (ult::DefaultFramebufferWidth == 1280 && ult::DefaultFramebufferHeight == 28) // for status monitor micro mode
|
||
cfg::LayerHeight += cfg::ScreenHeight/720. *verticalUnderscanPixels;
|
||
else if (ult::correctFrameSize)
|
||
cfg::LayerWidth += horizontalUnderscanPixels;
|
||
|
||
// NEW: Scale down to 1/4 size (0.5x in each dimension)
|
||
//static constexpr float scaleFactor = 0.5f;
|
||
//cfg::LayerWidth *= scaleFactor;
|
||
//cfg::LayerHeight *= scaleFactor;
|
||
|
||
if (this->m_initialized)
|
||
return;
|
||
|
||
//s32 layerZ = 0;
|
||
|
||
tsl::hlp::doWithSmSession([this, horizontalUnderscanPixels]{
|
||
|
||
ASSERT_FATAL(viInitialize(ViServiceType_Manager));
|
||
ASSERT_FATAL(viOpenDefaultDisplay(&this->m_display));
|
||
ASSERT_FATAL(viGetDisplayVsyncEvent(&this->m_display, &this->m_vsyncEvent));
|
||
ASSERT_FATAL(viCreateManagedLayer(&this->m_display, static_cast<ViLayerFlags>(0), 0, &__nx_vi_layer_id));
|
||
ASSERT_FATAL(viCreateLayer(&this->m_display, &this->m_layer));
|
||
ASSERT_FATAL(viSetLayerScalingMode(&this->m_layer, ViScalingMode_FitToLayer));
|
||
|
||
//if (s32 layerZ = 0; R_SUCCEEDED(viGetZOrderCountMax(&this->m_display, &layerZ)) && layerZ > 0)
|
||
// ASSERT_FATAL(viSetLayerZ(&this->m_layer, layerZ));
|
||
|
||
if (horizontalUnderscanPixels == 0) {
|
||
s32 layerZ = 0;
|
||
if (R_SUCCEEDED(viGetZOrderCountMax(&this->m_display, &layerZ)) && layerZ > 0) {
|
||
ASSERT_FATAL(viSetLayerZ(&this->m_layer, layerZ));
|
||
}
|
||
else {
|
||
ASSERT_FATAL(viSetLayerZ(&this->m_layer, 255)); // max value 255 as fallback
|
||
}
|
||
} else {
|
||
ASSERT_FATAL(viSetLayerZ(&this->m_layer, 34)); // 34 is the edge for underscanning
|
||
}
|
||
|
||
ASSERT_FATAL(tsl::hlp::viAddToLayerStack(&this->m_layer, ViLayerStack_Default));
|
||
ASSERT_FATAL(tsl::hlp::viAddToLayerStack(&this->m_layer, ViLayerStack_Screenshot));
|
||
ASSERT_FATAL(tsl::hlp::viAddToLayerStack(&this->m_layer, ViLayerStack_Recording));
|
||
ASSERT_FATAL(tsl::hlp::viAddToLayerStack(&this->m_layer, ViLayerStack_Arbitrary));
|
||
ASSERT_FATAL(tsl::hlp::viAddToLayerStack(&this->m_layer, ViLayerStack_LastFrame));
|
||
ASSERT_FATAL(tsl::hlp::viAddToLayerStack(&this->m_layer, ViLayerStack_Null));
|
||
ASSERT_FATAL(tsl::hlp::viAddToLayerStack(&this->m_layer, ViLayerStack_ApplicationForDebug));
|
||
ASSERT_FATAL(tsl::hlp::viAddToLayerStack(&this->m_layer, ViLayerStack_Lcd));
|
||
|
||
ASSERT_FATAL(viSetLayerSize(&this->m_layer, cfg::LayerWidth, cfg::LayerHeight));
|
||
ASSERT_FATAL(viSetLayerPosition(&this->m_layer, cfg::LayerPosX, cfg::LayerPosY));
|
||
ASSERT_FATAL(nwindowCreateFromLayer(&this->m_window, &this->m_layer));
|
||
ASSERT_FATAL(framebufferCreate(&this->m_framebuffer, &this->m_window, cfg::FramebufferWidth, cfg::FramebufferHeight, PIXEL_FORMAT_RGBA_4444, 2));
|
||
ASSERT_FATAL(setInitialize());
|
||
ASSERT_FATAL(this->initFonts());
|
||
setExit();
|
||
});
|
||
|
||
this->m_initialized = true;
|
||
}
|
||
|
||
/**
|
||
* @brief Exits the renderer and layer
|
||
*
|
||
*/
|
||
void exit() {
|
||
if (!this->m_initialized)
|
||
return;
|
||
|
||
// Cleanup shared font manager
|
||
FontManager::cleanup();
|
||
|
||
framebufferClose(&this->m_framebuffer);
|
||
nwindowClose(&this->m_window);
|
||
viDestroyManagedLayer(&this->m_layer);
|
||
viCloseDisplay(&this->m_display);
|
||
eventClose(&this->m_vsyncEvent);
|
||
viExit();
|
||
}
|
||
|
||
/**
|
||
* @brief Initializes Nintendo's shared fonts. Default and Extended
|
||
*
|
||
* @return Result
|
||
*/
|
||
Result initFonts() {
|
||
static PlFontData stdFontData, localFontData, extFontData;
|
||
|
||
// Nintendo's default font
|
||
TSL_R_TRY(plGetSharedFontByType(&stdFontData, PlSharedFontType_Standard));
|
||
|
||
u8 *fontBuffer = reinterpret_cast<u8*>(stdFontData.address);
|
||
stbtt_InitFont(&this->m_stdFont, fontBuffer, stbtt_GetFontOffsetForIndex(fontBuffer, 0));
|
||
|
||
u64 languageCode;
|
||
if (R_SUCCEEDED(setGetSystemLanguage(&languageCode))) {
|
||
// Check if need localization font
|
||
SetLanguage setLanguage;
|
||
TSL_R_TRY(setMakeLanguage(languageCode, &setLanguage));
|
||
this->m_hasLocalFont = true;
|
||
switch (setLanguage) {
|
||
case SetLanguage_ZHCN:
|
||
case SetLanguage_ZHHANS:
|
||
TSL_R_TRY(plGetSharedFontByType(&localFontData, PlSharedFontType_ChineseSimplified));
|
||
break;
|
||
case SetLanguage_KO:
|
||
TSL_R_TRY(plGetSharedFontByType(&localFontData, PlSharedFontType_KO));
|
||
break;
|
||
case SetLanguage_ZHTW:
|
||
case SetLanguage_ZHHANT:
|
||
TSL_R_TRY(plGetSharedFontByType(&localFontData, PlSharedFontType_ChineseTraditional));
|
||
break;
|
||
default:
|
||
this->m_hasLocalFont = false;
|
||
break;
|
||
}
|
||
|
||
if (this->m_hasLocalFont) {
|
||
fontBuffer = reinterpret_cast<u8*>(localFontData.address);
|
||
stbtt_InitFont(&this->m_localFont, fontBuffer, stbtt_GetFontOffsetForIndex(fontBuffer, 0));
|
||
}
|
||
}
|
||
|
||
// Nintendo's extended font containing a bunch of icons
|
||
TSL_R_TRY(plGetSharedFontByType(&extFontData, PlSharedFontType_NintendoExt));
|
||
|
||
fontBuffer = reinterpret_cast<u8*>(extFontData.address);
|
||
stbtt_InitFont(&this->m_extFont, fontBuffer, stbtt_GetFontOffsetForIndex(fontBuffer, 0));
|
||
|
||
// Initialize the shared font manager
|
||
FontManager::initializeFonts(&this->m_stdFont, &this->m_localFont,
|
||
&this->m_extFont, this->m_hasLocalFont);
|
||
|
||
return 0;
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Start a new frame
|
||
* @warning Don't call this more than once before calling \ref endFrame
|
||
*/
|
||
inline void startFrame() {
|
||
this->m_currentFramebuffer = framebufferBegin(&this->m_framebuffer, nullptr);
|
||
}
|
||
|
||
/**
|
||
* @brief End the current frame
|
||
* @warning Don't call this before calling \ref startFrame once
|
||
*/
|
||
inline void endFrame() {
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
// Check if rendering is active by testing if the stop event is NOT signaled
|
||
if (isRendering) { // Returns true if event is signaled
|
||
memcpy(this->getNextFramebuffer(), this->getCurrentFramebuffer(), this->getFramebufferSize());
|
||
|
||
static u32 lastFPS = 0;
|
||
static u64 cachedIntervalNs = 0;
|
||
|
||
if (__builtin_expect(TeslaFPS != lastFPS, 0)) { // Hint: FPS changes are rare
|
||
cachedIntervalNs = 1000*1000*1000ULL / TeslaFPS;
|
||
lastFPS = TeslaFPS;
|
||
}
|
||
|
||
// Wait for frame interval or rendering stop event
|
||
leventWait(&renderingStopEvent, cachedIntervalNs);
|
||
}
|
||
#endif
|
||
this->waitForVSync();
|
||
framebufferEnd(&this->m_framebuffer);
|
||
this->m_currentFramebuffer = nullptr;
|
||
if (tsl::clearGlyphCacheNow.load(std::memory_order_acquire)) {
|
||
tsl::clearGlyphCacheNow.store(false, std::memory_order_release);
|
||
tsl::gfx::FontManager::clearCache(); // exclusive clear
|
||
}
|
||
}
|
||
|
||
|
||
};
|
||
|
||
static std::pair<int, int> getUnderscanPixels() {
|
||
if (!ult::consoleIsDocked()) {
|
||
return {0, 0};
|
||
}
|
||
|
||
// Retrieve the TV settings
|
||
SetSysTvSettings tvSettings;
|
||
Result res = setsysGetTvSettings(&tvSettings);
|
||
if (R_FAILED(res)) {
|
||
// Handle error: return default underscan or log error
|
||
return {0, 0};
|
||
}
|
||
|
||
// The underscan value might not be a percentage, we need to interpret it correctly
|
||
const u32 underscanValue = tvSettings.underscan;
|
||
|
||
// Convert the underscan value to a fraction. Assuming 0 means no underscan and larger values represent
|
||
// greater underscan. Adjust this formula based on actual observed behavior or documentation.
|
||
const float underscanPercentage = 1.0f - (underscanValue / 100.0f);
|
||
|
||
// Original dimensions of the full 720p image (1280x720)
|
||
const float originalWidth = 1280;
|
||
const float originalHeight = 720;
|
||
|
||
// Adjust the width and height based on the underscan percentage
|
||
const float adjustedWidth = (originalWidth * underscanPercentage);
|
||
const float adjustedHeight = (originalHeight * underscanPercentage);
|
||
|
||
// Calculate the underscan in pixels (left/right and top/bottom)
|
||
const int horizontalUnderscanPixels = (originalWidth - adjustedWidth);
|
||
const int verticalUnderscanPixels = (originalHeight - adjustedHeight);
|
||
|
||
return {horizontalUnderscanPixels, verticalUnderscanPixels};
|
||
}
|
||
|
||
}
|
||
|
||
|
||
// Elements
|
||
|
||
namespace elm {
|
||
|
||
enum class TouchEvent {
|
||
Touch,
|
||
Hold,
|
||
Scroll,
|
||
Release,
|
||
None
|
||
};
|
||
|
||
/**
|
||
* @brief The top level Element of the libtesla UI library
|
||
* @note When creating your own elements, extend from this or one of it's sub classes
|
||
*/
|
||
class Element {
|
||
public:
|
||
|
||
Element() {}
|
||
virtual ~Element() {
|
||
m_clickListener = {}; // frees captures immediately
|
||
}
|
||
|
||
bool m_isTable = false; // Default to false for non-table elements
|
||
bool m_isItem = true;
|
||
|
||
|
||
u64 t_ns; // Changed from chrono::duration to nanoseconds
|
||
u8 saturation;
|
||
float progress;
|
||
|
||
s32 x, y;
|
||
s32 amplitude;
|
||
u64 m_animationStartTime; // Changed from chrono::time_point to nanoseconds
|
||
|
||
virtual bool isTable() const {
|
||
return m_isTable;
|
||
}
|
||
|
||
virtual bool isItem() const {
|
||
return m_isItem;
|
||
}
|
||
|
||
/**
|
||
* @brief Handles focus requesting
|
||
* @note This function should return the element to focus.
|
||
* When this element should be focused, return `this`.
|
||
* When one of it's child should be focused, return `this->child->requestFocus(oldFocus, direction)`
|
||
* When this element is not focusable, return `nullptr`
|
||
*
|
||
* @param oldFocus Previously focused element
|
||
* @param direction Direction in which focus moved. \ref FocusDirection::None is passed for the initial load
|
||
* @return Element to focus
|
||
*/
|
||
virtual inline Element* requestFocus(Element *oldFocus, FocusDirection direction) {
|
||
return nullptr;
|
||
}
|
||
|
||
/**
|
||
* @brief Function called when a joycon button got pressed
|
||
*
|
||
* @param keys Keys pressed in the last frame
|
||
* @return true when button press has been consumed
|
||
* @return false when button press should be passed on to the parent
|
||
*/
|
||
virtual inline bool onClick(u64 keys) {
|
||
return m_clickListener(keys);
|
||
}
|
||
|
||
/**
|
||
* @brief Called once per frame with the latest HID inputs
|
||
*
|
||
* @param keysDown Buttons pressed in the last frame
|
||
* @param keysHeld Buttons held down longer than one frame
|
||
* @param touchInput Last touch position
|
||
* @param leftJoyStick Left joystick position
|
||
* @param rightJoyStick Right joystick position
|
||
* @return Weather or not the input has been consumed
|
||
*/
|
||
virtual inline bool handleInput(u64 keysDown, u64 keysHeld, const HidTouchState &touchPos, HidAnalogStickState joyStickPosLeft, HidAnalogStickState joyStickPosRight) {
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* @brief Function called when the element got touched
|
||
* @todo Not yet implemented
|
||
*
|
||
* @param x X pos
|
||
* @param y Y pos
|
||
* @return true when touch input has been consumed
|
||
* @return false when touch input should be passed on to the parent
|
||
*/
|
||
virtual inline bool onTouch(TouchEvent event, s32 currX, s32 currY, s32 prevX, s32 prevY, s32 initialX, s32 initialY) {
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* @brief Called once per frame to draw the element
|
||
* @warning Do not call this yourself. Use \ref Element::frame(gfx::Renderer *renderer)
|
||
*
|
||
* @param renderer Renderer
|
||
*/
|
||
virtual void draw(gfx::Renderer *renderer) = 0;
|
||
|
||
/**
|
||
* @brief Called when the underlying Gui gets created and after calling \ref Gui::invalidate() to calculate positions and boundaries of the element
|
||
* @warning Do not call this yourself. Use \ref Element::invalidate()
|
||
*
|
||
* @param parentX Parent X pos
|
||
* @param parentY Parent Y pos
|
||
* @param parentWidth Parent Width
|
||
* @param parentHeight Parent Height
|
||
*/
|
||
virtual inline void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) = 0;
|
||
|
||
/**
|
||
* @brief Draws highlighting and the element itself
|
||
* @note When drawing children of a element in \ref Element::draw(gfx::Renderer *renderer), use `this->child->frame(renderer)` instead of calling draw directly
|
||
*
|
||
* @param renderer
|
||
*/
|
||
void inline frame(gfx::Renderer *renderer) {
|
||
|
||
if (this->m_focused) {
|
||
renderer->enableScissoring(0, ult::activeHeaderHeight, tsl::cfg::FramebufferWidth, tsl::cfg::FramebufferHeight-73-ult::activeHeaderHeight);
|
||
this->drawFocusBackground(renderer);
|
||
this->drawHighlight(renderer);
|
||
renderer->disableScissoring();
|
||
}
|
||
|
||
this->draw(renderer);
|
||
}
|
||
|
||
/**
|
||
* @brief Forces a layout recreation of a element
|
||
*
|
||
*/
|
||
void inline invalidate() {
|
||
const auto& parent = this->getParent();
|
||
|
||
if (parent == nullptr)
|
||
this->layout(0, 0, cfg::FramebufferWidth, cfg::FramebufferHeight);
|
||
else
|
||
this->layout(ELEMENT_BOUNDS(parent));
|
||
}
|
||
|
||
/**
|
||
* @brief Shake the highlight in the given direction to signal that the focus cannot move there
|
||
*
|
||
* @param direction Direction to shake highlight in
|
||
*/
|
||
void inline shakeHighlight(FocusDirection direction) {
|
||
this->m_highlightShaking = true;
|
||
this->m_highlightShakingDirection = direction;
|
||
this->m_highlightShakingStartTime = armTicksToNs(armGetSystemTick()); // Changed
|
||
}
|
||
|
||
/**
|
||
* @brief Triggers the blue click animation to signal a element has been clicked on
|
||
*
|
||
*/
|
||
void inline triggerClickAnimation() {
|
||
this->m_clickAnimationProgress = tsl::style::ListItemHighlightLength;
|
||
this->m_animationStartTime = armTicksToNs(armGetSystemTick()); // Changed
|
||
}
|
||
|
||
|
||
|
||
/**
|
||
* @brief Resets the click animation progress, canceling the animation
|
||
*/
|
||
void inline resetClickAnimation() {
|
||
this->m_clickAnimationProgress = 0;
|
||
}
|
||
|
||
/**
|
||
* @brief Draws the blue highlight animation when clicking on a button
|
||
* @note Override this if you have a element that e.g requires a non-rectangular animation or a different color
|
||
*
|
||
* @param renderer Renderer
|
||
*/
|
||
virtual void drawClickAnimation(gfx::Renderer *renderer) {
|
||
if (!m_isItem)
|
||
return;
|
||
if (ult::useSelectionBG) {
|
||
if (ult::expandedMemory)
|
||
renderer->drawRectMultiThreaded(this->getX() + x + 4, this->getY() + y, this->getWidth() - 8, this->getHeight(), aWithOpacity(selectionBGColor)); // CUSTOM MODIFICATION
|
||
else
|
||
renderer->drawRect(this->getX() + x + 4, this->getY() + y, this->getWidth() - 8, this->getHeight(), aWithOpacity(selectionBGColor));
|
||
}
|
||
|
||
saturation = tsl::style::ListItemHighlightSaturation * (float(this->m_clickAnimationProgress) / float(tsl::style::ListItemHighlightLength));
|
||
|
||
Color animColor = {0xF,0xF,0xF,0xF};
|
||
if (invertBGClickColor) {
|
||
const u8 inverted = 15-saturation;
|
||
animColor = {inverted, inverted, inverted, selectionBGColor.a};
|
||
} else {
|
||
animColor = {saturation, saturation, saturation, selectionBGColor.a};
|
||
}
|
||
if (ult::expandedMemory)
|
||
renderer->drawRectMultiThreaded(ELEMENT_BOUNDS(this), aWithOpacity(animColor));
|
||
else
|
||
renderer->drawRect(ELEMENT_BOUNDS(this), aWithOpacity(animColor));
|
||
|
||
// Cache time calculation - only compute once
|
||
static u64 lastTimeUpdate = 0;
|
||
static double cachedProgress = 0.0;
|
||
const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
|
||
// Only recalculate progress if enough time has passed (reduce computation frequency)
|
||
if (currentTime_ns - lastTimeUpdate > 16666666) { // ~60 FPS update rate
|
||
//double time_seconds = currentTime_ns / 1000000000.0;
|
||
cachedProgress = (std::cos(2.0 * ult::_M_PI * std::fmod(currentTime_ns / 1000000000.0 - 0.25, 1.0)) + 1.0) / 2.0;
|
||
lastTimeUpdate = currentTime_ns;
|
||
}
|
||
progress = cachedProgress;
|
||
|
||
Color clickColor1 = highlightColor1;
|
||
Color clickColor2 = clickColor;
|
||
|
||
if (progress >= 0.5) {
|
||
clickColor1 = clickColor;
|
||
clickColor2 = highlightColor2;
|
||
}
|
||
|
||
// Combine color interpolation into single calculation
|
||
highlightColor = {
|
||
static_cast<u8>((clickColor1.r - clickColor2.r) * progress + clickColor2.r),
|
||
static_cast<u8>((clickColor1.g - clickColor2.g) * progress + clickColor2.g),
|
||
static_cast<u8>((clickColor1.b - clickColor2.b) * progress + clickColor2.b),
|
||
0xF
|
||
};
|
||
|
||
x = 0;
|
||
y = 0;
|
||
if (this->m_highlightShaking) {
|
||
t_ns = currentTime_ns - this->m_highlightShakingStartTime;
|
||
if (t_ns >= 100000000) // 100ms in nanoseconds
|
||
this->m_highlightShaking = false;
|
||
else {
|
||
// Use faster random generation if available, or cache amplitude
|
||
static int cachedAmplitude = std::rand() % 5 + 5;
|
||
if (t_ns % 10000000 == 0) // Update amplitude less frequently
|
||
cachedAmplitude = std::rand() % 5 + 5;
|
||
amplitude = cachedAmplitude;
|
||
|
||
const int shakeOffset = shakeAnimation(t_ns, amplitude);
|
||
switch (this->m_highlightShakingDirection) {
|
||
case FocusDirection::Up: y = -shakeOffset; break;
|
||
case FocusDirection::Down: y = shakeOffset; break;
|
||
case FocusDirection::Left: x = -shakeOffset; break;
|
||
case FocusDirection::Right: x = shakeOffset; break;
|
||
default: break;
|
||
}
|
||
|
||
x = std::clamp(x, -amplitude, amplitude);
|
||
y = std::clamp(y, -amplitude, amplitude);
|
||
}
|
||
}
|
||
|
||
renderer->drawBorderedRoundedRect(this->getX() + x, this->getY() + y, this->getWidth() +4, this->getHeight(), 5, 5, a(highlightColor));
|
||
}
|
||
|
||
/**
|
||
* @brief Draws the back background when a element is highlighted
|
||
* @note Override this if you have a element that e.g requires a non-rectangular focus
|
||
*
|
||
* @param renderer Renderer
|
||
*/
|
||
virtual void drawFocusBackground(gfx::Renderer *renderer) {
|
||
if (this->m_clickAnimationProgress > 0) {
|
||
this->drawClickAnimation(renderer);
|
||
|
||
// Single time calculation and direct millisecond conversion
|
||
//const double elapsed_ms = (armTicksToNs(armGetSystemTick()) - this->m_animationStartTime) * 0.000001; // Direct conversion
|
||
|
||
// Direct calculation without intermediate multiplication
|
||
this->m_clickAnimationProgress = tsl::style::ListItemHighlightLength * (1.0f - ((armTicksToNs(armGetSystemTick()) - this->m_animationStartTime) * 0.000001) * 0.002f); // 0.002f = 1/500
|
||
|
||
// Clamp to 0 in one operation
|
||
if (this->m_clickAnimationProgress < 0) {
|
||
this->m_clickAnimationProgress = 0;
|
||
}
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief Draws the blue boarder when a element is highlighted
|
||
* @note Override this if you have a element that e.g requires a non-rectangular focus
|
||
*
|
||
* @param renderer Renderer
|
||
*/
|
||
virtual void drawHighlight(gfx::Renderer *renderer) { // CUSTOM MODIFICATION start
|
||
if (!m_isItem)
|
||
return;
|
||
|
||
// Use cached time calculation from drawClickAnimation if possible
|
||
static u64 lastHighlightUpdate = 0;
|
||
static double cachedHighlightProgress = 0.0;
|
||
const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
|
||
// Update progress at 60 FPS rate with high-precision calculation
|
||
if (currentTime_ns - lastHighlightUpdate > 16666666) {
|
||
// High precision time calculation - matches original timing exactly
|
||
//double time_seconds = currentTime_ns * 0.000000001; // Direct conversion like original
|
||
|
||
// Match original calculation exactly but with higher precision
|
||
cachedHighlightProgress = (std::cos(2.0 * ult::_M_PI * std::fmod(currentTime_ns * 0.000000001 - 0.25, 1.0)) + 1.0) * 0.5;
|
||
|
||
lastHighlightUpdate = currentTime_ns;
|
||
}
|
||
progress = cachedHighlightProgress;
|
||
|
||
// Cache the interpreter state check result to avoid atomic load overhead
|
||
static bool lastInterpreterState = false;
|
||
static u64 lastInterpreterCheck = 0;
|
||
if (currentTime_ns - lastInterpreterCheck > 50000000) { // Check every 50ms
|
||
lastInterpreterState = ult::runningInterpreter.load(std::memory_order_acquire);
|
||
lastInterpreterCheck = currentTime_ns;
|
||
}
|
||
|
||
if (lastInterpreterState) {
|
||
// High precision floating point color interpolation for interpreter colors
|
||
highlightColor = {
|
||
static_cast<u8>(highlightColor4.r + (highlightColor3.r - highlightColor4.r) * progress + 0.5),
|
||
static_cast<u8>(highlightColor4.g + (highlightColor3.g - highlightColor4.g) * progress + 0.5),
|
||
static_cast<u8>(highlightColor4.b + (highlightColor3.b - highlightColor4.b) * progress + 0.5),
|
||
0xF
|
||
};
|
||
} else {
|
||
// High precision floating point color interpolation for normal colors
|
||
highlightColor = {
|
||
static_cast<u8>(highlightColor2.r + (highlightColor1.r - highlightColor2.r) * progress + 0.5),
|
||
static_cast<u8>(highlightColor2.g + (highlightColor1.g - highlightColor2.g) * progress + 0.5),
|
||
static_cast<u8>(highlightColor2.b + (highlightColor1.b - highlightColor2.b) * progress + 0.5),
|
||
0xF
|
||
};
|
||
}
|
||
|
||
x = 0;
|
||
y = 0;
|
||
|
||
if (this->m_highlightShaking) {
|
||
t_ns = currentTime_ns - this->m_highlightShakingStartTime;
|
||
if (t_ns >= 100000000) // 100ms in nanoseconds
|
||
this->m_highlightShaking = false;
|
||
else {
|
||
// Use cached amplitude like in drawClickAnimation
|
||
static int cachedAmplitude = std::rand() % 5 + 5;
|
||
if (t_ns % 10000000 == 0)
|
||
cachedAmplitude = std::rand() % 5 + 5;
|
||
amplitude = cachedAmplitude;
|
||
|
||
const int shakeOffset = shakeAnimation(t_ns, amplitude);
|
||
switch (this->m_highlightShakingDirection) {
|
||
case FocusDirection::Up: y = -shakeOffset; break;
|
||
case FocusDirection::Down: y = shakeOffset; break;
|
||
case FocusDirection::Left: x = -shakeOffset; break;
|
||
case FocusDirection::Right: x = shakeOffset; break;
|
||
default: break;
|
||
}
|
||
|
||
x = std::clamp(x, -amplitude, amplitude);
|
||
y = std::clamp(y, -amplitude, amplitude);
|
||
}
|
||
}
|
||
|
||
if (this->m_clickAnimationProgress == 0) {
|
||
if (ult::useSelectionBG) {
|
||
if (ult::expandedMemory)
|
||
renderer->drawRectMultiThreaded(this->getX() + x + 4, this->getY() + y, this->getWidth() - 12 +4, this->getHeight(), aWithOpacity(selectionBGColor)); // CUSTOM MODIFICATION
|
||
else
|
||
renderer->drawRect(this->getX() + x + 4, this->getY() + y, this->getWidth() - 12 +4, this->getHeight(), aWithOpacity(selectionBGColor));
|
||
}
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
// Determine the active percentage to use
|
||
//float activePercentage = 0.0f;
|
||
//if (ult::downloadPercentage > 0) {
|
||
// activePercentage = ult::downloadPercentage;
|
||
//} else if (ult::unzipPercentage > 0) {
|
||
// activePercentage = ult::unzipPercentage;
|
||
//} else if (ult::copyPercentage > 0) {
|
||
// activePercentage = ult::copyPercentage;
|
||
//}
|
||
const float activePercentage = ult::displayPercentage.load(std::memory_order_acquire);
|
||
if (activePercentage > 0){
|
||
if (ult::expandedMemory)
|
||
renderer->drawRectMultiThreaded(this->getX() + x + 4, this->getY() + y, (this->getWidth()- 12 +4)*(activePercentage * 0.01f), this->getHeight(), aWithOpacity(progressColor)); // Direct percentage conversion
|
||
else
|
||
renderer->drawRect(this->getX() + x + 4, this->getY() + y, (this->getWidth()- 12 +4)*(activePercentage * 0.01f), this->getHeight(), aWithOpacity(progressColor)); // Direct percentage conversion
|
||
}
|
||
#endif
|
||
|
||
renderer->drawBorderedRoundedRect(this->getX() + x, this->getY() + y, this->getWidth() +4, this->getHeight(), 5, 5, a(highlightColor));
|
||
}
|
||
|
||
ult::onTrackBar.store(false, std::memory_order_release);
|
||
}
|
||
|
||
|
||
|
||
/**
|
||
* @brief Sets the boundaries of this view
|
||
*
|
||
* @param x Start X pos
|
||
* @param y Start Y pos
|
||
* @param width Width
|
||
* @param height Height
|
||
*/
|
||
inline void setBoundaries(s32 x, s32 y, s32 width, s32 height) {
|
||
this->m_x = x;
|
||
this->m_y = y;
|
||
this->m_width = width;
|
||
this->m_height = height;
|
||
}
|
||
|
||
/**
|
||
* @brief Adds a click listener to the element
|
||
*
|
||
* @param clickListener Click listener called with keys that were pressed last frame. Callback should return true if keys got consumed
|
||
*/
|
||
virtual inline void setClickListener(std::function<bool(u64 keys)> clickListener) {
|
||
this->m_clickListener = clickListener;
|
||
}
|
||
|
||
/**
|
||
* @brief Gets the element's X position
|
||
*
|
||
* @return X position
|
||
*/
|
||
inline s32 getX() { return this->m_x; }
|
||
/**
|
||
* @brief Gets the element's Y position
|
||
*
|
||
* @return Y position
|
||
*/
|
||
inline s32 getY() { return this->m_y; }
|
||
/**
|
||
* @brief Gets the element's Width
|
||
*
|
||
* @return Width
|
||
*/
|
||
inline s32 getWidth() { return this->m_width; }
|
||
/**
|
||
* @brief Gets the element's Height
|
||
*
|
||
* @return Height
|
||
*/
|
||
inline s32 getHeight() { return this->m_height; }
|
||
|
||
inline s32 getTopBound() { return this->getY(); }
|
||
inline s32 getLeftBound() { return this->getX(); }
|
||
inline s32 getRightBound() { return this->getX() + this->getWidth(); }
|
||
inline s32 getBottomBound() { return this->getY() + this->getHeight(); }
|
||
|
||
/**
|
||
* @brief Check if the coordinates are in the elements bounds
|
||
*
|
||
* @return true if coordinates are in bounds, false otherwise
|
||
*/
|
||
bool inBounds(s32 touchX, s32 touchY) {
|
||
//static u32 ult::layerEdge = cfg::LayerPosX == 0 ? 0 : (1280-448);
|
||
return touchX >= this->getLeftBound() + int(ult::layerEdge) && touchX <= this->getRightBound() + int(ult::layerEdge) && touchY >= this->getTopBound() && touchY <= this->getBottomBound();
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the element's parent
|
||
* @note This is required to handle focus and button downpassing properly
|
||
*
|
||
* @param parent Parent
|
||
*/
|
||
inline void setParent(Element *parent) { this->m_parent = parent; }
|
||
|
||
/**
|
||
* @brief Get the element's parent
|
||
*
|
||
* @return Parent
|
||
*/
|
||
inline Element* getParent() { return this->m_parent; }
|
||
|
||
|
||
virtual inline std::vector<Element*> getChildren() const {
|
||
return {}; // Return empty vector for simplicity
|
||
}
|
||
|
||
/**
|
||
* @brief Marks this element as focused or unfocused to draw the highlight
|
||
*
|
||
* @param focused Focused
|
||
*/
|
||
virtual inline void setFocused(bool focused) {
|
||
this->m_focused = focused;
|
||
this->m_clickAnimationProgress = 0;
|
||
}
|
||
|
||
virtual bool matchesJumpCriteria(const std::string& jumpText, const std::string& jumpValue, bool contains) const {
|
||
return false; // Default implementation for non-ListItem elements
|
||
}
|
||
|
||
|
||
static InputMode getInputMode() { return Element::s_inputMode; }
|
||
|
||
static void setInputMode(InputMode mode) { Element::s_inputMode = mode; }
|
||
|
||
protected:
|
||
constexpr static inline auto a = &gfx::Renderer::a;
|
||
constexpr static inline auto aWithOpacity = &gfx::Renderer::aWithOpacity;
|
||
bool m_focused = false;
|
||
u8 m_clickAnimationProgress = 0;
|
||
|
||
// Highlight shake animation
|
||
bool m_highlightShaking = false;
|
||
u64 m_highlightShakingStartTime; // Changed from chrono::time_point to nanoseconds
|
||
FocusDirection m_highlightShakingDirection;
|
||
|
||
static inline InputMode s_inputMode;
|
||
|
||
/**
|
||
* @brief Shake animation calculation based on a damped sine wave
|
||
*
|
||
* @param t_ns Passed time in nanoseconds
|
||
* @param a Amplitude
|
||
* @return Damped sine wave output
|
||
*/
|
||
inline int shakeAnimation(u64 t_ns, float a) {
|
||
//float w = 0.2F;
|
||
//float tau = 0.05F;
|
||
|
||
// Convert nanoseconds to microseconds for the calculation
|
||
const int t_us = t_ns / 1000;
|
||
|
||
return roundf(a * exp(-(0.05F * t_us) * sin(0.2F * t_us)));
|
||
}
|
||
|
||
private:
|
||
friend class Gui;
|
||
|
||
s32 m_x = 0, m_y = 0, m_width = 0, m_height = 0;
|
||
Element *m_parent = nullptr;
|
||
std::vector<Element*> m_children;
|
||
std::function<bool(u64 keys)> m_clickListener = [](u64) { return false; };
|
||
};
|
||
|
||
//static std::vector<Element*> m_lastFrameItems; // for smooth handling of jumpToItem navigation
|
||
//static bool m_hasValidFrame = false;
|
||
//static float m_lastFrameOffset = 0.0f;
|
||
// Static cache with instance validation
|
||
|
||
//#if IS_STATUS_MONITOR_DIRECTIVE
|
||
// /**
|
||
// * @brief A Element that exposes the renderer directly to draw custom views easily
|
||
// */
|
||
// class CustomDrawer : public Element {
|
||
// public:
|
||
// /**
|
||
// * @brief Constructor
|
||
// * @note This element should only be used to draw static things the user cannot interact with e.g info text, images, etc.
|
||
// *
|
||
// * @param renderFunc Callback that will be called once every frame to draw this view
|
||
// */
|
||
// CustomDrawer(std::function<void(gfx::Renderer*, u16 x, u16 y, u16 w, u16 h)> renderFunc) : Element(), m_renderFunc(renderFunc) {}
|
||
// virtual ~CustomDrawer() {
|
||
// m_isTable = true;
|
||
// }
|
||
//
|
||
// virtual void draw(gfx::Renderer* renderer) override {
|
||
// this->m_renderFunc(renderer, this->getX(), this->getY(), this->getWidth(), this->getHeight());
|
||
// }
|
||
//
|
||
// virtual void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {}
|
||
//
|
||
// private:
|
||
// std::function<void(gfx::Renderer*, u16 x, u16 y, u16 w, u16 h)> m_renderFunc;
|
||
// };
|
||
//#else
|
||
/**
|
||
* @brief A Element that exposes the renderer directly to draw custom views easily
|
||
*/
|
||
class CustomDrawer : public Element {
|
||
public:
|
||
/**
|
||
* @brief Constructor
|
||
* @note This element should only be used to draw static things the user cannot interact with e.g info text, images, etc.
|
||
*
|
||
* @param renderFunc Callback that will be called once every frame to draw this view
|
||
*/
|
||
CustomDrawer(std::function<void(gfx::Renderer* r, s32 x, s32 y, s32 w, s32 h)> renderFunc) : Element(), m_renderFunc(renderFunc) {
|
||
m_isItem = false;
|
||
m_isTable = true;
|
||
}
|
||
|
||
virtual ~CustomDrawer() {}
|
||
|
||
virtual void draw(gfx::Renderer* renderer) override {
|
||
//renderer->enableScissoring(ELEMENT_BOUNDS(this));
|
||
this->m_renderFunc(renderer, ELEMENT_BOUNDS(this));
|
||
//renderer->disableScissoring();
|
||
}
|
||
|
||
virtual void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {
|
||
|
||
}
|
||
|
||
private:
|
||
std::function<void(gfx::Renderer*, s32 x, s32 y, s32 w, s32 h)> m_renderFunc;
|
||
};
|
||
//#endif
|
||
|
||
/**
|
||
* @brief A Element that exposes the renderer directly to draw custom views easily
|
||
*/
|
||
class TableDrawer : public Element {
|
||
public:
|
||
TableDrawer(std::function<void(gfx::Renderer* r, s32 x, s32 y, s32 w, s32 h)> renderFunc, bool _hideTableBackground, size_t _endGap, bool _isScrollable = false)
|
||
: Element(), m_renderFunc(renderFunc), hideTableBackground(_hideTableBackground), endGap(_endGap), isScrollable(_isScrollable) {
|
||
m_isTable = isScrollable; // Mark this element as a table
|
||
m_isItem = false;
|
||
}
|
||
|
||
virtual ~TableDrawer() {}
|
||
|
||
virtual void draw(gfx::Renderer* renderer) override {
|
||
|
||
renderer->enableScissoring(0, 88, tsl::cfg::FramebufferWidth, tsl::cfg::FramebufferHeight - 73 - 97 +2+5);
|
||
|
||
if (!hideTableBackground)
|
||
renderer->drawRoundedRect(this->getX() + 4+2, this->getY()-4-1, this->getWidth() +2 + 1, this->getHeight() + 20 - endGap+2, 10.0, aWithOpacity(tableBGColor));
|
||
|
||
m_renderFunc(renderer, this->getX() + 4, this->getY(), this->getWidth() + 4, this->getHeight());
|
||
|
||
renderer->disableScissoring();
|
||
}
|
||
|
||
virtual void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {}
|
||
|
||
|
||
virtual bool onClick(u64 keys) {
|
||
return false;
|
||
}
|
||
|
||
virtual Element* requestFocus(Element *oldFocus, FocusDirection direction) override {
|
||
return nullptr;
|
||
}
|
||
|
||
private:
|
||
std::function<void(gfx::Renderer*, s32 x, s32 y, s32 w, s32 h)> m_renderFunc;
|
||
bool hideTableBackground = false;
|
||
size_t endGap = 3;
|
||
bool isScrollable = false;
|
||
};
|
||
|
||
|
||
/**
|
||
* @brief The base frame which can contain another view
|
||
*
|
||
*/
|
||
class OverlayFrame : public Element {
|
||
public:
|
||
/**
|
||
* @brief Constructor
|
||
*
|
||
* @param title Name of the Overlay drawn bolt at the top
|
||
* @param subtitle Subtitle drawn bellow the title e.g version number
|
||
*/
|
||
std::string m_title;
|
||
std::string m_subtitle;
|
||
|
||
bool m_noClickableItems;
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
std::string m_menuMode; // CUSTOM MODIFICATION
|
||
std::string m_colorSelection; // CUSTOM MODIFICATION
|
||
std::string m_pageLeftName; // CUSTOM MODIFICATION
|
||
std::string m_pageRightName; // CUSTOM MODIFICATION
|
||
|
||
|
||
tsl::Color titleColor = {0xF,0xF,0xF,0xF};
|
||
static constexpr double cycleDuration = 1.6;
|
||
float counter = 0;
|
||
float countOffset;
|
||
float progress;
|
||
float letterWidth;
|
||
#endif
|
||
|
||
#if USING_WIDGET_DIRECTIVE
|
||
bool m_showWidget = false;
|
||
#endif
|
||
|
||
float x, y;
|
||
int offset, y_offset;
|
||
int fontSize;
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
OverlayFrame(const std::string& title, const std::string& subtitle, const bool& _noClickableItems=false, const std::string& menuMode = "", const std::string& colorSelection = "", const std::string& pageLeftName = "", const std::string& pageRightName = "")
|
||
: Element(), m_title(title), m_subtitle(subtitle), m_noClickableItems(_noClickableItems), m_menuMode(menuMode), m_colorSelection(colorSelection), m_pageLeftName(pageLeftName), m_pageRightName(pageRightName) {
|
||
#else
|
||
OverlayFrame(const std::string& title, const std::string& subtitle, const bool& _noClickableItems=false)
|
||
: Element(), m_title(title), m_subtitle(subtitle), m_noClickableItems(_noClickableItems) {
|
||
#endif
|
||
ult::activeHeaderHeight = 97;
|
||
ult::loadWallpaperFileWhenSafe();
|
||
m_isItem = false;
|
||
}
|
||
|
||
~OverlayFrame() {
|
||
delete m_contentElement;
|
||
}
|
||
|
||
#if USING_FPS_INDICATOR_DIRECTIVE
|
||
// Function to calculate FPS
|
||
inline float updateFPS(double currentTimeCount) {
|
||
static double lastUpdateTime = currentTimeCount;
|
||
static int frameCount = 0;
|
||
static float fps = 0.0f;
|
||
|
||
++frameCount;
|
||
const double elapsedTime = currentTimeCount - lastUpdateTime;
|
||
|
||
if (elapsedTime >= 1.0) { // Update FPS every second
|
||
fps = frameCount / static_cast<float>(elapsedTime);
|
||
lastUpdateTime = currentTimeCount;
|
||
frameCount = 0;
|
||
}
|
||
return fps;
|
||
}
|
||
#endif
|
||
|
||
// CUSTOM SECTION START
|
||
void draw(gfx::Renderer *renderer) override {
|
||
if (!ult::themeIsInitialized.load(std::memory_order_acquire)) {
|
||
ult::themeIsInitialized.store(true, std::memory_order_release);
|
||
tsl::initializeThemeVars();
|
||
}
|
||
|
||
|
||
renderer->fillScreen(a(defaultBackgroundColor));
|
||
renderer->drawWallpaper();
|
||
|
||
y = 50;
|
||
offset = 0;
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
// Current interpreter state (atomic<bool>)
|
||
const bool interpreterIsRunningNow = ult::runningInterpreter.load(std::memory_order_acquire) && (ult::downloadPercentage.load(std::memory_order_acquire) != -1 || ult::unzipPercentage.load(std::memory_order_acquire) != -1 || ult::copyPercentage.load(std::memory_order_acquire) != -1) ;
|
||
|
||
|
||
if (m_noClickableItems != ult::noClickableItems.load(std::memory_order_acquire)) {
|
||
ult::noClickableItems.store(m_noClickableItems, std::memory_order_release);
|
||
}
|
||
|
||
const bool isUltrahandMenu = (m_title == ult::CAPITAL_ULTRAHAND_PROJECT_NAME &&
|
||
m_subtitle.find("Ultrahand Package") == std::string::npos &&
|
||
m_subtitle.find("Ultrahand Script") == std::string::npos);
|
||
|
||
if (isUltrahandMenu) {
|
||
#if USING_WIDGET_DIRECTIVE
|
||
renderer->drawWidget();
|
||
#endif
|
||
|
||
if (ult::touchingMenu.load(std::memory_order_acquire) && ult::inMainMenu.load(std::memory_order_acquire)) {
|
||
renderer->drawRoundedRect(0.0f + 7, 12.0f, 245.0f - 13, 73.0f, 10.0f, a(clickColor));
|
||
}
|
||
|
||
x = 20;
|
||
fontSize = 42;
|
||
offset = 6;
|
||
countOffset = 0;
|
||
|
||
if (ult::useDynamicLogo) {
|
||
const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
|
||
// High precision time calculation to prevent stepping artifacts
|
||
const double currentTimeCount = static_cast<double>(currentTime_ns) / 1000000000.0;
|
||
|
||
// Pre-calculate time base once per frame for consistent wave sync
|
||
const double timeBase = std::fmod(currentTimeCount, cycleDuration);
|
||
|
||
// Pre-calculate wave constants - NOW MATCHES CYCLE DURATION
|
||
const double waveScale = 2.0 * ult::_M_PI / cycleDuration;
|
||
static constexpr double phaseShift = ult::_M_PI / 2.0;
|
||
|
||
for (const char letter : ult::SPLIT_PROJECT_NAME_1) {
|
||
// Smooth, precise wave calculation
|
||
const double wavePhase = waveScale * (timeBase + static_cast<double>(countOffset));
|
||
const double rawProgress = std::cos(wavePhase - phaseShift);
|
||
|
||
// Apply double smoothstep for ultra-smooth color transitions
|
||
const double normalizedProgress = (rawProgress + 1.0) * 0.5; // Convert to 0-1
|
||
const double smoothedProgress = normalizedProgress * normalizedProgress * (3.0 - 2.0 * normalizedProgress);
|
||
// Apply smoothstep again for even smoother transitions
|
||
const double ultraSmoothProgress = smoothedProgress * smoothedProgress * (3.0 - 2.0 * smoothedProgress);
|
||
|
||
// Pure floating point interpolation - no dithering to eliminate flicker
|
||
const double blend = std::max(0.0, std::min(1.0, ultraSmoothProgress));
|
||
|
||
// High precision floating point color interpolation
|
||
const tsl::Color highlightColor = {
|
||
static_cast<u8>(dynamicLogoRGB1.r + (dynamicLogoRGB2.r - dynamicLogoRGB1.r) * blend + 0.5),
|
||
static_cast<u8>(dynamicLogoRGB1.g + (dynamicLogoRGB2.g - dynamicLogoRGB1.g) * blend + 0.5),
|
||
static_cast<u8>(dynamicLogoRGB1.b + (dynamicLogoRGB2.b - dynamicLogoRGB1.b) * blend + 0.5),
|
||
15
|
||
};
|
||
|
||
const std::string letterStr(1, letter);
|
||
x += renderer->drawString(letterStr, false, x, y + offset, fontSize, highlightColor).first;
|
||
countOffset -= static_cast<float>(cycleDuration / 8.0);
|
||
}
|
||
} else {
|
||
for (const char letter : ult::SPLIT_PROJECT_NAME_1) {
|
||
const std::string letterStr(1, letter);
|
||
x += renderer->drawString(letterStr, false, x, y + offset, fontSize, logoColor1).first;
|
||
}
|
||
}
|
||
|
||
renderer->drawString(ult::SPLIT_PROJECT_NAME_2, false, x, y + offset, fontSize, (logoColor2));
|
||
|
||
} else {
|
||
if (m_showWidget)
|
||
renderer->drawWidget();
|
||
|
||
x = 20;
|
||
y = 52 -2;
|
||
fontSize = 32;
|
||
|
||
if (m_subtitle.find("Ultrahand Script") != std::string::npos) {
|
||
renderer->drawString(m_title, false, x, y, fontSize, (defaultScriptColor));
|
||
} else {
|
||
titleColor = defaultPackageColor; // Default to green
|
||
|
||
// Ultra-fast color selection using first character optimization
|
||
if (!m_colorSelection.empty()) {
|
||
const char firstChar = m_colorSelection[0];
|
||
const size_t len = m_colorSelection.length();
|
||
|
||
// Fast path: check first char + length for unique combinations
|
||
switch (firstChar) {
|
||
case 'g': // green
|
||
if (len == 5 && m_colorSelection == "green") {
|
||
titleColor = {0x0, 0xF, 0x0, 0xF};
|
||
}
|
||
break;
|
||
case 'r': // red
|
||
if (len == 3 && m_colorSelection == "red") {
|
||
titleColor = RGB888("#F7253E");
|
||
}
|
||
break;
|
||
case 'b': // blue
|
||
if (len == 4 && m_colorSelection == "blue") {
|
||
titleColor = {0x7, 0x7, 0xF, 0xF};
|
||
}
|
||
break;
|
||
case 'y': // yellow
|
||
if (len == 6 && m_colorSelection == "yellow") {
|
||
titleColor = {0xF, 0xF, 0x0, 0xF};
|
||
}
|
||
break;
|
||
case 'o': // orange
|
||
if (len == 6 && m_colorSelection == "orange") {
|
||
titleColor = {0xFF, 0xA5, 0x00, 0xFF};
|
||
}
|
||
break;
|
||
case 'p': // pink or purple
|
||
if (len == 4 && m_colorSelection == "pink") {
|
||
titleColor = {0xFF, 0x69, 0xB4, 0xFF};
|
||
} else if (len == 6 && m_colorSelection == "purple") {
|
||
titleColor = {0x80, 0x00, 0x80, 0xFF};
|
||
}
|
||
break;
|
||
case 'w': // white
|
||
if (len == 5 && m_colorSelection == "white") {
|
||
titleColor = {0xF, 0xF, 0xF, 0xF};
|
||
}
|
||
break;
|
||
case 'u': // ultra
|
||
if (len == 5 && m_colorSelection == "ultra") {
|
||
for (const char letter : m_title) {
|
||
progress = ult::calculateAmplitude(counter - x * 0.0001F);
|
||
|
||
const tsl::Color highlightColor = {
|
||
static_cast<u8>((0xA - 0xF) * (3 - 1.5 * progress) + 0xF),
|
||
static_cast<u8>((0xA - 0xF) * 1.5 * progress + 0xF),
|
||
static_cast<u8>((0xA - 0xF) * (1.25 - progress) + 0xF),
|
||
0xF
|
||
};
|
||
|
||
const std::string letterStr(1, letter);
|
||
renderer->drawString(letterStr.c_str(), false, x, y, fontSize, (highlightColor));
|
||
const auto letterWidth = renderer->getTextDimensions(letterStr, false, fontSize).first;
|
||
x += letterWidth;
|
||
counter -= 0.00004F;
|
||
}
|
||
goto skip_normal_draw;
|
||
}
|
||
break;
|
||
case '#': // hex color
|
||
if (len == 7 && ult::isValidHexColor(m_colorSelection.substr(1))) {
|
||
titleColor = RGB888(m_colorSelection.substr(1));
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
|
||
renderer->drawString(m_title, false, x, y, fontSize, (titleColor));
|
||
y += 2;
|
||
skip_normal_draw:;
|
||
}
|
||
}
|
||
static const std::vector<std::string> specialChars2 = {""};
|
||
if (m_title == ult::CAPITAL_ULTRAHAND_PROJECT_NAME) {
|
||
renderer->drawStringWithColoredSections(ult::versionLabel, false, specialChars2, 20, y+25, 15, (bannerVersionTextColor), textSeparatorColor);
|
||
} else {
|
||
std::string subtitle = m_subtitle;
|
||
const size_t pos = subtitle.find("?Ultrahand Script");
|
||
if (pos != std::string::npos) {
|
||
subtitle.erase(pos, 17); // "?Ultrahand Script".length() = 17
|
||
}
|
||
renderer->drawStringWithColoredSections(subtitle, false, specialChars2, 20, y+23, 15, (bannerVersionTextColor), textSeparatorColor);
|
||
}
|
||
|
||
#else
|
||
if (m_noClickableItems != ult::noClickableItems.load(std::memory_order_acquire)) {
|
||
ult::noClickableItems.store(m_noClickableItems, std::memory_order_release);
|
||
}
|
||
#if USING_WIDGET_DIRECTIVE
|
||
if (m_showWidget)
|
||
renderer->drawWidget();
|
||
#endif
|
||
|
||
renderer->drawString(m_title, false, 20, 52-2, 32, (defaultOverlayColor));
|
||
renderer->drawString(m_subtitle, false, 20, y+2+23+2, 15, (bannerVersionTextColor));
|
||
#endif
|
||
|
||
renderer->drawRect(15, tsl::cfg::FramebufferHeight - 73, tsl::cfg::FramebufferWidth - 30, 1, a(bottomSeparatorColor));
|
||
|
||
// Compute gap width once from GAP_1 and derive halfGap
|
||
const float gapWidth = renderer->getTextDimensions(ult::GAP_1, false, 23).first;
|
||
|
||
// Calculate text widths for buttons depending on launch mode and interpreter state
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
const float backTextWidth = renderer->getTextDimensions(
|
||
"\uE0E1" + ult::GAP_2 + (!interpreterIsRunningNow ? ult::BACK : ult::HIDE), false, 23).first;
|
||
const float selectTextWidth = renderer->getTextDimensions(
|
||
"\uE0E0" + ult::GAP_2 + (!interpreterIsRunningNow ? ult::OK : ult::CANCEL), false, 23).first;
|
||
#else
|
||
const float backTextWidth = renderer->getTextDimensions(
|
||
"\uE0E1" + ult::GAP_2 + ult::BACK, false, 23).first;
|
||
const float selectTextWidth = renderer->getTextDimensions(
|
||
"\uE0E0" + ult::GAP_2 + ult::OK, false, 23).first;
|
||
#endif
|
||
|
||
const float _halfGap = gapWidth / 2.0f;
|
||
if (_halfGap != ult::halfGap.load(std::memory_order_acquire))
|
||
ult::halfGap.store(_halfGap, std::memory_order_release);
|
||
|
||
// Total button widths include half-gap padding on both sides
|
||
const float _backWidth = backTextWidth + gapWidth;
|
||
if (_backWidth != ult::backWidth.load(std::memory_order_acquire))
|
||
ult::backWidth.store(_backWidth, std::memory_order_release);
|
||
const float _selectWidth = selectTextWidth + gapWidth;
|
||
if (_selectWidth != ult::selectWidth.load(std::memory_order_acquire))
|
||
ult::selectWidth.store(_selectWidth, std::memory_order_release);
|
||
|
||
// Set initial button position
|
||
static constexpr float buttonStartX = 30;
|
||
const float buttonY = static_cast<float>(cfg::FramebufferHeight - 73 + 1);
|
||
|
||
// Draw back button if touched
|
||
if (ult::touchingBack) {
|
||
renderer->drawRoundedRect(buttonStartX+2 - _halfGap, buttonY, _backWidth-1, 73.0f, 10.0f, a(clickColor));
|
||
}
|
||
|
||
// Draw select button (to the right of back) if touched
|
||
if (ult::touchingSelect.load(std::memory_order_acquire) && !m_noClickableItems) {
|
||
renderer->drawRoundedRect(buttonStartX+2 - _halfGap + _backWidth+1, buttonY,
|
||
_selectWidth-2, 73.0f, 10.0f, a(clickColor));
|
||
}
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
// Handle optional next page button when in launcher mode and appropriate conditions are met
|
||
if (!interpreterIsRunningNow && (ult::inMainMenu.load(std::memory_order_acquire) ||
|
||
!m_pageLeftName.empty() || !m_pageRightName.empty())) {
|
||
// Construct next-page label inline without creating temporary strings
|
||
const float _nextPageWidth = renderer->getTextDimensions(
|
||
!m_pageLeftName.empty() ? ("\uE0ED" + ult::GAP_2 + m_pageLeftName) :
|
||
!m_pageRightName.empty() ? ("\uE0EE" + ult::GAP_2 + m_pageRightName) :
|
||
(ult::inMainMenu.load(std::memory_order_acquire) ?
|
||
((m_menuMode == "packages" ?
|
||
(ult::usePageSwap ? "\uE0EE" : "\uE0ED") :
|
||
(ult::usePageSwap ? "\uE0ED" : "\uE0EE")) +
|
||
ult::GAP_2 + (ult::inOverlaysPage.load(std::memory_order_acquire) ?
|
||
ult::PACKAGES : ult::OVERLAYS_ABBR)) :
|
||
""),
|
||
false, 23).first + gapWidth;
|
||
|
||
if (_nextPageWidth != ult::nextPageWidth.load(std::memory_order_acquire))
|
||
ult::nextPageWidth.store(_nextPageWidth, std::memory_order_release);
|
||
|
||
// Draw next-page button if touched
|
||
if (ult::touchingNextPage.load(std::memory_order_acquire)) {
|
||
float nextX = buttonStartX+2 - _halfGap + _backWidth +1;
|
||
if (!m_noClickableItems)
|
||
nextX += _selectWidth;
|
||
|
||
renderer->drawRoundedRect(nextX, buttonY,
|
||
_nextPageWidth-2,
|
||
73.0f, 10.0f, a(clickColor));
|
||
}
|
||
}
|
||
#endif
|
||
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
|
||
const std::string menuBottomLine =
|
||
"\uE0E1" + ult::GAP_2 +
|
||
(interpreterIsRunningNow ? ult::HIDE : ult::BACK) + ult::GAP_1 +
|
||
(!m_noClickableItems && !interpreterIsRunningNow
|
||
? "\uE0E0" + ult::GAP_2 + ult::OK + ult::GAP_1
|
||
: "") +
|
||
(interpreterIsRunningNow
|
||
? "\uE0E5" + ult::GAP_2 + ult::CANCEL + ult::GAP_1
|
||
: "") +
|
||
(!interpreterIsRunningNow
|
||
? (!ult::usePageSwap
|
||
? (m_menuMode == "packages"
|
||
? "\uE0ED" + ult::GAP_2 + ult::OVERLAYS_ABBR
|
||
: m_menuMode == "overlays"
|
||
? "\uE0EE" + ult::GAP_2 + ult::PACKAGES
|
||
: "")
|
||
: (m_menuMode == "packages"
|
||
? "\uE0EE" + ult::GAP_2 + ult::OVERLAYS_ABBR
|
||
: m_menuMode == "overlays"
|
||
? "\uE0ED" + ult::GAP_2 + ult::PACKAGES
|
||
: ""))
|
||
: "") +
|
||
(!interpreterIsRunningNow && !m_pageLeftName.empty()
|
||
? "\uE0ED" + ult::GAP_2 + m_pageLeftName
|
||
: !interpreterIsRunningNow && !m_pageRightName.empty()
|
||
? "\uE0EE" + ult::GAP_2 + m_pageRightName
|
||
: "");
|
||
#else
|
||
const std::string menuBottomLine =
|
||
"\uE0E1" + ult::GAP_2 + ult::BACK + ult::GAP_1 +
|
||
(!m_noClickableItems
|
||
? "\uE0E0" + ult::GAP_2 + ult::OK + ult::GAP_1
|
||
: "");
|
||
#endif
|
||
|
||
// Render the text - it starts halfGap inside the first button, so edgePadding + halfGap
|
||
static const std::vector<std::string> specialChars = {"\uE0E1","\uE0E0","\uE0ED","\uE0EE","\uE0E5"};
|
||
renderer->drawStringWithColoredSections(menuBottomLine, false, specialChars,
|
||
buttonStartX, 693, 23,
|
||
(bottomTextColor), (buttonColor));
|
||
|
||
#if USING_FPS_INDICATOR_DIRECTIVE
|
||
// Update and display FPS
|
||
const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
const double currentTime_seconds = currentTime_ns / 1000000000.0;
|
||
const float currentFps = updateFPS(currentTime_seconds);
|
||
|
||
static char fpsBuffer[32];
|
||
static float lastFps = -1.0f;
|
||
|
||
// Only update string if FPS changed significantly
|
||
if (std::abs(currentFps - lastFps) > 0.1f) {
|
||
snprintf(fpsBuffer, sizeof(fpsBuffer), "FPS: %.2f", currentFps);
|
||
lastFps = currentFps;
|
||
}
|
||
static constexpr auto whiteColor = tsl::Color(0xF,0xF,0xF,0xF);
|
||
renderer->drawString(fpsBuffer, false, 20, tsl::cfg::FramebufferHeight - 60, 20, whiteColor);
|
||
#endif
|
||
|
||
if (m_contentElement != nullptr)
|
||
m_contentElement->frame(renderer);
|
||
|
||
if (!ult::useRightAlignment)
|
||
renderer->drawRect(447, 0, 448, 720, a(edgeSeparatorColor));
|
||
else
|
||
renderer->drawRect(0, 0, 1, 720, a(edgeSeparatorColor));
|
||
}
|
||
// CUSTOM SECTION END
|
||
|
||
inline void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {
|
||
setBoundaries(parentX, parentY, parentWidth, parentHeight);
|
||
|
||
if (m_contentElement != nullptr) {
|
||
m_contentElement->setBoundaries(parentX + 35, parentY + 97, parentWidth - 85, parentHeight - 73 - 105);
|
||
m_contentElement->invalidate();
|
||
}
|
||
}
|
||
|
||
inline Element* requestFocus(Element *oldFocus, FocusDirection direction) override {
|
||
return m_contentElement ? m_contentElement->requestFocus(oldFocus, direction) : nullptr;
|
||
}
|
||
|
||
inline bool onTouch(TouchEvent event, s32 currX, s32 currY, s32 prevX, s32 prevY, s32 initialX, s32 initialY) {
|
||
// Discard touches outside bounds
|
||
if (!m_contentElement || !m_contentElement->inBounds(currX, currY))
|
||
return false;
|
||
|
||
return m_contentElement->onTouch(event, currX, currY, prevX, prevY, initialX, initialY);
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the content of the frame
|
||
*
|
||
* @param content Element
|
||
*/
|
||
inline void setContent(Element *content) {
|
||
delete m_contentElement;
|
||
m_contentElement = content;
|
||
|
||
if (content != nullptr) {
|
||
m_contentElement->setParent(this);
|
||
invalidate();
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief Changes the title of the menu
|
||
*
|
||
* @param title Title to change to
|
||
*/
|
||
inline void setTitle(const std::string &title) {
|
||
m_title = title;
|
||
}
|
||
|
||
/**
|
||
* @brief Changes the subtitle of the menu
|
||
*
|
||
* @param title Subtitle to change to
|
||
*/
|
||
inline void setSubtitle(const std::string &subtitle) {
|
||
m_subtitle = subtitle;
|
||
}
|
||
|
||
protected:
|
||
Element *m_contentElement = nullptr;
|
||
};
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
|
||
/**
|
||
* @brief The base frame which can contain another view
|
||
*
|
||
*/
|
||
class HeaderOverlayFrame : public Element {
|
||
public:
|
||
/**
|
||
* @brief Constructor
|
||
*
|
||
* @param title Name of the Overlay drawn bolt at the top
|
||
* @param subtitle Subtitle drawn bellow the title e.g version number
|
||
*/
|
||
std::string m_title;
|
||
std::string m_subtitle;
|
||
bool m_noClickableItems;
|
||
|
||
float x, y;
|
||
int offset, y_offset;
|
||
int fontSize;
|
||
|
||
HeaderOverlayFrame(const std::string& title, const std::string& subtitle, const bool& _noClickableItems=false)
|
||
: Element(), m_title(title), m_subtitle(subtitle), m_noClickableItems(_noClickableItems) {
|
||
ult::activeHeaderHeight = 97;
|
||
|
||
if (FullMode)
|
||
ult::loadWallpaperFileWhenSafe();
|
||
else
|
||
svcSleepThread(180'000); // sleep thread for initial values to auto-load
|
||
|
||
m_isItem = false;
|
||
}
|
||
|
||
virtual ~HeaderOverlayFrame() {
|
||
if (this->m_contentElement != nullptr)
|
||
delete this->m_contentElement;
|
||
}
|
||
|
||
|
||
virtual void draw(gfx::Renderer *renderer) override {
|
||
if (!ult::themeIsInitialized.load(std::memory_order_acquire) && FullMode) {
|
||
ult::themeIsInitialized.store(true, std::memory_order_release);
|
||
tsl::initializeThemeVars();
|
||
}
|
||
|
||
if (m_noClickableItems != ult::noClickableItems.load(std::memory_order_acquire)) {
|
||
ult::noClickableItems.store(m_noClickableItems, std::memory_order_release);
|
||
}
|
||
|
||
|
||
if (FullMode == true) {
|
||
renderer->fillScreen(a(defaultBackgroundColor));
|
||
renderer->drawWallpaper();
|
||
} else {
|
||
renderer->fillScreen({ 0x0, 0x0, 0x0, 0x0});
|
||
}
|
||
|
||
y = 50;
|
||
offset = 0;
|
||
|
||
renderer->drawString(this->m_title, false, 20, 50, 32, (defaultOverlayColor));
|
||
renderer->drawString(this->m_subtitle, false, 20, y+2+23+2, 15, (bannerVersionTextColor));
|
||
|
||
if (FullMode == true)
|
||
renderer->drawRect(15, tsl::cfg::FramebufferHeight - 73, tsl::cfg::FramebufferWidth - 30, 1, a(bottomSeparatorColor));
|
||
|
||
// Set initial button position
|
||
static constexpr float buttonStartX = 30;
|
||
|
||
if (FullMode && !deactivateOriginalFooter) {
|
||
// Get the exact gap width from ult::GAP_1
|
||
static const auto gapWidth = renderer->getTextDimensions(ult::GAP_1, false, 23).first;
|
||
static const float _halfGap = gapWidth / 2.0f;
|
||
if (_halfGap != ult::halfGap.load(std::memory_order_acquire))
|
||
ult::halfGap.store(_halfGap, std::memory_order_release);
|
||
|
||
// Calculate text dimensions for buttons without gaps
|
||
static const auto backTextWidth = renderer->getTextDimensions("\uE0E1" + ult::GAP_2 + ult::BACK, false, 23).first;
|
||
static const auto selectTextWidth = renderer->getTextDimensions("\uE0E0" + ult::GAP_2 + ult::OK, false, 23).first;
|
||
|
||
// Update widths to include the half-gap padding on each side
|
||
static const float _backWidth = backTextWidth + gapWidth;
|
||
if (_backWidth != ult::backWidth.load(std::memory_order_acquire))
|
||
ult::backWidth.store(_backWidth, std::memory_order_release);
|
||
static const float _selectWidth = selectTextWidth + gapWidth;
|
||
if (_selectWidth != ult::selectWidth.load(std::memory_order_acquire))
|
||
ult::selectWidth.store(_selectWidth, std::memory_order_release);
|
||
|
||
static const float buttonY = static_cast<float>(cfg::FramebufferHeight - 73 + 1);
|
||
|
||
// Draw back button rectangle
|
||
if (ult::touchingBack.load(std::memory_order_acquire)) {
|
||
renderer->drawRoundedRect(buttonStartX+2 - _halfGap, buttonY, _backWidth-1, 73.0f, 10.0f, a(clickColor));
|
||
}
|
||
|
||
// Draw select button rectangle (starts right after back button)
|
||
if (ult::touchingSelect.load(std::memory_order_acquire) && !m_noClickableItems) {
|
||
renderer->drawRoundedRect(buttonStartX+2 - _halfGap + _backWidth+1, buttonY,
|
||
_selectWidth-2, 73.0f, 10.0f, a(clickColor));
|
||
}
|
||
}
|
||
|
||
// Render the text with special character handling
|
||
if (!deactivateOriginalFooter) {
|
||
const std::string menuBottomLine =
|
||
"\uE0E1" + ult::GAP_2 + ult::BACK + ult::GAP_1 +
|
||
(!m_noClickableItems
|
||
? "\uE0E0" + ult::GAP_2 + ult::OK + ult::GAP_1
|
||
: "");
|
||
static const std::vector<std::string> specialChars = {"\uE0E1","\uE0E0","\uE0ED","\uE0EE","\uE0E5"};
|
||
renderer->drawStringWithColoredSections(menuBottomLine, false, specialChars, buttonStartX, 693, 23, (bottomTextColor), (buttonColor));
|
||
}
|
||
|
||
|
||
if (this->m_contentElement != nullptr)
|
||
this->m_contentElement->frame(renderer);
|
||
|
||
if (FullMode) {
|
||
if (!ult::useRightAlignment)
|
||
renderer->drawRect(447, 0, 448, 720, a(edgeSeparatorColor));
|
||
else
|
||
renderer->drawRect(0, 0, 1, 720, a(edgeSeparatorColor));
|
||
}
|
||
}
|
||
|
||
|
||
|
||
virtual void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {
|
||
this->setBoundaries(parentX, parentY, parentWidth, parentHeight);
|
||
|
||
if (this->m_contentElement != nullptr) {
|
||
//this->m_contentElement->setBoundaries(parentX + 35, parentY + 140, parentWidth - 85, parentHeight - 73 - 105); // CUSTOM MODIFICATION
|
||
this->m_contentElement->setBoundaries(parentX + 35, parentY + ult::activeHeaderHeight, parentWidth - 85, parentHeight - 73 - 105);
|
||
this->m_contentElement->invalidate();
|
||
}
|
||
}
|
||
virtual inline Element* requestFocus(Element *oldFocus, FocusDirection direction) override {
|
||
if (this->m_contentElement != nullptr)
|
||
return this->m_contentElement->requestFocus(oldFocus, direction);
|
||
else
|
||
return nullptr;
|
||
}
|
||
|
||
virtual inline bool onTouch(TouchEvent event, s32 currX, s32 currY, s32 prevX, s32 prevY, s32 initialX, s32 initialY) {
|
||
// Discard touches outside bounds
|
||
if (!this->m_contentElement->inBounds(currX, currY))
|
||
return false;
|
||
|
||
if (this->m_contentElement != nullptr)
|
||
return this->m_contentElement->onTouch(event, currX, currY, prevX, prevY, initialX, initialY);
|
||
else return false;
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the content of the frame
|
||
*
|
||
* @param content Element
|
||
*/
|
||
inline void setContent(Element *content) {
|
||
if (this->m_contentElement != nullptr)
|
||
delete this->m_contentElement;
|
||
|
||
this->m_contentElement = content;
|
||
|
||
if (content != nullptr) {
|
||
this->m_contentElement->setParent(this);
|
||
this->invalidate();
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief Changes the title of the menu
|
||
*
|
||
* @param title Title to change to
|
||
*/
|
||
inline void setTitle(const std::string &title) {
|
||
this->m_title = title;
|
||
}
|
||
|
||
/**
|
||
* @brief Changes the subtitle of the menu
|
||
*
|
||
* @param title Subtitle to change to
|
||
*/
|
||
inline void setSubtitle(const std::string &subtitle) {
|
||
this->m_subtitle = subtitle;
|
||
}
|
||
|
||
protected:
|
||
Element *m_contentElement = nullptr;
|
||
|
||
//std::string m_title, m_subtitle;
|
||
};
|
||
#else
|
||
/**
|
||
* @brief The base frame which can contain another view with a customizable header
|
||
*
|
||
*/
|
||
class HeaderOverlayFrame : public Element {
|
||
public:
|
||
#if USING_WIDGET_DIRECTIVE
|
||
bool m_showWidget = false;
|
||
#endif
|
||
|
||
HeaderOverlayFrame(u16 headerHeight = 175) : Element(), m_headerHeight(headerHeight) {
|
||
ult::activeHeaderHeight = headerHeight;
|
||
// Load the bitmap file into memory
|
||
ult::loadWallpaperFileWhenSafe();
|
||
m_isItem = false;
|
||
|
||
}
|
||
virtual ~HeaderOverlayFrame() {
|
||
if (this->m_contentElement != nullptr)
|
||
delete this->m_contentElement;
|
||
|
||
if (this->m_header != nullptr)
|
||
delete this->m_header;
|
||
}
|
||
|
||
virtual void draw(gfx::Renderer *renderer) override {
|
||
if (!ult::themeIsInitialized.load(std::memory_order_acquire)) {
|
||
ult::themeIsInitialized.store(true, std::memory_order_release);
|
||
tsl::initializeThemeVars();
|
||
}
|
||
|
||
renderer->fillScreen(a(defaultBackgroundColor));
|
||
renderer->drawWallpaper();
|
||
//renderer->drawRect(tsl::cfg::FramebufferWidth - 1, 0, 1, tsl::cfg::FramebufferHeight, a(0xF222));
|
||
renderer->drawRect(15, tsl::cfg::FramebufferHeight - 73, tsl::cfg::FramebufferWidth - 30, 1, a(bottomSeparatorColor));
|
||
|
||
#if USING_WIDGET_DIRECTIVE
|
||
if (m_showWidget)
|
||
renderer->drawWidget();
|
||
#endif
|
||
|
||
// Get the exact gap width from ult::GAP_1
|
||
const float gapWidth = renderer->getTextDimensions(ult::GAP_1, false, 23).first;
|
||
const float _halfGap = gapWidth / 2.0f;
|
||
if (_halfGap != ult::halfGap.load(std::memory_order_acquire))
|
||
ult::halfGap.store(_halfGap, std::memory_order_release);
|
||
|
||
// Calculate text dimensions for buttons without gaps
|
||
const float backTextWidth = renderer->getTextDimensions("\uE0E1" + ult::GAP_2 + ult::BACK, false, 23).first;
|
||
const float selectTextWidth = renderer->getTextDimensions("\uE0E0" + ult::GAP_2 + ult::OK, false, 23).first;
|
||
|
||
// Store final widths with gap padding included
|
||
const float _backWidth = backTextWidth + gapWidth;
|
||
if (_backWidth != ult::backWidth.load(std::memory_order_acquire))
|
||
ult::backWidth.store(_backWidth, std::memory_order_release);
|
||
const float _selectWidth = selectTextWidth + gapWidth;
|
||
if (_selectWidth != ult::selectWidth.load(std::memory_order_acquire))
|
||
ult::selectWidth.store(_selectWidth, std::memory_order_release);
|
||
|
||
// Set initial button position
|
||
static constexpr float buttonStartX = 30;
|
||
const float buttonY = static_cast<float>(cfg::FramebufferHeight - 73 + 1);
|
||
|
||
// Draw back button rectangle
|
||
if (ult::touchingBack.load(std::memory_order_acquire)) {
|
||
renderer->drawRoundedRect(buttonStartX+2 - _halfGap, buttonY, _backWidth-1, 73.0f, 10.0f, a(clickColor));
|
||
}
|
||
|
||
// Draw select button rectangle
|
||
if (ult::touchingSelect.load(std::memory_order_acquire)) {
|
||
renderer->drawRoundedRect(buttonStartX+2 - _halfGap + _backWidth+1, buttonY,
|
||
_selectWidth-2, 73.0f, 10.0f, a(clickColor));
|
||
}
|
||
|
||
// Draw bottom text
|
||
const std::string menuBottomLine = "\uE0E1" + ult::GAP_2 + ult::BACK + ult::GAP_1 +
|
||
"\uE0E0" + ult::GAP_2 + ult::OK + ult::GAP_1;
|
||
renderer->drawStringWithColoredSections(menuBottomLine, false,
|
||
{"\uE0E1", "\uE0E0", "\uE0ED", "\uE0EE"},
|
||
buttonStartX, 693, 23,
|
||
bottomTextColor, buttonColor);
|
||
|
||
if (this->m_header != nullptr)
|
||
this->m_header->frame(renderer);
|
||
|
||
if (this->m_contentElement != nullptr)
|
||
this->m_contentElement->frame(renderer);
|
||
|
||
if (!ult::useRightAlignment)
|
||
renderer->drawRect(447, 0, 448, 720, a(edgeSeparatorColor));
|
||
else
|
||
renderer->drawRect(0, 0, 1, 720, a(edgeSeparatorColor));
|
||
}
|
||
|
||
virtual inline void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {
|
||
this->setBoundaries(parentX, parentY, parentWidth, parentHeight);
|
||
|
||
if (this->m_contentElement != nullptr) {
|
||
this->m_contentElement->setBoundaries(parentX + 35, parentY + this->m_headerHeight, parentWidth - 85, parentHeight - 73 - this->m_headerHeight -8);
|
||
this->m_contentElement->invalidate();
|
||
}
|
||
|
||
if (this->m_header != nullptr) {
|
||
this->m_header->setBoundaries(parentX, parentY, parentWidth, this->m_headerHeight);
|
||
this->m_header->invalidate();
|
||
}
|
||
}
|
||
|
||
virtual inline bool onTouch(TouchEvent event, s32 currX, s32 currY, s32 prevX, s32 prevY, s32 initialX, s32 initialY) {
|
||
// Discard touches outside bounds
|
||
if (!this->m_contentElement->inBounds(currX, currY))
|
||
return false;
|
||
|
||
if (this->m_contentElement != nullptr)
|
||
return this->m_contentElement->onTouch(event, currX, currY, prevX, prevY, initialX, initialY);
|
||
else return false;
|
||
}
|
||
|
||
virtual inline Element* requestFocus(Element *oldFocus, FocusDirection direction) override {
|
||
if (this->m_contentElement != nullptr)
|
||
return this->m_contentElement->requestFocus(oldFocus, direction);
|
||
else
|
||
return nullptr;
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the content of the frame
|
||
*
|
||
* @param content Element
|
||
*/
|
||
inline void setContent(Element *content) {
|
||
if (this->m_contentElement != nullptr)
|
||
delete this->m_contentElement;
|
||
|
||
this->m_contentElement = content;
|
||
|
||
if (content != nullptr) {
|
||
this->m_contentElement->setParent(this);
|
||
this->invalidate();
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the header of the frame
|
||
*
|
||
* @param header Header custom drawer
|
||
*/
|
||
inline void setHeader(CustomDrawer *header) {
|
||
if (this->m_header != nullptr)
|
||
delete this->m_header;
|
||
|
||
this->m_header = header;
|
||
|
||
if (header != nullptr) {
|
||
this->m_header->setParent(this);
|
||
this->invalidate();
|
||
}
|
||
}
|
||
|
||
protected:
|
||
Element *m_contentElement = nullptr;
|
||
CustomDrawer *m_header = nullptr;
|
||
|
||
u16 m_headerHeight;
|
||
};
|
||
#endif
|
||
|
||
/**
|
||
* @brief Single color rectangle element mainly used for debugging to visualize boundaries
|
||
*
|
||
*/
|
||
class DebugRectangle : public Element {
|
||
public:
|
||
/**
|
||
* @brief Constructor
|
||
*
|
||
* @param color Color of the rectangle
|
||
*/
|
||
DebugRectangle(Color color) : Element(), m_color(color) {
|
||
m_isItem = false;
|
||
}
|
||
virtual ~DebugRectangle() {}
|
||
|
||
virtual void draw(gfx::Renderer *renderer) override {
|
||
renderer->drawRect(ELEMENT_BOUNDS(this), a(this->m_color));
|
||
}
|
||
|
||
virtual void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {}
|
||
|
||
private:
|
||
Color m_color;
|
||
};
|
||
|
||
|
||
class ListItem; // forward declaration
|
||
|
||
static std::mutex s_lastFrameItemsMutex;
|
||
static std::vector<Element*> s_lastFrameItems;
|
||
static std::atomic<bool> s_isForwardCache(false); // NEW VARIABLE FOR FORWARD CACHING
|
||
static std::atomic<bool> s_hasValidFrame(false);
|
||
static std::atomic<s32> s_cachedTopBound{0};
|
||
static std::atomic<s32> s_cachedBottomBound{0};
|
||
static std::atomic<s32> s_cachedHeight{0};
|
||
static std::atomic<s32> s_cachedListHeight{0};
|
||
static std::atomic<s32> s_cachedActualContentBottom{0};
|
||
static std::atomic<bool> s_shouldDrawScrollbar(false);
|
||
static std::atomic<u32> s_cachedScrollbarHeight{0};
|
||
static std::atomic<u32> s_cachedScrollbarOffset{0};
|
||
static std::atomic<u32> s_cachedScrollbarX{0};
|
||
static std::atomic<u32> s_cachedScrollbarY{0};
|
||
static std::atomic<float> s_currentScrollVelocity{0};
|
||
|
||
static std::atomic<bool> s_directionalKeyReleased{false};
|
||
static std::atomic<bool> s_cacheForwardFrameOnce(true);
|
||
static std::atomic<bool> lastInternalTouchRelease(true);
|
||
static std::atomic<bool> s_hasClearedCache(false);
|
||
|
||
//static std::atomic<bool> s_skipCaching(false);
|
||
|
||
static std::mutex s_safeToSwapMutex;
|
||
//static std::mutex s_safeTransitionMutex;
|
||
static std::atomic<bool> s_safeToSwap{false};
|
||
|
||
static std::atomic<bool> fullDeconstruction{false};
|
||
static std::atomic<bool> skipDeconstruction{false};
|
||
static std::atomic<bool> skipOnce{false};
|
||
|
||
static std::atomic<bool> isTableScrolling{false};
|
||
|
||
class List : public Element {
|
||
|
||
public:
|
||
List() : Element() {
|
||
if (fullDeconstruction.load(std::memory_order_acquire)) {
|
||
return;
|
||
}
|
||
|
||
s_safeToSwap.store(false, std::memory_order_release);
|
||
//s_directionalKeyReleased.store(false, std::memory_order_release);
|
||
//std::lock_guard<std::mutex> lock(s_safeTransitionMutex);
|
||
//s_safeToSwap.store(false, std::memory_order_release);
|
||
|
||
// Initialize instance state
|
||
m_hasForwardCached = false;
|
||
m_pendingJump = false;
|
||
m_cachingDisabled = false;
|
||
m_clearList = false;
|
||
m_focusedIndex = 0;
|
||
m_offset = 0;
|
||
m_nextOffset = 0;
|
||
m_listHeight = 0;
|
||
actualItemCount = 0;
|
||
m_isItem = false;
|
||
|
||
{
|
||
std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex);
|
||
|
||
s_hasClearedCache.store(false, std::memory_order_release);
|
||
|
||
if (skipDeconstruction.load(std::memory_order_acquire)) {
|
||
purgePendingItems();
|
||
} else {
|
||
s_cacheForwardFrameOnce.store(true, std::memory_order_release);
|
||
skipOnce.store(false, std::memory_order_release);
|
||
}
|
||
}
|
||
}
|
||
|
||
virtual ~List() {
|
||
if (fullDeconstruction.load(std::memory_order_acquire)) {
|
||
std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex); // Add this
|
||
|
||
purgePendingItems();
|
||
|
||
if (s_isForwardCache.load(std::memory_order_acquire)) {
|
||
clearStaticCacheUnsafe(true);
|
||
s_isForwardCache.store(false, std::memory_order_release);
|
||
} else {
|
||
clearStaticCacheUnsafe();
|
||
}
|
||
clearItems();
|
||
|
||
return;
|
||
}
|
||
|
||
s_safeToSwap.store(false, std::memory_order_release);
|
||
//s_directionalKeyReleased.store(false, std::memory_order_release);
|
||
//std::lock_guard<std::mutex> lock(s_safeTransitionMutex);
|
||
//s_safeToSwap.store(false, std::memory_order_release);
|
||
|
||
// NOW take mutex for shared static variable operations
|
||
{
|
||
std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex);
|
||
|
||
if (!skipDeconstruction.load(std::memory_order_acquire)) {
|
||
purgePendingItems();
|
||
|
||
if (!s_isForwardCache.load(std::memory_order_acquire)) {
|
||
clearStaticCacheUnsafe();
|
||
clearItems();
|
||
}
|
||
|
||
s_isForwardCache.store(false, std::memory_order_release);
|
||
s_cacheForwardFrameOnce.store(true, std::memory_order_release);
|
||
}
|
||
|
||
if (m_cachingDisabled || (skipOnce.load(std::memory_order_acquire) && skipDeconstruction.load(std::memory_order_acquire))) {
|
||
purgePendingItems();
|
||
clearItems();
|
||
} else if (skipDeconstruction.load(std::memory_order_acquire)) {
|
||
skipOnce.store(true, std::memory_order_release);
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
virtual void draw(gfx::Renderer* renderer) override {
|
||
if (fullDeconstruction.load(std::memory_order_acquire)) {
|
||
return;
|
||
}
|
||
|
||
s_safeToSwap.store(false, std::memory_order_release);
|
||
std::lock_guard<std::mutex> lock(s_safeToSwapMutex);
|
||
//s_safeToSwap.store(false, std::memory_order_release);
|
||
|
||
// Early exit optimizations
|
||
if (m_clearList) {
|
||
if (!s_isForwardCache.load(std::memory_order_acquire)) {
|
||
clearStaticCacheUnsafe();
|
||
|
||
} else {
|
||
clearStaticCacheUnsafe(true);
|
||
}
|
||
clearItems();
|
||
s_isForwardCache.store(false, std::memory_order_release);
|
||
s_cacheForwardFrameOnce.store(true, std::memory_order_release);
|
||
return;
|
||
}
|
||
{
|
||
std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex);
|
||
// Process pending operations in batch
|
||
if (!m_itemsToAdd.empty()) addPendingItems();
|
||
if (!m_itemsToRemove.empty()) removePendingItems();
|
||
}
|
||
|
||
// Only lock when checking s_lastFrameItems.empty()
|
||
bool shouldResetCache = false;
|
||
{
|
||
std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex);
|
||
if (!s_hasValidFrame.load(std::memory_order_acquire) && s_lastFrameItems.empty() &&
|
||
!s_cacheForwardFrameOnce.load(std::memory_order_acquire)) {
|
||
shouldResetCache = true;
|
||
}
|
||
}
|
||
|
||
if (shouldResetCache) {
|
||
s_cacheForwardFrameOnce.store(true, std::memory_order_release);
|
||
}
|
||
|
||
// This part is for fixing returning to Ultrahand without rendering that first frame skip
|
||
static bool checkOnce = true;
|
||
if (checkOnce && m_pendingJump && !s_hasValidFrame.load(std::memory_order_acquire) &&
|
||
!s_isForwardCache.load(std::memory_order_acquire)) {
|
||
checkOnce = false;
|
||
return;
|
||
} else {
|
||
static bool checkOnce2 = true;
|
||
if (checkOnce2) {
|
||
checkOnce = true;
|
||
checkOnce2 = false;
|
||
}
|
||
}
|
||
|
||
// Check if we should render cached frame
|
||
if ((m_pendingJump || !m_hasForwardCached) &&
|
||
(s_hasValidFrame.load(std::memory_order_acquire) || s_isForwardCache.load(std::memory_order_acquire))) {
|
||
{
|
||
std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex);
|
||
// Render using cached frame state if available
|
||
renderCachedFrame(renderer); // This method handles its own locking
|
||
|
||
// Clear cache after rendering
|
||
if (s_isForwardCache.load(std::memory_order_acquire))
|
||
clearStaticCacheUnsafe(true); // This method handles its own locking
|
||
else
|
||
clearStaticCacheUnsafe(); // This method handles its own locking
|
||
}
|
||
|
||
return;
|
||
}
|
||
|
||
// Cache bounds for hot loop
|
||
const s32 topBound = getTopBound();
|
||
const s32 bottomBound = getBottomBound();
|
||
const s32 height = getHeight();
|
||
|
||
renderer->enableScissoring(getLeftBound(), topBound-8, getWidth() + 8, height + 14);
|
||
|
||
{
|
||
std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex);
|
||
// Optimized visibility culling
|
||
for (Element* entry : m_items) {
|
||
if (entry->getBottomBound() > topBound && entry->getTopBound() < bottomBound) {
|
||
entry->frame(renderer);
|
||
}
|
||
}
|
||
}
|
||
|
||
renderer->disableScissoring();
|
||
|
||
// Draw scrollbar only when needed
|
||
if (m_listHeight > height) {
|
||
drawScrollbar(renderer, height);
|
||
updateScrollAnimation();
|
||
}
|
||
|
||
// Handle caching operations - lock only for the critical section
|
||
{
|
||
std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex);
|
||
|
||
if (!s_isForwardCache.load(std::memory_order_acquire) && s_hasValidFrame.load(std::memory_order_acquire)) {
|
||
// Clear cache after rendering (this is called within the lock)
|
||
clearStaticCacheUnsafe(); // New unsafe version for use within lock
|
||
s_hasValidFrame.store(false, std::memory_order_release);
|
||
s_cacheForwardFrameOnce.store(true, std::memory_order_release);
|
||
}
|
||
|
||
if (!m_cachingDisabled) {
|
||
if (s_cacheForwardFrameOnce.load(std::memory_order_acquire) &&
|
||
!s_hasValidFrame.load(std::memory_order_acquire)) {
|
||
// Cache current frame (this is called within the lock)
|
||
cacheCurrentFrameUnsafe(true); // New unsafe version for use within lock
|
||
s_cacheForwardFrameOnce.store(false, std::memory_order_release);
|
||
s_isForwardCache.store(true, std::memory_order_release);
|
||
s_hasValidFrame.store(true, std::memory_order_release);
|
||
m_hasForwardCached = true;
|
||
}
|
||
cacheCurrentScrollbar();
|
||
}
|
||
|
||
//if (m_cachingDisabled ||(s_hasValidFrame.load(std::memory_order_acquire) && s_isForwardCache.load(std::memory_order_acquire)))
|
||
// s_safeToSwap.store(true, std::memory_order_release);
|
||
}
|
||
s_safeToSwap.store(true, std::memory_order_release);
|
||
}
|
||
|
||
|
||
virtual void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {
|
||
s32 y = getY() - m_offset;
|
||
|
||
// Position all items first (don't calculate m_listHeight here)
|
||
for (Element* entry : m_items) {
|
||
entry->setBoundaries(getX(), y, getWidth(), entry->getHeight());
|
||
entry->invalidate();
|
||
y += entry->getHeight();
|
||
}
|
||
|
||
|
||
// Calculate total height AFTER all invalidations are done
|
||
m_listHeight = BOTTOM_PADDING;
|
||
for (Element* entry : m_items) {
|
||
m_listHeight += entry->getHeight();
|
||
}
|
||
}
|
||
|
||
// Fixed onTouch method - prevents controller state corruption
|
||
virtual bool onTouch(TouchEvent event, s32 currX, s32 currY, s32 prevX, s32 prevY, s32 initialX, s32 initialY) override {
|
||
// Quick bounds check
|
||
if (!inBounds(currX, currY)) return false;
|
||
|
||
// Forward to children first
|
||
for (Element* item : m_items) {
|
||
if (item->onTouch(event, currX, currY, prevX, prevY, initialX, initialY)) {
|
||
return true;
|
||
}
|
||
}
|
||
|
||
// Handle scrolling
|
||
if (event != TouchEvent::Release && Element::getInputMode() == InputMode::TouchScroll) {
|
||
if (prevX && prevY) {
|
||
m_nextOffset += (prevY - currY);
|
||
m_nextOffset = std::clamp(m_nextOffset, 0.0f, static_cast<float>(m_listHeight - getHeight()));
|
||
|
||
// Track that we're touch scrolling
|
||
m_touchScrollActive = true;
|
||
}
|
||
return true;
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
|
||
inline void addItem(Element* element, u16 height = 0, ssize_t index = -1) {
|
||
if (!element) return;
|
||
|
||
// First item optimization
|
||
if (actualItemCount == 0 && element->m_isItem) {
|
||
auto* customDrawer = new tsl::elm::CustomDrawer([](gfx::Renderer*, s32, s32, s32, s32) {});
|
||
customDrawer->setBoundaries(getX(), getY(), getWidth(), 29+4);
|
||
customDrawer->setParent(this);
|
||
customDrawer->invalidate();
|
||
m_itemsToAdd.emplace_back(-1, customDrawer);
|
||
}
|
||
|
||
if (height) {
|
||
element->setBoundaries(getX(), getY(), getWidth(), height);
|
||
}
|
||
|
||
element->setParent(this);
|
||
element->invalidate();
|
||
m_itemsToAdd.emplace_back(index, element);
|
||
++actualItemCount;
|
||
}
|
||
|
||
virtual void removeItem(Element *element) {
|
||
if (element) m_itemsToRemove.push_back(element);
|
||
}
|
||
|
||
virtual void removeIndex(size_t index) {
|
||
if (index < m_items.size()) removeItem(m_items[index]);
|
||
}
|
||
|
||
inline void clear() {
|
||
m_clearList = true;
|
||
}
|
||
|
||
virtual Element* requestFocus(Element* oldFocus, FocusDirection direction) override {
|
||
if (m_clearList || !m_itemsToAdd.empty()) return nullptr;
|
||
|
||
static bool delayedHandle = false;
|
||
|
||
// NEW: Handle pending jump to specific item
|
||
if (m_pendingJump && !delayedHandle) {
|
||
delayedHandle = true;
|
||
return handleJumpToItem(oldFocus);
|
||
} else if (m_pendingJump) {
|
||
m_pendingJump = false;
|
||
delayedHandle = false;
|
||
return handleJumpToItem(oldFocus); // needs to be handled 2x for proper rendering
|
||
}
|
||
|
||
if (jumpToBottom.load(std::memory_order_acquire)) {
|
||
jumpToBottom.store(false, std::memory_order_release);
|
||
return handleJumpToBottom(oldFocus);
|
||
}
|
||
if (jumpToTop.load(std::memory_order_acquire)) {
|
||
jumpToTop.store(false, std::memory_order_release);
|
||
return handleJumpToTop(oldFocus);
|
||
}
|
||
if (skipDown.load(std::memory_order_acquire)) {
|
||
skipDown.store(false, std::memory_order_release);
|
||
return handleSkipDown(oldFocus);
|
||
}
|
||
if (skipUp.load(std::memory_order_acquire)) {
|
||
skipUp.store(false, std::memory_order_release);
|
||
return handleSkipUp(oldFocus);
|
||
}
|
||
|
||
|
||
if (direction == FocusDirection::None) {
|
||
return handleInitialFocus(oldFocus);
|
||
}
|
||
else if (direction == FocusDirection::Down) {
|
||
return handleDownFocus(oldFocus);
|
||
}
|
||
else if (direction == FocusDirection::Up) {
|
||
return handleUpFocus(oldFocus);
|
||
}
|
||
|
||
return oldFocus;
|
||
}
|
||
|
||
inline void jumpToItem(const std::string& text = "", const std::string& value = "", bool exactMatch=true) {
|
||
//if (!text.empty() || !value.empty())
|
||
// m_pendingJump = true;
|
||
//else
|
||
// m_pendingJump = false;
|
||
//if (g_overlayFilename == "ovlmenu.ovl") return;
|
||
if (!text.empty() || !value.empty()) {
|
||
m_pendingJump = true;
|
||
m_jumpToText = text;
|
||
m_jumpToValue = value;
|
||
m_jumpToExactMatch = exactMatch;
|
||
}
|
||
|
||
}
|
||
|
||
virtual Element* getItemAtIndex(u32 index) {
|
||
return (m_items.size() <= index) ? nullptr : m_items[index];
|
||
}
|
||
|
||
virtual s32 getIndexInList(Element *element) {
|
||
auto it = std::find(m_items.begin(), m_items.end(), element);
|
||
return (it == m_items.end()) ? -1 : static_cast<s32>(it - m_items.begin());
|
||
}
|
||
|
||
virtual s32 getLastIndex() {
|
||
return static_cast<s32>(m_items.size()) - 1;
|
||
}
|
||
|
||
virtual void setFocusedIndex(u32 index) {
|
||
if (m_items.size() > index) {
|
||
m_focusedIndex = index;
|
||
updateScrollOffset();
|
||
}
|
||
}
|
||
|
||
inline void onDirectionalKeyReleased() {
|
||
m_hasWrappedInCurrentSequence = false;
|
||
m_lastNavigationResult = NavigationResult::None;
|
||
m_isHolding = false;
|
||
m_stoppedAtBoundary = false;
|
||
m_lastNavigationTime = 0;
|
||
m_lastScrollTime = 0;
|
||
}
|
||
|
||
inline void disableCaching() {
|
||
m_cachingDisabled = true;
|
||
}
|
||
|
||
protected:
|
||
|
||
std::vector<Element*> m_items;
|
||
u16 m_focusedIndex = 0;
|
||
|
||
float m_offset = 0, m_nextOffset = 0;
|
||
s32 m_listHeight = 0;
|
||
|
||
bool m_clearList = false;
|
||
std::vector<Element*> m_itemsToRemove;
|
||
std::vector<std::pair<ssize_t, Element*>> m_itemsToAdd;
|
||
std::vector<float> prefixSums;
|
||
|
||
// Instance identification
|
||
//const size_t m_instanceId;
|
||
|
||
// Enhanced navigation state tracking
|
||
bool m_justWrapped = false;
|
||
bool m_isHolding = false;
|
||
bool m_stoppedAtBoundary = false;
|
||
u64 m_lastNavigationTime = 0;
|
||
static constexpr u64 HOLD_THRESHOLD_NS = 100000000ULL; // 100ms
|
||
|
||
size_t actualItemCount = 0;
|
||
|
||
// Jump to navigation variables
|
||
std::string m_jumpToText;
|
||
std::string m_jumpToValue;
|
||
bool m_jumpToExactMatch = false;
|
||
bool m_pendingJump = false;
|
||
bool m_hasForwardCached = false;
|
||
bool m_cachingDisabled = false; // New flag to disable caching
|
||
|
||
//bool m_hasRenderedCache = false;
|
||
|
||
// Stack variables for hot path - reused to avoid allocations
|
||
u32 scrollbarHeight;
|
||
u32 scrollbarOffset;
|
||
u32 prevOffset;
|
||
static constexpr float SCROLLBAR_X_OFFSET = 21.0f;
|
||
static constexpr float SCROLLBAR_Y_OFFSET = 3.0f;
|
||
static constexpr float SCROLLBAR_HEIGHT_TRIM = 6.0f;
|
||
|
||
//static constexpr float smoothingFactor = 0.15f;
|
||
//static constexpr float dampingFactor = 0.3f;
|
||
static constexpr float TABLE_SCROLL_STEP_SIZE = 10;
|
||
static constexpr float TABLE_SCROLL_STEP_SIZE_CLICK = 22;
|
||
static constexpr float BOTTOM_PADDING = 7.0f;
|
||
static constexpr float VIEW_CENTER_OFFSET = 7.0f;
|
||
|
||
u64 m_lastScrollTime = 0;
|
||
|
||
float m_scrollVelocity = 0.0f;
|
||
|
||
bool m_touchScrollActive = false;
|
||
|
||
enum class NavigationResult {
|
||
None,
|
||
Success,
|
||
HitBoundary,
|
||
Wrapped
|
||
};
|
||
|
||
bool m_hasWrappedInCurrentSequence = false;
|
||
NavigationResult m_lastNavigationResult = NavigationResult::None;
|
||
|
||
private:
|
||
// Method to explicitly preserve cache when navigating away
|
||
//void preserveCacheForReturn() {
|
||
// if (m_instanceId == s_cachedInstanceId && s_hasValidFrame) {
|
||
// // Cache is already preserved for this instance
|
||
// return;
|
||
// }
|
||
// cacheCurrentFrame();
|
||
//}
|
||
|
||
// Method to check if this instance has a valid cached frame
|
||
//bool hasCachedFrame() const {
|
||
// return s_hasValidFrame && s_cachedInstanceId == m_instanceId;
|
||
//}
|
||
|
||
//static size_t generateInstanceId() {
|
||
// return s_nextInstanceId++;
|
||
//}
|
||
|
||
// Thread-safe versions (handle their own locking)
|
||
static void clearStaticCache(bool preservePointers = false) {
|
||
std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex);
|
||
clearStaticCacheUnsafe(preservePointers);
|
||
}
|
||
|
||
void cacheCurrentFrame(bool preservePointers = false) {
|
||
std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex);
|
||
cacheCurrentFrameUnsafe(preservePointers);
|
||
}
|
||
|
||
|
||
static void clearStaticCacheUnsafe(bool preservePointers = false) {
|
||
//std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex);
|
||
if (!preservePointers) {
|
||
// Normal case: delete elements and clear
|
||
for (Element* el : s_lastFrameItems) {
|
||
delete el;
|
||
}
|
||
}
|
||
|
||
s_lastFrameItems.clear();
|
||
//s_lastFrameItems.shrink_to_fit();
|
||
|
||
// CRITICAL: Always reset these, even for forward cache!
|
||
s_hasValidFrame.store(false, std::memory_order_release); // This MUST be false after clearing
|
||
s_isForwardCache.store(false, std::memory_order_release);
|
||
|
||
s_cachedTopBound.store(0, std::memory_order_release);
|
||
s_cachedBottomBound.store(0, std::memory_order_release);
|
||
s_cachedHeight.store(0, std::memory_order_release);
|
||
s_cachedListHeight.store(0, std::memory_order_release);
|
||
s_cachedActualContentBottom.store(0, std::memory_order_release);
|
||
|
||
s_shouldDrawScrollbar.store(false, std::memory_order_release);
|
||
s_cachedScrollbarHeight.store(0, std::memory_order_release);
|
||
s_cachedScrollbarOffset.store(0, std::memory_order_release);
|
||
s_cachedScrollbarX.store(0, std::memory_order_release);
|
||
s_cachedScrollbarY.store(0, std::memory_order_release);
|
||
}
|
||
|
||
void cacheCurrentFrameUnsafe(bool preservePointers = false) {
|
||
//std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex);
|
||
if (!preservePointers) {
|
||
for (Element* el : s_lastFrameItems) delete el;
|
||
}
|
||
|
||
s_lastFrameItems = m_items;
|
||
|
||
// Store new cache values using atomic stores
|
||
s_cachedTopBound.store(getTopBound(), std::memory_order_release);
|
||
s_cachedBottomBound.store(getBottomBound(), std::memory_order_release);
|
||
s_cachedHeight.store(getHeight(), std::memory_order_release);
|
||
s_cachedListHeight.store(m_listHeight, std::memory_order_release);
|
||
|
||
if (preservePointers)
|
||
s_isForwardCache.store(true, std::memory_order_release);
|
||
|
||
s_hasValidFrame.store(true, std::memory_order_release);
|
||
}
|
||
|
||
void cacheCurrentScrollbar() {
|
||
const s32 cachedHeight = s_cachedHeight.load(std::memory_order_acquire);
|
||
const s32 cachedListHeight = s_cachedListHeight.load(std::memory_order_acquire);
|
||
|
||
s_shouldDrawScrollbar.store((cachedListHeight > cachedHeight), std::memory_order_release);
|
||
|
||
if (s_shouldDrawScrollbar.load(std::memory_order_acquire)) {
|
||
const float viewHeight = static_cast<float>(cachedHeight);
|
||
const float totalHeight = static_cast<float>(cachedListHeight);
|
||
const u32 maxScroll = std::max(static_cast<u32>(totalHeight - viewHeight), 1u);
|
||
|
||
u32 scrollbarHeight = std::min(
|
||
static_cast<u32>((viewHeight * viewHeight) / totalHeight),
|
||
static_cast<u32>(viewHeight)
|
||
);
|
||
|
||
u32 scrollbarOffset = std::min(
|
||
static_cast<u32>((m_offset / maxScroll) * (viewHeight - scrollbarHeight)),
|
||
static_cast<u32>(viewHeight - scrollbarHeight) // corrected potential bug
|
||
);
|
||
|
||
scrollbarHeight -= SCROLLBAR_HEIGHT_TRIM;
|
||
|
||
s_cachedScrollbarHeight.store(scrollbarHeight, std::memory_order_release);
|
||
s_cachedScrollbarOffset.store(scrollbarOffset, std::memory_order_release);
|
||
s_cachedScrollbarX.store(getRightBound() + SCROLLBAR_X_OFFSET, std::memory_order_release);
|
||
s_cachedScrollbarY.store(getY() + scrollbarOffset + SCROLLBAR_Y_OFFSET, std::memory_order_release);
|
||
}
|
||
}
|
||
|
||
void renderCachedFrame(gfx::Renderer* renderer) {
|
||
const s32 cachedTopBound = s_cachedTopBound.load(std::memory_order_acquire);
|
||
const s32 cachedBottomBound = s_cachedBottomBound.load(std::memory_order_acquire);
|
||
const s32 cachedHeight = s_cachedHeight.load(std::memory_order_acquire);
|
||
|
||
renderer->enableScissoring(getLeftBound(), cachedTopBound - 8, getWidth() + 8, cachedHeight + 14);
|
||
|
||
for (Element* entry : s_lastFrameItems) {
|
||
if (entry &&
|
||
entry->getBottomBound() > cachedTopBound &&
|
||
entry->getTopBound() < cachedBottomBound) {
|
||
entry->frame(renderer);
|
||
}
|
||
}
|
||
|
||
renderer->disableScissoring();
|
||
|
||
if (s_shouldDrawScrollbar.load(std::memory_order_acquire)) {
|
||
const u32 scrollbarX = s_cachedScrollbarX.load(std::memory_order_acquire);
|
||
const u32 scrollbarY = s_cachedScrollbarY.load(std::memory_order_acquire);
|
||
const u32 scrollbarHeight = s_cachedScrollbarHeight.load(std::memory_order_acquire);
|
||
|
||
renderer->drawRect(scrollbarX, scrollbarY, 5, scrollbarHeight, a(trackBarColor));
|
||
renderer->drawCircle(scrollbarX + 2, scrollbarY, 2, true, a(trackBarColor));
|
||
renderer->drawCircle(scrollbarX + 2, scrollbarY + scrollbarHeight, 2, true, a(trackBarColor));
|
||
}
|
||
}
|
||
|
||
|
||
void clearItems() {
|
||
//std::lock_guard<std::mutex> lock(s_lastFrameItemsMutex);
|
||
// Clear static cache if it belongs to this instance
|
||
//if (s_cachedInstanceId == m_instanceId) {
|
||
// clearStaticCache();
|
||
//}
|
||
|
||
for (Element* item : m_items) delete item;
|
||
m_items = {};
|
||
//m_items.clear();
|
||
//m_items.shrink_to_fit();
|
||
m_offset = 0;
|
||
m_focusedIndex = 0;
|
||
invalidate();
|
||
m_clearList = false;
|
||
actualItemCount = 0;
|
||
}
|
||
|
||
void addPendingItems() {
|
||
for (auto [index, element] : m_itemsToAdd) {
|
||
element->invalidate();
|
||
if (index >= 0 && static_cast<size_t>(index) < m_items.size()) {
|
||
m_items.insert(m_items.begin() + index, element);
|
||
} else {
|
||
m_items.push_back(element);
|
||
}
|
||
}
|
||
m_itemsToAdd = {};
|
||
//m_itemsToAdd.clear();
|
||
//m_itemsToAdd.shrink_to_fit();
|
||
invalidate();
|
||
updateScrollOffset();
|
||
}
|
||
|
||
void removePendingItems() {
|
||
//size_t index;
|
||
for (Element* element : m_itemsToRemove) {
|
||
auto it = std::find(m_items.begin(), m_items.end(), element);
|
||
if (it != m_items.end()) {
|
||
const size_t index = static_cast<size_t>(it - m_items.begin());
|
||
m_items.erase(it);
|
||
if (m_focusedIndex >= index && m_focusedIndex > 0) {
|
||
--m_focusedIndex;
|
||
}
|
||
delete element;
|
||
}
|
||
}
|
||
m_itemsToRemove = {};
|
||
//m_itemsToRemove.clear();
|
||
//m_itemsToRemove.shrink_to_fit();
|
||
invalidate();
|
||
updateScrollOffset();
|
||
}
|
||
|
||
void purgePendingItems() {
|
||
for (auto& [_, element] : m_itemsToAdd) {
|
||
if (element) { element->invalidate(); delete element; }
|
||
}
|
||
m_itemsToAdd = {};
|
||
//m_itemsToAdd.clear();
|
||
//m_itemsToAdd.shrink_to_fit();
|
||
|
||
//size_t index;
|
||
for (Element* element : m_itemsToRemove) {
|
||
auto it = std::find(m_items.begin(), m_items.end(), element);
|
||
if (it != m_items.end()) {
|
||
//index = static_cast<std::size_t>(it - m_items.begin());
|
||
const u16 index16 = static_cast<u16>(static_cast<std::size_t>(it - m_items.begin()));
|
||
element->invalidate();
|
||
delete element;
|
||
m_items.erase(it);
|
||
|
||
constexpr u16 noFocus = static_cast<u16>(0xFFFF);
|
||
if (m_focusedIndex == index16)
|
||
m_focusedIndex = noFocus;
|
||
else if (m_focusedIndex != noFocus && m_focusedIndex > index16)
|
||
--m_focusedIndex;
|
||
}
|
||
}
|
||
m_itemsToRemove = {};
|
||
//m_itemsToRemove.clear();
|
||
//m_itemsToRemove.shrink_to_fit();
|
||
|
||
invalidate();
|
||
updateScrollOffset();
|
||
}
|
||
|
||
|
||
void drawScrollbar(gfx::Renderer* renderer, s32 height) {
|
||
const float viewHeight = static_cast<float>(height);
|
||
const float totalHeight = static_cast<float>(m_listHeight);
|
||
const u32 maxScrollableHeight = std::max(static_cast<u32>(totalHeight - viewHeight), 1u);
|
||
|
||
scrollbarHeight = std::min(static_cast<u32>((viewHeight * viewHeight) / totalHeight),
|
||
static_cast<u32>(viewHeight));
|
||
|
||
scrollbarOffset = std::min(static_cast<u32>((m_offset / maxScrollableHeight) * (viewHeight - scrollbarHeight)),
|
||
static_cast<u32>(viewHeight - scrollbarHeight));
|
||
|
||
const u32 scrollbarX = getRightBound() + SCROLLBAR_X_OFFSET;
|
||
const u32 scrollbarY = getY() + scrollbarOffset+SCROLLBAR_Y_OFFSET;
|
||
|
||
scrollbarHeight -= SCROLLBAR_HEIGHT_TRIM; // shorten very slightly
|
||
|
||
renderer->drawRect(scrollbarX, scrollbarY, 5, scrollbarHeight, a(trackBarColor));
|
||
renderer->drawCircle(scrollbarX + 2, scrollbarY, 2, true, a(trackBarColor));
|
||
renderer->drawCircle(scrollbarX + 2, scrollbarY + scrollbarHeight, 2, true, a(trackBarColor));
|
||
}
|
||
|
||
|
||
inline void updateScrollAnimation() {
|
||
if (Element::getInputMode() == InputMode::Controller) {
|
||
// Clear touch flag when in controller mode
|
||
m_touchScrollActive = false;
|
||
|
||
// Calculate distance to target
|
||
const float diff = m_nextOffset - m_offset;
|
||
const float distance = std::abs(diff);
|
||
|
||
// ENHANCED BOUNDARY SNAPPING: More aggressive snapping for boundaries
|
||
if (distance < 1.0f) { // Increased threshold from 0.5f
|
||
m_offset = m_nextOffset;
|
||
m_scrollVelocity = 0.0f;
|
||
s_currentScrollVelocity.store(m_scrollVelocity, std::memory_order_release);
|
||
|
||
if (prevOffset != m_offset) {
|
||
invalidate();
|
||
prevOffset = m_offset;
|
||
}
|
||
return;
|
||
}
|
||
|
||
// SPECIAL CASE: If target is exactly 0 or max, be more aggressive
|
||
const float maxOffset = static_cast<float>(m_listHeight - getHeight());
|
||
if (m_nextOffset == 0.0f || m_nextOffset == maxOffset) {
|
||
if (distance < 3.0f) { // Larger snap zone for boundaries
|
||
m_offset = m_nextOffset;
|
||
m_scrollVelocity = 0.0f;
|
||
s_currentScrollVelocity.store(m_scrollVelocity, std::memory_order_release);
|
||
|
||
if (prevOffset != m_offset) {
|
||
invalidate();
|
||
prevOffset = m_offset;
|
||
}
|
||
|
||
return;
|
||
}
|
||
}
|
||
|
||
// Emergency correction if item is going out of bounds
|
||
if (m_focusedIndex < m_items.size()) {
|
||
float itemTop = 0.0f;
|
||
for (size_t i = 0; i < m_focusedIndex; ++i) {
|
||
itemTop += m_items[i]->getHeight();
|
||
}
|
||
const float itemBottom = itemTop + m_items[m_focusedIndex]->getHeight();
|
||
|
||
//float viewTop = m_offset;
|
||
const float viewBottom = m_offset + getHeight();
|
||
|
||
if (itemTop < m_offset || itemBottom > viewBottom) {
|
||
float emergencySpeed = 0.6f;
|
||
|
||
if (itemBottom < m_offset || itemTop > viewBottom) {
|
||
emergencySpeed = 0.9f;
|
||
}
|
||
|
||
m_offset += diff * emergencySpeed;
|
||
m_scrollVelocity = diff * 0.3f;
|
||
s_currentScrollVelocity.store(m_scrollVelocity, std::memory_order_release);
|
||
|
||
if (prevOffset != m_offset) {
|
||
invalidate();
|
||
prevOffset = m_offset;
|
||
}
|
||
return;
|
||
}
|
||
}
|
||
|
||
// Rest of your existing smooth scrolling logic...
|
||
const bool isLargeJump = distance > getHeight() * 1.5f;
|
||
const bool isFromRest = std::abs(m_scrollVelocity) < 2.0f;
|
||
|
||
if (isLargeJump && isFromRest) {
|
||
static constexpr float gentleAcceleration = 0.08f;
|
||
static constexpr float gentleDamping = 0.85f;
|
||
|
||
const float targetVelocity = diff * gentleAcceleration;
|
||
m_scrollVelocity += (targetVelocity - m_scrollVelocity) * gentleDamping;
|
||
} else {
|
||
const float urgency = std::min(distance / getHeight(), 1.0f);
|
||
const float accelerationFactor = 0.18f + (0.24f * urgency);
|
||
const float dampingFactor = 0.48f - (0.18f * urgency);
|
||
|
||
const float targetVelocity = diff * accelerationFactor;
|
||
m_scrollVelocity += (targetVelocity - m_scrollVelocity) * dampingFactor;
|
||
}
|
||
|
||
// Apply velocity
|
||
m_offset += m_scrollVelocity;
|
||
|
||
// ENHANCED overshoot prevention with better boundary handling
|
||
if ((m_scrollVelocity > 0 && m_offset > m_nextOffset) ||
|
||
(m_scrollVelocity < 0 && m_offset < m_nextOffset)) {
|
||
m_offset = m_nextOffset;
|
||
m_scrollVelocity = 0.0f;
|
||
}
|
||
|
||
// ADDITIONAL: Force exact boundary values
|
||
if (m_nextOffset == 0.0f && m_offset < 1.0f) {
|
||
m_offset = 0.0f;
|
||
m_scrollVelocity = 0.0f;
|
||
} else if (m_nextOffset == maxOffset && m_offset > maxOffset - 1.0f) {
|
||
m_offset = maxOffset;
|
||
m_scrollVelocity = 0.0f;
|
||
}
|
||
|
||
s_currentScrollVelocity.store(m_scrollVelocity, std::memory_order_release);
|
||
|
||
} else if (Element::getInputMode() == InputMode::TouchScroll) {
|
||
// Your existing touch scroll logic...
|
||
m_offset = m_nextOffset;
|
||
m_scrollVelocity = 0.0f;
|
||
|
||
if (m_touchScrollActive) {
|
||
const float viewCenter = m_offset + (getHeight() / 2.0f);
|
||
float accumHeight = 0.0f;
|
||
|
||
float itemHeight, itemCenter;
|
||
for (size_t i = 0; i < m_items.size(); ++i) {
|
||
itemHeight = m_items[i]->getHeight();
|
||
itemCenter = accumHeight + (itemHeight / 2.0f);
|
||
|
||
if (itemCenter >= viewCenter) {
|
||
m_focusedIndex = i;
|
||
break;
|
||
}
|
||
|
||
accumHeight += itemHeight;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (prevOffset != m_offset) {
|
||
invalidate();
|
||
prevOffset = m_offset;
|
||
}
|
||
}
|
||
|
||
Element* handleInitialFocus(Element* oldFocus) {
|
||
const size_t itemCount = m_items.size();
|
||
if (itemCount == 0) return nullptr;
|
||
|
||
size_t startIndex = 0;
|
||
|
||
// Calculate starting index based on current scroll position
|
||
if (!oldFocus && m_offset > 0) {
|
||
float elementHeight = 0.0f;
|
||
const size_t maxIndex = itemCount - 1;
|
||
while (elementHeight < m_offset && startIndex < maxIndex) {
|
||
elementHeight += m_items[startIndex]->getHeight();
|
||
++startIndex;
|
||
}
|
||
}
|
||
|
||
//resetNavigationState();
|
||
|
||
// Save current offset to prevent scroll jumping
|
||
const float savedOffset = m_offset;
|
||
const float savedNextOffset = m_nextOffset;
|
||
|
||
// Single loop with wraparound logic - visits each item exactly once
|
||
for (size_t count = 0; count < itemCount; ++count) {
|
||
const size_t i = (startIndex + count) % itemCount;
|
||
|
||
if (!m_items[i]->isTable()) {
|
||
Element* const newFocus = m_items[i]->requestFocus(oldFocus, FocusDirection::None);
|
||
if (newFocus && newFocus != oldFocus) {
|
||
m_focusedIndex = i;
|
||
m_offset = savedOffset;
|
||
m_nextOffset = savedNextOffset;
|
||
return newFocus;
|
||
}
|
||
}
|
||
}
|
||
|
||
return nullptr;
|
||
}
|
||
|
||
|
||
inline Element* handleDownFocus(Element* oldFocus) {
|
||
static bool triggerShakeOnce = true;
|
||
updateHoldState();
|
||
|
||
// Check if the next item is non-focusable BEFORE we do anything else
|
||
if (m_focusedIndex + 1 < int(m_items.size())) {
|
||
Element* nextItem = m_items[m_focusedIndex + 1];
|
||
if (!nextItem->m_isItem) {
|
||
isTableScrolling.store(true, std::memory_order_release); // Set this IMMEDIATELY
|
||
}
|
||
}
|
||
|
||
// If holding and at boundary, try to scroll first
|
||
if (m_isHolding && m_stoppedAtBoundary && !isAtBottom()) {
|
||
scrollDown();
|
||
m_stoppedAtBoundary = false;
|
||
return oldFocus;
|
||
}
|
||
|
||
Element* result = navigateDown(oldFocus);
|
||
|
||
if (result != oldFocus) {
|
||
m_lastNavigationResult = NavigationResult::Success;
|
||
m_stoppedAtBoundary = false;
|
||
triggerShakeOnce = true;
|
||
return result;
|
||
}
|
||
|
||
// Check if we can still scroll down
|
||
if (!isAtBottom()) {
|
||
scrollDown();
|
||
return oldFocus;
|
||
}
|
||
|
||
// At absolute bottom - check for wrapping
|
||
if (!m_isHolding && !m_hasWrappedInCurrentSequence && isAtBottom()) {
|
||
//if (s_directionalKeyReleased.load(std::memory_order_acquire)) {
|
||
// s_directionalKeyReleased.store(false, std::memory_order_release);
|
||
// m_hasWrappedInCurrentSequence = true;
|
||
// m_lastNavigationResult = NavigationResult::Wrapped;
|
||
// triggerShakeOnce = true; // Reset when wrapping
|
||
// return handleJumpToTop(oldFocus);
|
||
//} else
|
||
// s_directionalKeyReleased.store(false, std::memory_order_release);
|
||
|
||
s_directionalKeyReleased.store(false, std::memory_order_release);
|
||
m_hasWrappedInCurrentSequence = true;
|
||
m_lastNavigationResult = NavigationResult::Wrapped;
|
||
triggerShakeOnce = true; // Reset when wrapping
|
||
return handleJumpToTop(oldFocus);
|
||
}
|
||
|
||
// Set boundary flag
|
||
if (m_isHolding) {
|
||
m_stoppedAtBoundary = true;
|
||
if (triggerShakeOnce) {
|
||
// Find the last focusable item to shake
|
||
for (ssize_t i = static_cast<ssize_t>(m_focusedIndex); i >= 0; --i) {
|
||
if (m_items[i]->m_isItem) {
|
||
m_items[i]->shakeHighlight(FocusDirection::Down);
|
||
break;
|
||
}
|
||
}
|
||
triggerShakeOnce = false;
|
||
}
|
||
} else
|
||
triggerShakeOnce = true;
|
||
|
||
m_lastNavigationResult = NavigationResult::HitBoundary;
|
||
return oldFocus;
|
||
}
|
||
|
||
inline Element* handleUpFocus(Element* oldFocus) {
|
||
static bool triggerShakeOnce = true;
|
||
updateHoldState();
|
||
|
||
// Check if the previous item is non-focusable BEFORE we do anything else
|
||
if (m_focusedIndex > 0) {
|
||
Element* prevItem = m_items[m_focusedIndex - 1];
|
||
if (prevItem->isTable()) {
|
||
isTableScrolling.store(true, std::memory_order_release); // Set this IMMEDIATELY
|
||
}
|
||
}
|
||
|
||
// If holding and at boundary, try to scroll first
|
||
if (m_isHolding && m_stoppedAtBoundary && !isAtTop()) {
|
||
scrollUp();
|
||
m_stoppedAtBoundary = false;
|
||
return oldFocus;
|
||
}
|
||
|
||
Element* result = navigateUp(oldFocus);
|
||
|
||
if (result != oldFocus) {
|
||
m_lastNavigationResult = NavigationResult::Success;
|
||
m_stoppedAtBoundary = false;
|
||
triggerShakeOnce = true;
|
||
return result;
|
||
}
|
||
|
||
// Check if we can still scroll up
|
||
if (!isAtTop()) {
|
||
scrollUp();
|
||
return oldFocus;
|
||
}
|
||
|
||
// At absolute top - check for wrapping
|
||
if (!m_isHolding && !m_hasWrappedInCurrentSequence && isAtTop()) {
|
||
//if (s_directionalKeyReleased.load(std::memory_order_acquire)) {
|
||
// s_directionalKeyReleased.store(false, std::memory_order_release);
|
||
// m_hasWrappedInCurrentSequence = true;
|
||
// m_lastNavigationResult = NavigationResult::Wrapped;
|
||
// triggerShakeOnce = true; // Reset when wrapping
|
||
// return handleJumpToBottom(oldFocus);
|
||
//} else
|
||
// s_directionalKeyReleased.store(false, std::memory_order_release);
|
||
|
||
s_directionalKeyReleased.store(false, std::memory_order_release);
|
||
m_hasWrappedInCurrentSequence = true;
|
||
m_lastNavigationResult = NavigationResult::Wrapped;
|
||
triggerShakeOnce = true; // Reset when wrapping
|
||
return handleJumpToBottom(oldFocus);
|
||
}
|
||
|
||
// Set boundary flag
|
||
if (m_isHolding) {
|
||
m_stoppedAtBoundary = true;
|
||
if (triggerShakeOnce) {
|
||
// Find the first focusable item to shake
|
||
for (size_t i = m_focusedIndex; i < m_items.size(); ++i) {
|
||
if (m_items[i]->m_isItem) {
|
||
m_items[i]->shakeHighlight(FocusDirection::Up);
|
||
break;
|
||
}
|
||
}
|
||
triggerShakeOnce = false;
|
||
}
|
||
} else
|
||
triggerShakeOnce = true;
|
||
|
||
m_lastNavigationResult = NavigationResult::HitBoundary;
|
||
return oldFocus;
|
||
}
|
||
|
||
|
||
inline bool isAtTop() {
|
||
if (m_items.empty()) return true;
|
||
|
||
// Check if we're at scroll position 0
|
||
if (m_offset != 0.0f) return false;
|
||
|
||
// Even at offset 0, check if the first item is actually visible
|
||
// This handles cases where the first item might be partially above viewport
|
||
if (!m_items.empty()) {
|
||
Element* firstItem = m_items[0];
|
||
return firstItem->getTopBound() >= getTopBound();
|
||
}
|
||
|
||
return true;
|
||
}
|
||
|
||
inline bool isAtBottom() {
|
||
if (m_items.empty()) return true;
|
||
|
||
// First check: are we at the maximum scroll offset?
|
||
//float maxOffset = static_cast<float>(m_listHeight - getHeight());
|
||
const bool atMaxOffset = (m_offset >= static_cast<float>(m_listHeight - getHeight()));
|
||
|
||
// If list is shorter than viewport, we're always at bottom
|
||
if (m_listHeight <= getHeight()) return true;
|
||
|
||
// If we're not at max offset, we're definitely not at bottom
|
||
if (!atMaxOffset) return false;
|
||
|
||
// At max offset - now check if the last item is actually fully visible
|
||
// This prevents wrap-around when there's still content below viewport
|
||
if (!m_items.empty()) {
|
||
Element* lastItem = m_items.back();
|
||
//s32 lastItemBottom = lastItem->getBottomBound();
|
||
//s32 viewportBottom = getBottomBound();
|
||
|
||
// We're truly at bottom only if:
|
||
// 1. We're at max scroll offset AND
|
||
// 2. The last item's bottom is at or above the viewport bottom
|
||
return lastItem->getBottomBound() <= getBottomBound();
|
||
}
|
||
|
||
return atMaxOffset;
|
||
}
|
||
|
||
// Helper to check if there are any focusable items
|
||
inline bool hasAnyFocusableItems() {
|
||
for (size_t i = 0; i < m_items.size(); ++i) {
|
||
//Element* test = m_items[i]->requestFocus(nullptr, FocusDirection::None);
|
||
//
|
||
//if (test) return true;
|
||
if (m_items[i]->m_isItem) return true;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
|
||
inline void updateHoldState() {
|
||
const u64 currentTime = armTicksToNs(armGetSystemTick());
|
||
if ((m_lastNavigationTime != 0 && (currentTime - m_lastNavigationTime) < HOLD_THRESHOLD_NS)) {
|
||
m_isHolding = true;
|
||
} else {
|
||
m_isHolding = false;
|
||
m_stoppedAtBoundary = false;
|
||
m_hasWrappedInCurrentSequence = false;
|
||
}
|
||
m_lastNavigationTime = currentTime;
|
||
}
|
||
|
||
inline void resetNavigationState() {
|
||
m_hasWrappedInCurrentSequence = false;
|
||
m_lastNavigationResult = NavigationResult::None;
|
||
m_isHolding = false;
|
||
m_stoppedAtBoundary = false;
|
||
m_lastNavigationTime = 0;
|
||
}
|
||
|
||
inline Element* handleJumpToItem(Element* oldFocus) {
|
||
resetNavigationState();
|
||
invalidate();
|
||
|
||
const bool needsScroll = m_listHeight > getHeight();
|
||
const float viewHeight = static_cast<float>(getHeight());
|
||
const float maxOffset = needsScroll ? m_listHeight - viewHeight : 0.0f;
|
||
|
||
float h = 0.0f;
|
||
|
||
//float itemHeight, itemCenterPos, viewportCenter, idealOffset;
|
||
|
||
for (size_t i = 0; i < m_items.size(); ++i) {
|
||
m_focusedIndex = i;
|
||
|
||
Element* newFocus = m_items[i]->requestFocus(oldFocus, FocusDirection::Down);
|
||
if (newFocus && newFocus != oldFocus && m_items[i]->matchesJumpCriteria(m_jumpToText, m_jumpToValue, m_jumpToExactMatch)) {
|
||
// CHANGED: Calculate center of the item and center it in viewport
|
||
const float itemHeight = m_items[i]->getHeight();
|
||
// For middle items, use centering logic
|
||
const float itemCenterPos = h + (itemHeight / 2.0f); // FIXED: Use center, not bottom
|
||
const float viewportCenter = viewHeight / 2.0f + VIEW_CENTER_OFFSET + 0.5f; // Same offset as updateScrollOffset
|
||
//float idealOffset = itemCenterPos - viewportCenter;
|
||
|
||
// Clamp to valid bounds (same as updateScrollOffset)
|
||
const float idealOffset = std::max(0.0f, std::min(itemCenterPos - viewportCenter, maxOffset));
|
||
|
||
// Set both current and target offset
|
||
m_offset = m_nextOffset = idealOffset;
|
||
|
||
return newFocus;
|
||
}
|
||
|
||
h += m_items[i]->getHeight();
|
||
}
|
||
|
||
// No match found
|
||
return handleInitialFocus(oldFocus);
|
||
}
|
||
|
||
// Core navigation logic
|
||
// Optimized version with variable definitions pulled outside the loop
|
||
inline Element* navigateDown(Element* oldFocus) {
|
||
size_t searchIndex = m_focusedIndex + 1;
|
||
|
||
// If currently on a table that needs more scrolling
|
||
if (m_focusedIndex < m_items.size() && m_items[m_focusedIndex]->isTable()) {
|
||
Element* currentTable = m_items[m_focusedIndex];
|
||
if (currentTable->getBottomBound() > getBottomBound()) {
|
||
isTableScrolling.store(true, std::memory_order_release);
|
||
scrollDown();
|
||
return oldFocus;
|
||
}
|
||
}
|
||
|
||
// Cache invariant values (legitimate optimization)
|
||
const s32 viewBottom = getBottomBound();
|
||
const float containerHeight = getHeight();
|
||
const float offsetPlusHeight = m_offset + containerHeight;
|
||
|
||
while (searchIndex < m_items.size()) {
|
||
Element* item = m_items[searchIndex];
|
||
m_focusedIndex = searchIndex;
|
||
|
||
if (item->isTable()) {
|
||
// Table needs scrolling
|
||
const s32 tableBottom = item->getBottomBound();
|
||
if (tableBottom > viewBottom) {
|
||
isTableScrolling.store(true, std::memory_order_release);
|
||
scrollDown();
|
||
return oldFocus;
|
||
}
|
||
searchIndex++;
|
||
continue;
|
||
}
|
||
|
||
// Try to focus this item
|
||
Element* newFocus = item->requestFocus(oldFocus, FocusDirection::Down);
|
||
if (newFocus && newFocus != oldFocus) {
|
||
// ONLY reset when we successfully focus something
|
||
isTableScrolling.store(false, std::memory_order_release);
|
||
updateScrollOffset();
|
||
return newFocus;
|
||
} else {
|
||
// Non-focusable item (gap/header)
|
||
const float itemBottom = calculateItemPosition(searchIndex) + item->getHeight();
|
||
if (itemBottom > offsetPlusHeight) {
|
||
isTableScrolling.store(true, std::memory_order_release); // Treat gaps/headers like tables
|
||
scrollDown();
|
||
return oldFocus;
|
||
}
|
||
searchIndex++;
|
||
}
|
||
}
|
||
|
||
return oldFocus;
|
||
}
|
||
|
||
inline Element* navigateUp(Element* oldFocus) {
|
||
if (m_focusedIndex == 0) return oldFocus;
|
||
ssize_t searchIndex = static_cast<ssize_t>(m_focusedIndex) - 1;
|
||
|
||
// If currently on a table that needs more scrolling
|
||
if (m_focusedIndex < m_items.size() && m_items[m_focusedIndex]->isTable()) {
|
||
Element* currentTable = m_items[m_focusedIndex];
|
||
if (currentTable->getTopBound() < getTopBound()) {
|
||
isTableScrolling.store(true, std::memory_order_release);
|
||
scrollUp();
|
||
return oldFocus;
|
||
}
|
||
}
|
||
|
||
// Cache invariant values (legitimate optimization)
|
||
const s32 viewTop = getTopBound();
|
||
const float offset = m_offset; // Cache in case m_offset is volatile or has accessor overhead
|
||
|
||
while (searchIndex >= 0) {
|
||
Element* item = m_items[searchIndex];
|
||
m_focusedIndex = static_cast<size_t>(searchIndex);
|
||
|
||
if (item->isTable()) {
|
||
// Table needs scrolling
|
||
const s32 tableTop = item->getTopBound();
|
||
if (tableTop < viewTop) {
|
||
isTableScrolling.store(true, std::memory_order_release);
|
||
scrollUp();
|
||
return oldFocus;
|
||
}
|
||
searchIndex--;
|
||
continue;
|
||
}
|
||
|
||
// Try to focus this item
|
||
Element* newFocus = item->requestFocus(oldFocus, FocusDirection::Up);
|
||
if (newFocus && newFocus != oldFocus) {
|
||
// ONLY reset when we successfully focus something
|
||
isTableScrolling.store(false, std::memory_order_release);
|
||
updateScrollOffset();
|
||
return newFocus;
|
||
} else {
|
||
// Non-focusable item (gap/header)
|
||
const float itemTop = calculateItemPosition(static_cast<size_t>(searchIndex));
|
||
if (itemTop < offset) {
|
||
isTableScrolling.store(true, std::memory_order_release); // Treat gaps/headers like tables
|
||
scrollUp();
|
||
return oldFocus;
|
||
}
|
||
searchIndex--;
|
||
}
|
||
}
|
||
|
||
return oldFocus;
|
||
}
|
||
|
||
// Helper method to calculate an item's position in the list
|
||
inline float calculateItemPosition(size_t index) {
|
||
float position = 0.0f;
|
||
for (size_t i = 0; i < index && i < m_items.size(); ++i) {
|
||
position += m_items[i]->getHeight();
|
||
}
|
||
return position;
|
||
}
|
||
|
||
// Enhanced scroll methods that ensure we always reach boundaries
|
||
//inline bool canScrollDown() {
|
||
// if (m_listHeight <= getHeight()) return false;
|
||
// float maxOffset = static_cast<float>(m_listHeight - getHeight());
|
||
// return (m_nextOffset < maxOffset - 0.1f) && (m_offset < maxOffset - 0.1f);
|
||
//}
|
||
//
|
||
//inline bool canScrollUp() {
|
||
// return (m_nextOffset > 0.1f) || (m_offset > 0.1f);
|
||
//}
|
||
|
||
|
||
//u64 m_lastScrollNavigationTime = 0;
|
||
//bool m_isHoldingOnTable = false;
|
||
|
||
// Enhanced scroll methods that snap to exact boundaries
|
||
inline void scrollDown() {
|
||
const u64 currentTime = armTicksToNs(armGetSystemTick());
|
||
|
||
// Calculate frame time
|
||
float frameTimeMs = 0.0f;
|
||
if (m_lastScrollTime != 0) {
|
||
frameTimeMs = static_cast<float>(currentTime - m_lastScrollTime) / 1000000.0f;
|
||
}
|
||
m_lastScrollTime = currentTime;
|
||
|
||
// Use original frame-based amounts
|
||
float scrollAmount = m_isHolding ? TABLE_SCROLL_STEP_SIZE : TABLE_SCROLL_STEP_SIZE_CLICK;
|
||
|
||
// If frame took longer than ~33ms (slower than 30fps), scale up the scroll amount
|
||
if (frameTimeMs > 33.0f) {
|
||
const float scaleFactor = frameTimeMs / 16.67f; // 16.67ms = 60fps baseline
|
||
scrollAmount *= std::min(scaleFactor, 3.0f); // Cap at 3x for very slow frames
|
||
}
|
||
|
||
m_nextOffset = std::min(m_nextOffset + scrollAmount,
|
||
static_cast<float>(m_listHeight - getHeight()));
|
||
}
|
||
|
||
inline void scrollUp() {
|
||
const u64 currentTime = armTicksToNs(armGetSystemTick());
|
||
|
||
// Calculate frame time
|
||
float frameTimeMs = 0.0f;
|
||
if (m_lastScrollTime != 0) {
|
||
frameTimeMs = static_cast<float>(currentTime - m_lastScrollTime) / 1000000.0f;
|
||
}
|
||
m_lastScrollTime = currentTime;
|
||
|
||
// Use original frame-based amounts
|
||
float scrollAmount = m_isHolding ? TABLE_SCROLL_STEP_SIZE : TABLE_SCROLL_STEP_SIZE_CLICK;
|
||
|
||
// If frame took longer than ~33ms (slower than 30fps), scale up the scroll amount
|
||
if (frameTimeMs > 33.0f) {
|
||
const float scaleFactor = frameTimeMs / 16.67f; // 16.67ms = 60fps baseline
|
||
scrollAmount *= std::min(scaleFactor, 3.0f); // Cap at 3x for very slow frames
|
||
}
|
||
|
||
m_nextOffset = std::max(m_nextOffset - scrollAmount, 0.0f);
|
||
}
|
||
|
||
// Add these methods to handle jumps with smooth scrolling
|
||
Element* handleJumpToBottom(Element* oldFocus) {
|
||
if (m_items.empty()) return oldFocus;
|
||
|
||
invalidate();
|
||
resetNavigationState();
|
||
jumpToBottom.store(false, std::memory_order_release); // Reset flag
|
||
|
||
// Calculate target offset once (good optimization to keep)
|
||
const float targetOffset = (m_listHeight > getHeight()) ?
|
||
static_cast<float>(m_listHeight - getHeight()) : 0.0f;
|
||
static constexpr float tolerance = 5.0f;
|
||
|
||
// Find the last focusable item
|
||
size_t lastFocusableIndex = m_items.size();
|
||
for (ssize_t i = static_cast<ssize_t>(m_items.size()) - 1; i >= 0; --i) {
|
||
Element* test = m_items[i]->requestFocus(nullptr, FocusDirection::None);
|
||
if (test) {
|
||
lastFocusableIndex = static_cast<size_t>(i);
|
||
break;
|
||
}
|
||
}
|
||
|
||
// Check if we're already at the bottom with proper tolerance
|
||
bool alreadyAtBottom = false;
|
||
if (lastFocusableIndex < m_items.size()) {
|
||
alreadyAtBottom = (m_focusedIndex == lastFocusableIndex) &&
|
||
(std::abs(m_nextOffset - targetOffset) <= tolerance);
|
||
}
|
||
|
||
if (alreadyAtBottom) {
|
||
return oldFocus; // Already at bottom, do nothing
|
||
}
|
||
|
||
// Not at bottom - perform the jump using SAME logic as wrapToBottom
|
||
if (lastFocusableIndex < m_items.size()) {
|
||
m_focusedIndex = lastFocusableIndex;
|
||
m_nextOffset = targetOffset; // Use same calculation as wrapToBottom
|
||
|
||
Element* newFocus = m_items[lastFocusableIndex]->requestFocus(oldFocus, FocusDirection::None);
|
||
if (newFocus && newFocus != oldFocus) {
|
||
return newFocus;
|
||
}
|
||
}
|
||
|
||
// No focusable items - just set target to bottom (same as wrapToBottom)
|
||
m_nextOffset = targetOffset;
|
||
return oldFocus;
|
||
}
|
||
|
||
Element* handleJumpToTop(Element* oldFocus) {
|
||
if (m_items.empty()) return oldFocus;
|
||
|
||
invalidate();
|
||
resetNavigationState();
|
||
jumpToTop.store(false, std::memory_order_release); // Reset flag
|
||
|
||
// Define constants for clarity and consistency
|
||
static constexpr float targetOffset = 0.0f;
|
||
static constexpr float tolerance = 0.0f;
|
||
|
||
// Find the first focusable item
|
||
size_t firstFocusableIndex = m_items.size(); // Default to invalid
|
||
for (size_t i = 0; i < m_items.size(); ++i) {
|
||
Element* test = m_items[i]->requestFocus(nullptr, FocusDirection::None);
|
||
if (test) {
|
||
firstFocusableIndex = i;
|
||
break;
|
||
}
|
||
}
|
||
|
||
// Check if we're already at the top with proper tolerance
|
||
bool alreadyAtTop = false;
|
||
if (firstFocusableIndex < m_items.size()) {
|
||
alreadyAtTop = (m_focusedIndex == firstFocusableIndex) &&
|
||
(std::abs(m_nextOffset - targetOffset) <= tolerance);
|
||
}
|
||
|
||
if (alreadyAtTop) {
|
||
return oldFocus; // Already at top, do nothing
|
||
}
|
||
|
||
// Not at top - perform the jump using SAME logic as wrapToTop
|
||
if (firstFocusableIndex < m_items.size()) {
|
||
m_focusedIndex = firstFocusableIndex;
|
||
m_nextOffset = targetOffset; // Same as wrapToTop
|
||
|
||
Element* newFocus = m_items[firstFocusableIndex]->requestFocus(oldFocus, FocusDirection::None);
|
||
if (newFocus && newFocus != oldFocus) {
|
||
return newFocus;
|
||
}
|
||
}
|
||
|
||
// No focusable items - just set target to top (same as wrapToTop)
|
||
m_nextOffset = targetOffset;
|
||
return oldFocus;
|
||
}
|
||
|
||
Element* handleSkipDown(Element* oldFocus) {
|
||
if (m_items.empty()) return oldFocus;
|
||
|
||
invalidate();
|
||
resetNavigationState();
|
||
|
||
// Calculate target offset once (good optimization to keep)
|
||
const float targetOffset = (m_listHeight > getHeight()) ?
|
||
static_cast<float>(m_listHeight - getHeight()) : 0.0f;
|
||
static constexpr float tolerance = 0.0f;
|
||
|
||
// Find the last focusable item
|
||
size_t lastFocusableIndex = m_items.size();
|
||
for (ssize_t i = static_cast<ssize_t>(m_items.size()) - 1; i >= 0; --i) {
|
||
Element* test = m_items[i]->requestFocus(nullptr, FocusDirection::None);
|
||
if (test) {
|
||
lastFocusableIndex = static_cast<size_t>(i);
|
||
break;
|
||
}
|
||
}
|
||
|
||
// Check if we're already at the bottom with proper tolerance
|
||
bool alreadyAtBottom = false;
|
||
if (lastFocusableIndex < m_items.size()) {
|
||
alreadyAtBottom = (m_focusedIndex == lastFocusableIndex) &&
|
||
(std::abs(m_nextOffset - targetOffset) <= tolerance);
|
||
}
|
||
|
||
if (alreadyAtBottom) {
|
||
return oldFocus; // Already at bottom, do nothing
|
||
}
|
||
|
||
|
||
const float viewHeight = static_cast<float>(getHeight());
|
||
const float maxOffset = (m_listHeight > getHeight()) ? static_cast<float>(m_listHeight - getHeight()) : 0.0f;
|
||
|
||
// Calculate the target viewport center after skipping
|
||
const float targetViewportTop = std::min(m_offset + viewHeight, maxOffset);
|
||
|
||
// Check if we traveled less than a full viewport
|
||
const float actualTravelDistance = targetViewportTop - m_offset;
|
||
const bool traveledFullViewport = (actualTravelDistance >= viewHeight - tolerance);
|
||
const float targetViewportCenter = targetViewportTop + (viewHeight / 2.0f + VIEW_CENTER_OFFSET);
|
||
|
||
// Find the item closest to the center of the new viewport
|
||
float itemTop = 0.0f;
|
||
size_t targetIndex = 0;
|
||
bool foundFocusable = false;
|
||
float bestDistance = std::numeric_limits<float>::max();
|
||
|
||
for (size_t i = 0; i < m_items.size(); ++i) {
|
||
const float itemHeight = m_items[i]->getHeight();
|
||
const float itemCenter = itemTop + (itemHeight / 2.0f);
|
||
const float distanceFromCenter = std::abs(itemCenter - targetViewportCenter);
|
||
|
||
// Check if this item is focusable and closer to center
|
||
Element* test = m_items[i]->requestFocus(nullptr, FocusDirection::None);
|
||
if (test && test->m_isItem && distanceFromCenter < bestDistance) {
|
||
targetIndex = i;
|
||
bestDistance = distanceFromCenter;
|
||
foundFocusable = true;
|
||
}
|
||
|
||
itemTop += itemHeight;
|
||
}
|
||
|
||
if (foundFocusable) {
|
||
bool nearBottom = true;
|
||
if (targetIndex > m_focusedIndex && traveledFullViewport) {
|
||
m_focusedIndex = targetIndex;
|
||
nearBottom = false;
|
||
}
|
||
isTableScrolling.store(false, std::memory_order_release);
|
||
updateScrollOffset(); // This will center the cursor properly
|
||
|
||
Element* newFocus = m_items[targetIndex]->requestFocus(oldFocus, FocusDirection::None);
|
||
return (newFocus && newFocus != oldFocus && !nearBottom && traveledFullViewport) ? newFocus : handleJumpToBottom(oldFocus);
|
||
} else {
|
||
isTableScrolling.store(true, std::memory_order_release);
|
||
m_nextOffset = targetViewportTop;
|
||
|
||
// NEW: Find the last focusable item that's still visible
|
||
float searchItemTop = 0.0f;
|
||
size_t lastVisibleFocusable = m_focusedIndex; // Default to current
|
||
|
||
for (size_t i = 0; i < m_items.size(); ++i) {
|
||
const float itemHeight = m_items[i]->getHeight();
|
||
const float itemBottom = searchItemTop + itemHeight;
|
||
|
||
// Stop if we've gone past the new viewport
|
||
if (searchItemTop >= targetViewportTop + viewHeight) break;
|
||
|
||
// Check if this item is focusable and visible in the new viewport
|
||
if (itemBottom > targetViewportTop) {
|
||
Element* test = m_items[i]->requestFocus(nullptr, FocusDirection::None);
|
||
if (test && test->m_isItem) {
|
||
lastVisibleFocusable = i;
|
||
}
|
||
}
|
||
|
||
searchItemTop += itemHeight;
|
||
}
|
||
|
||
// Focus on the last visible focusable item
|
||
if (lastVisibleFocusable != m_focusedIndex) {
|
||
m_focusedIndex = lastVisibleFocusable;
|
||
Element* newFocus = m_items[m_focusedIndex]->requestFocus(oldFocus, FocusDirection::None);
|
||
if (newFocus && newFocus != oldFocus) {
|
||
return newFocus;
|
||
}
|
||
}
|
||
}
|
||
|
||
return oldFocus;
|
||
}
|
||
|
||
Element* handleSkipUp(Element* oldFocus) {
|
||
if (m_items.empty()) return oldFocus;
|
||
|
||
invalidate();
|
||
resetNavigationState();
|
||
|
||
// Define constants for clarity and consistency
|
||
static constexpr float targetOffset = 0.0f;
|
||
static constexpr float tolerance = 0.0f;
|
||
|
||
// Find the first focusable item
|
||
size_t firstFocusableIndex = m_items.size(); // Default to invalid
|
||
for (size_t i = 0; i < m_items.size(); ++i) {
|
||
Element* test = m_items[i]->requestFocus(nullptr, FocusDirection::None);
|
||
if (test) {
|
||
firstFocusableIndex = i;
|
||
break;
|
||
}
|
||
}
|
||
|
||
// Check if we're already at the top with proper tolerance
|
||
bool alreadyAtTop = false;
|
||
if (firstFocusableIndex < m_items.size()) {
|
||
alreadyAtTop = (m_focusedIndex == firstFocusableIndex) &&
|
||
(std::abs(m_nextOffset - targetOffset) <= tolerance);
|
||
}
|
||
|
||
if (alreadyAtTop) {
|
||
return oldFocus; // Already at top, do nothing
|
||
}
|
||
|
||
|
||
const float viewHeight = static_cast<float>(getHeight());
|
||
|
||
// Calculate the target viewport center after skipping
|
||
const float targetViewportTop = std::max(0.0f, m_offset - viewHeight);
|
||
|
||
// Check if we traveled less than a full viewport
|
||
const float actualTravelDistance = m_offset - targetViewportTop;
|
||
const bool traveledFullViewport = (actualTravelDistance >= viewHeight - tolerance);
|
||
|
||
const float targetViewportCenter = targetViewportTop + (viewHeight / 2.0f + VIEW_CENTER_OFFSET);
|
||
|
||
// Find the item closest to the center of the new viewport
|
||
float itemTop = 0.0f;
|
||
size_t targetIndex = 0;
|
||
bool foundFocusable = false;
|
||
float bestDistance = std::numeric_limits<float>::max();
|
||
|
||
for (size_t i = 0; i < m_items.size(); ++i) {
|
||
const float itemHeight = m_items[i]->getHeight();
|
||
const float itemCenter = itemTop + (itemHeight / 2.0f);
|
||
const float distanceFromCenter = std::abs(itemCenter - targetViewportCenter);
|
||
|
||
// Check if this item is focusable and closer to center
|
||
Element* test = m_items[i]->requestFocus(nullptr, FocusDirection::None);
|
||
if (test && test->m_isItem && distanceFromCenter < bestDistance) {
|
||
targetIndex = i;
|
||
bestDistance = distanceFromCenter;
|
||
foundFocusable = true;
|
||
}
|
||
|
||
itemTop += itemHeight;
|
||
}
|
||
|
||
if (foundFocusable) {
|
||
|
||
bool nearTop = true;
|
||
if (targetIndex < m_focusedIndex && traveledFullViewport) {
|
||
m_focusedIndex = targetIndex;
|
||
nearTop = false;
|
||
}
|
||
//if (traveledFullViewport)
|
||
isTableScrolling.store(false, std::memory_order_release);
|
||
updateScrollOffset(); // This will center the cursor properly
|
||
|
||
Element* newFocus = m_items[targetIndex]->requestFocus(oldFocus, FocusDirection::None);
|
||
return (newFocus && newFocus != oldFocus && !nearTop && traveledFullViewport) ? newFocus : handleJumpToTop(oldFocus);
|
||
} else {
|
||
isTableScrolling.store(true, std::memory_order_release);
|
||
m_nextOffset = targetViewportTop;
|
||
|
||
// NEW: Find the first focusable item that's still visible
|
||
float searchItemTop = 0.0f;
|
||
size_t firstVisibleFocusable = m_focusedIndex; // Default to current
|
||
|
||
for (size_t i = 0; i < m_items.size(); ++i) {
|
||
const float itemHeight = m_items[i]->getHeight();
|
||
const float itemBottom = searchItemTop + itemHeight;
|
||
|
||
// Check if this item is visible in the new viewport
|
||
if (itemBottom > targetViewportTop && searchItemTop < targetViewportTop + viewHeight) {
|
||
Element* test = m_items[i]->requestFocus(nullptr, FocusDirection::None);
|
||
if (test && test->m_isItem) {
|
||
firstVisibleFocusable = i;
|
||
break; // Take the first one for skip up
|
||
}
|
||
}
|
||
|
||
searchItemTop += itemHeight;
|
||
}
|
||
|
||
// Focus on the first visible focusable item
|
||
if (firstVisibleFocusable != m_focusedIndex) {
|
||
m_focusedIndex = firstVisibleFocusable;
|
||
Element* newFocus = m_items[m_focusedIndex]->requestFocus(oldFocus, FocusDirection::None);
|
||
if (newFocus && newFocus != oldFocus) {
|
||
return newFocus;
|
||
}
|
||
}
|
||
}
|
||
|
||
return oldFocus;
|
||
}
|
||
|
||
|
||
inline void initializePrefixSums() {
|
||
prefixSums.clear();
|
||
prefixSums.resize(m_items.size() + 1, 0.0f);
|
||
|
||
for (size_t i = 1; i < prefixSums.size(); ++i) {
|
||
prefixSums[i] = prefixSums[i - 1] + m_items[i - 1]->getHeight();
|
||
}
|
||
}
|
||
|
||
|
||
// Keep your EXACT original updateScrollOffset() method unchanged:
|
||
virtual void updateScrollOffset() {
|
||
if (Element::getInputMode() != InputMode::Controller) return;
|
||
|
||
if (m_listHeight <= getHeight()) {
|
||
m_nextOffset = m_offset = 0;
|
||
return;
|
||
}
|
||
|
||
// Calculate position of focused item
|
||
float itemPos = 0.0f;
|
||
for (size_t i = 0; i < m_focusedIndex && i < m_items.size(); ++i) {
|
||
itemPos += m_items[i]->getHeight();
|
||
}
|
||
|
||
// Get the focused item's height
|
||
const float itemHeight = (m_focusedIndex < m_items.size()) ? m_items[m_focusedIndex]->getHeight() : 0.0f;
|
||
|
||
// Calculate viewport height
|
||
const float viewHeight = static_cast<float>(getHeight());
|
||
|
||
// FIXED: Special handling for the first focusable item
|
||
//if (m_focusedIndex == 0 || itemPos <= viewHeight * 0.3f) {
|
||
// // For items at the very top or very close to top, snap to absolute zero
|
||
// m_nextOffset = 0.0f;
|
||
// return;
|
||
//}
|
||
|
||
// FIXED: Special handling for items near the bottom
|
||
const float maxOffset = static_cast<float>(m_listHeight - getHeight());
|
||
//const float itemBottom = itemPos + itemHeight;
|
||
//if (itemBottom >= m_listHeight - (viewHeight * 0.3f)) {
|
||
// // For items near the bottom, snap to max offset
|
||
// m_nextOffset = maxOffset;
|
||
// return;
|
||
//}
|
||
|
||
// For middle items, use centering logic
|
||
const float itemCenterPos = itemPos + (itemHeight / 2.0f);
|
||
const float viewportCenter = viewHeight / 2.0f + VIEW_CENTER_OFFSET + 0.5f; // add slight offset
|
||
//float idealOffset = itemCenterPos - viewportCenter;
|
||
|
||
// Clamp to valid scroll bounds
|
||
const float idealOffset = std::max(0.0f, std::min(itemCenterPos - viewportCenter, maxOffset));
|
||
|
||
// Set target for smooth animation
|
||
m_nextOffset = idealOffset;
|
||
|
||
//m_nextOffset = std::max(0.0f, std::min(itemPos + itemHeight * 0.5f - (viewHeight * 0.5f + 7.0f), maxOffset));
|
||
}
|
||
|
||
};
|
||
|
||
|
||
|
||
|
||
/**
|
||
* @brief A item that goes into a list
|
||
*
|
||
*/
|
||
class ListItem : public Element {
|
||
public:
|
||
u32 width, height;
|
||
u64 m_touchStartTime_ns;
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
ListItem(const std::string& text, const std::string& value = "", bool isMini = false, bool useScriptKey = true)
|
||
: Element(), m_text(text), m_value(value), m_listItemHeight(isMini ? tsl::style::MiniListItemDefaultHeight : tsl::style::ListItemDefaultHeight), m_useScriptKey(useScriptKey) {
|
||
m_isItem = true;
|
||
m_text_clean = m_text;
|
||
ult::removeTag(m_text_clean);
|
||
applyInitialTranslations();
|
||
if (!value.empty()) applyInitialTranslations(true);
|
||
}
|
||
#else
|
||
ListItem(const std::string& text, const std::string& value = "", bool isMini = false)
|
||
: Element(), m_text(text), m_value(value), m_listItemHeight(isMini ? tsl::style::MiniListItemDefaultHeight : tsl::style::ListItemDefaultHeight) {
|
||
m_isItem = true;
|
||
m_text_clean = m_text;
|
||
ult::removeTag(m_text_clean);
|
||
applyInitialTranslations();
|
||
if (!value.empty()) applyInitialTranslations(true);
|
||
}
|
||
#endif
|
||
|
||
virtual ~ListItem() = default;
|
||
|
||
virtual void draw(gfx::Renderer *renderer) override {
|
||
const bool useClickTextColor = m_touched && Element::getInputMode() == InputMode::Touch && ult::touchInBounds;
|
||
|
||
if (useClickTextColor) [[unlikely]] {
|
||
if (ult::expandedMemory)
|
||
renderer->drawRectMultiThreaded(this->getX() + 4, this->getY(), this->getWidth() - 8, this->getHeight(), aWithOpacity(clickColor));
|
||
else
|
||
renderer->drawRect(this->getX() + 4, this->getY(), this->getWidth() - 8, this->getHeight(), aWithOpacity(clickColor));
|
||
}
|
||
|
||
const s32 yOffset = ((tsl::style::ListItemDefaultHeight - m_listItemHeight) /2.0) +0.5f; // Bit shift for division by 2
|
||
|
||
if (!m_maxWidth) [[unlikely]] {
|
||
calculateWidths(renderer);
|
||
}
|
||
|
||
// Optimized separator drawing
|
||
const float topBound = this->getTopBound();
|
||
const float bottomBound = this->getBottomBound();
|
||
static float lastBottomBound = 0.0f;
|
||
|
||
if (lastBottomBound != topBound) [[unlikely]] {
|
||
renderer->drawRect(this->getX() + 4, topBound, this->getWidth() + 10, 1, a(separatorColor));
|
||
}
|
||
renderer->drawRect(this->getX() + 4, bottomBound, this->getWidth() + 10, 1, a(separatorColor));
|
||
lastBottomBound = bottomBound;
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
static const std::vector<std::string> specialChars = {ult::STAR_SYMBOL};
|
||
#else
|
||
static const std::vector<std::string> specialChars = {};
|
||
#endif
|
||
// Fast path for non-truncated text
|
||
if (!m_truncated) [[likely]] {
|
||
const Color textColor = m_focused
|
||
? (!ult::useSelectionText
|
||
? (m_hasCustomTextColor ? m_customTextColor : defaultTextColor)
|
||
: (useClickTextColor
|
||
? clickTextColor
|
||
: selectedTextColor))
|
||
: (m_hasCustomTextColor
|
||
? m_customTextColor
|
||
: (useClickTextColor
|
||
? clickTextColor
|
||
: defaultTextColor));
|
||
|
||
renderer->drawStringWithColoredSections(m_text_clean, false, specialChars, this->getX() + 19, this->getY() + 45 - yOffset, 23,
|
||
textColor, (m_focused ? starColor : selectionStarColor));
|
||
} else {
|
||
drawTruncatedText(renderer, yOffset, useClickTextColor, specialChars);
|
||
}
|
||
|
||
if (!m_value.empty()) [[likely]] {
|
||
drawValue(renderer, yOffset, useClickTextColor);
|
||
}
|
||
}
|
||
|
||
virtual void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {
|
||
this->setBoundaries(this->getX() + 3, this->getY(), this->getWidth() + 9, m_listItemHeight);
|
||
}
|
||
|
||
virtual bool onClick(u64 keys) override {
|
||
if (keys & KEY_A) [[likely]] {
|
||
if (m_useClickAnimation)
|
||
triggerClickAnimation();
|
||
} else if (keys & (KEY_UP | KEY_DOWN | KEY_LEFT | KEY_RIGHT)) [[unlikely]] {
|
||
m_clickAnimationProgress = 0;
|
||
}
|
||
return Element::onClick(keys);
|
||
}
|
||
|
||
virtual bool onTouch(TouchEvent event, s32 currX, s32 currY, s32 prevX, s32 prevY, s32 initialX, s32 initialY) override {
|
||
if (event == TouchEvent::Touch) [[likely]] {
|
||
if ((m_touched = inBounds(currX, currY))) [[likely]] {
|
||
m_touchStartTime_ns = armTicksToNs(armGetSystemTick());
|
||
}
|
||
return false;
|
||
}
|
||
|
||
if (event == TouchEvent::Release && m_touched) [[likely]] {
|
||
m_touched = false;
|
||
if (Element::getInputMode() == InputMode::Touch) [[likely]] {
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
const s64 keyToUse = determineKeyOnTouchRelease(m_useScriptKey);
|
||
const bool handled = onClick(keyToUse);
|
||
#else
|
||
const s64 keyToUse = determineKeyOnTouchRelease(false);
|
||
const bool handled = onClick(keyToUse);
|
||
#endif
|
||
m_clickAnimationProgress = 0;
|
||
return handled;
|
||
}
|
||
}
|
||
return false;
|
||
}
|
||
|
||
virtual void setFocused(bool state) override {
|
||
if (state != m_focused) [[likely]] {
|
||
m_scroll = false;
|
||
m_scrollOffset = 0;
|
||
timeIn_ns = armTicksToNs(armGetSystemTick());
|
||
Element::setFocused(state);
|
||
}
|
||
}
|
||
|
||
virtual inline Element* requestFocus(Element *oldFocus, FocusDirection direction) override {
|
||
return this;
|
||
}
|
||
|
||
inline void setText(const std::string& text) {
|
||
if (m_text != text) [[likely]] {
|
||
m_text = text;
|
||
m_text_clean = m_text;
|
||
ult::removeTag(m_text_clean);
|
||
resetTextProperties();
|
||
applyInitialTranslations();
|
||
}
|
||
}
|
||
|
||
inline void setValue(const std::string& value, bool faint = false) {
|
||
if (m_value != value || m_faint != faint) [[likely]] {
|
||
m_value = value;
|
||
m_faint = faint;
|
||
//m_useVersionColor = useVersionColor;
|
||
m_maxWidth = 0;
|
||
if (!value.empty()) applyInitialTranslations(true);
|
||
}
|
||
}
|
||
|
||
inline void setTextColor(Color color) {
|
||
m_customTextColor = color;
|
||
m_hasCustomTextColor = true;
|
||
}
|
||
|
||
inline void setValueColor(Color color) {
|
||
m_customValueColor = color;
|
||
m_hasCustomValueColor = true;
|
||
}
|
||
|
||
inline void clearTextColor() {
|
||
m_hasCustomTextColor = false;
|
||
}
|
||
|
||
inline void clearValueColor() {
|
||
m_hasCustomValueColor = false;
|
||
}
|
||
|
||
inline void disableClickAnimation() {
|
||
m_useClickAnimation = false;
|
||
}
|
||
|
||
inline void enableClickAnimation() {
|
||
m_useClickAnimation = true;
|
||
}
|
||
|
||
|
||
inline const std::string& getText() const noexcept {
|
||
return m_text;
|
||
}
|
||
|
||
inline const std::string& getValue() const noexcept {
|
||
return m_value;
|
||
}
|
||
|
||
//virtual bool matchesJumpCriteria(const std::string& jumpText, const std::string& jumpValue) const override {
|
||
// return matchesJumpCriteria(jumpText, jumpValue, true); // Default to exact match
|
||
//}
|
||
|
||
virtual bool matchesJumpCriteria(const std::string& jumpText, const std::string& jumpValue, bool exactMatch=true) const {
|
||
if (jumpText.empty() && jumpValue.empty()) return false;
|
||
|
||
bool textMatches, valueMatches;
|
||
if (exactMatch) {
|
||
textMatches = (m_text == jumpText);
|
||
valueMatches = (m_value == jumpValue);
|
||
} else { // contains check
|
||
textMatches = (m_text.find(jumpText) != std::string::npos);
|
||
valueMatches = (m_value.find(jumpValue) != std::string::npos);
|
||
}
|
||
|
||
if (jumpText.empty() && !jumpValue.empty())
|
||
return valueMatches;
|
||
else if (!jumpText.empty() && jumpValue.empty())
|
||
return textMatches;
|
||
|
||
return (textMatches && valueMatches);
|
||
}
|
||
|
||
protected:
|
||
u64 timeIn_ns;
|
||
std::string m_text;
|
||
std::string m_text_clean;
|
||
std::string m_value;
|
||
std::string m_scrollText;
|
||
std::string m_ellipsisText;
|
||
u32 m_listItemHeight;
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
bool m_useScriptKey = false;
|
||
#endif
|
||
|
||
bool m_scroll = false;
|
||
bool m_truncated = false;
|
||
bool m_faint = false;
|
||
//bool m_useVersionColor = false;
|
||
bool m_touched = false;
|
||
|
||
bool m_hasCustomTextColor = false;
|
||
bool m_hasCustomValueColor = false;
|
||
Color m_customTextColor = {0};
|
||
Color m_customValueColor = {0};
|
||
bool m_useClickAnimation = true;
|
||
|
||
float m_scrollOffset = 0.0f;
|
||
u32 m_maxWidth = 0;
|
||
u32 m_textWidth = 0;
|
||
|
||
private:
|
||
void applyInitialTranslations(bool isValue = false) {
|
||
std::string& target = isValue ? m_value : m_text_clean;
|
||
ult::applyLangReplacements(target, isValue);
|
||
ult::convertComboToUnicode(target);
|
||
|
||
// Thread-safe translation cache access
|
||
#ifdef UI_OVERRIDE_PATH
|
||
{
|
||
// Use the processed target as the key for translation lookup
|
||
const std::string originalKey = target;
|
||
|
||
std::shared_lock<std::shared_mutex> readLock(tsl::gfx::s_translationCacheMutex);
|
||
auto translatedIt = ult::translationCache.find(originalKey);
|
||
if (translatedIt != ult::translationCache.end()) {
|
||
target = translatedIt->second; // Apply translation to target, not text
|
||
} else {
|
||
// Need to upgrade to write lock
|
||
readLock.unlock();
|
||
std::unique_lock<std::shared_mutex> writeLock(tsl::gfx::s_translationCacheMutex);
|
||
|
||
// Double-check pattern
|
||
translatedIt = ult::translationCache.find(originalKey);
|
||
if (translatedIt != ult::translationCache.end()) {
|
||
target = translatedIt->second;
|
||
} else {
|
||
// Store the original as both key and value if no translation exists
|
||
ult::translationCache[originalKey] = originalKey;
|
||
// target already contains the correct value, no need to modify
|
||
}
|
||
}
|
||
}
|
||
#endif
|
||
}
|
||
|
||
void calculateWidths(gfx::Renderer* renderer) {
|
||
if (m_value.empty()) {
|
||
m_maxWidth = getWidth() - 62;
|
||
} else {
|
||
//auto valueWidth = renderer->getTextDimensions(m_value, false, 20).first;
|
||
m_maxWidth = getWidth() - renderer->getTextDimensions(m_value, false, 20).first - 66;
|
||
}
|
||
|
||
//auto textWidth = renderer->getTextDimensions(m_text, false, 23).first;
|
||
const u32 width = renderer->getTextDimensions(m_text_clean, false, 23).first;
|
||
m_truncated = width > m_maxWidth + 20;
|
||
|
||
if (m_truncated) [[unlikely]] {
|
||
// Optimized string building for scroll text
|
||
m_scrollText.clear();
|
||
m_scrollText.reserve(m_text_clean.size() * 2 + 8); // Pre-allocate for text + spaces + text
|
||
|
||
// Build scroll text efficiently: "text text"
|
||
m_scrollText.append(m_text_clean).append(" ");
|
||
//auto scrollWidth = renderer->getTextDimensions(m_scrollText, false, 23).first;
|
||
m_textWidth = renderer->getTextDimensions(m_scrollText, false, 23).first;
|
||
m_scrollText.append(m_text_clean);
|
||
|
||
m_ellipsisText = renderer->limitStringLength(m_text_clean, false, 23, m_maxWidth);
|
||
} else {
|
||
m_textWidth = width;
|
||
}
|
||
}
|
||
|
||
void drawTruncatedText(gfx::Renderer* renderer, s32 yOffset, bool useClickTextColor, const std::vector<std::string>& specialSymbols = {}) {
|
||
if (m_focused) {
|
||
renderer->enableScissoring(getX() + 6, 97, m_maxWidth + (m_value.empty() ? 49 : 27), tsl::cfg::FramebufferHeight - 170);
|
||
|
||
//Color textColor = m_hasCustomTextColor ? m_customTextColor :
|
||
// (useClickTextColor ? clickTextColor : selectedTextColor);
|
||
|
||
renderer->drawStringWithColoredSections(m_scrollText, false, specialSymbols, getX() + 19 - static_cast<s32>(m_scrollOffset), getY() + 45 - yOffset, 23,
|
||
!ult::useSelectionText ? defaultTextColor: (useClickTextColor ? clickTextColor : selectedTextColor), (starColor));
|
||
renderer->disableScissoring();
|
||
handleScrolling();
|
||
} else {
|
||
//Color textColor = m_hasCustomTextColor ? m_customTextColor :
|
||
// (useClickTextColor ? clickTextColor : defaultTextColor);
|
||
|
||
renderer->drawStringWithColoredSections(m_ellipsisText, false, specialSymbols, getX() + 19, getY() + 45 - yOffset, 23,
|
||
m_hasCustomTextColor ? m_customTextColor : (useClickTextColor ? clickTextColor : defaultTextColor), (starColor));
|
||
}
|
||
}
|
||
|
||
|
||
void handleScrolling() {
|
||
const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
const u64 elapsed_ns = currentTime_ns - timeIn_ns;
|
||
|
||
// Frame rate compensation - cache calculations to reduce stutter
|
||
static u64 lastUpdateTime = 0;
|
||
static float cachedScrollOffset = 0.0f;
|
||
|
||
// Pre-compute constants as statics to avoid recalculation
|
||
static bool constantsInitialized = false;
|
||
static double totalCycleDuration;
|
||
static double delayDuration;
|
||
static double scrollDuration;
|
||
static double accelTime;
|
||
static double constantVelocityTime;
|
||
static double maxVelocity;
|
||
static double accelDistance;
|
||
static double constantVelocityDistance;
|
||
static double minScrollDistance;
|
||
static double invAccelTime; // 1/accelTime for multiplication instead of division
|
||
static double invDecelTime; // 1/decelTime for multiplication instead of division
|
||
static double invBillion; // 1/1000000000.0 for ns to seconds conversion
|
||
|
||
if (!constantsInitialized || minScrollDistance != static_cast<double>(m_textWidth)) {
|
||
// Constants for velocity-based scrolling
|
||
delayDuration = 2.0;
|
||
static constexpr double pauseDuration = 1.0;
|
||
maxVelocity = 166.0;
|
||
accelTime = 0.5;
|
||
static constexpr double decelTime = 0.5;
|
||
|
||
// Pre-calculate derived constants
|
||
minScrollDistance = static_cast<double>(m_textWidth);
|
||
accelDistance = 0.5 * maxVelocity * accelTime;
|
||
const double decelDistance = 0.5 * maxVelocity * decelTime;
|
||
constantVelocityDistance = std::max(0.0, minScrollDistance - accelDistance - decelDistance);
|
||
constantVelocityTime = constantVelocityDistance / maxVelocity;
|
||
scrollDuration = accelTime + constantVelocityTime + decelTime;
|
||
totalCycleDuration = delayDuration + scrollDuration + pauseDuration;
|
||
|
||
// Pre-calculate reciprocals for faster division
|
||
invAccelTime = 1.0 / accelTime;
|
||
invDecelTime = 1.0 / decelTime;
|
||
invBillion = 1.0 / 1000000000.0;
|
||
|
||
constantsInitialized = true;
|
||
}
|
||
|
||
// Fast ns to seconds conversion
|
||
const double elapsed_seconds = static_cast<double>(elapsed_ns) * invBillion;
|
||
|
||
// Update at consistent intervals regardless of frame rate
|
||
if (currentTime_ns - lastUpdateTime >= 8333333ULL) { // ~120 FPS update rate
|
||
// Use std::fmod for modulo - it's optimized and faster than loops
|
||
const double cyclePosition = std::fmod(elapsed_seconds, totalCycleDuration);
|
||
|
||
if (cyclePosition < delayDuration) [[likely]] {
|
||
// Delay phase - no scrolling
|
||
cachedScrollOffset = 0.0f;
|
||
} else if (cyclePosition < delayDuration + scrollDuration) [[likely]] {
|
||
// Scrolling phase - velocity-based movement
|
||
const double scrollTime = cyclePosition - delayDuration;
|
||
double distance;
|
||
|
||
if (scrollTime <= accelTime) {
|
||
// Acceleration phase - quadratic ease-in
|
||
const double t = scrollTime * invAccelTime; // Multiply instead of divide
|
||
const double smoothT = t * t;
|
||
distance = smoothT * accelDistance;
|
||
} else if (scrollTime <= accelTime + constantVelocityTime) {
|
||
// Constant velocity phase
|
||
const double constantTime = scrollTime - accelTime;
|
||
distance = accelDistance + (constantTime * maxVelocity);
|
||
} else {
|
||
// Deceleration phase - quadratic ease-out
|
||
const double decelStartTime = accelTime + constantVelocityTime;
|
||
const double t = (scrollTime - decelStartTime) * invDecelTime; // Multiply instead of divide
|
||
const double oneMinusT = 1.0 - t;
|
||
const double smoothT = 1.0 - oneMinusT * oneMinusT; // Avoid repeated calculation
|
||
distance = accelDistance + constantVelocityDistance + (smoothT * (minScrollDistance - accelDistance - constantVelocityDistance));
|
||
}
|
||
|
||
// Use branchless min with conditional move behavior
|
||
cachedScrollOffset = static_cast<float>(distance < minScrollDistance ? distance : minScrollDistance);
|
||
} else [[unlikely]] {
|
||
// Pause phase - stay at end
|
||
cachedScrollOffset = static_cast<float>(m_textWidth);
|
||
}
|
||
|
||
lastUpdateTime = currentTime_ns;
|
||
}
|
||
|
||
// Use cached value for consistent display
|
||
m_scrollOffset = cachedScrollOffset;
|
||
|
||
// Reset timer when cycle completes
|
||
if (elapsed_seconds >= totalCycleDuration) [[unlikely]] {
|
||
timeIn_ns = currentTime_ns;
|
||
}
|
||
}
|
||
|
||
void drawValue(gfx::Renderer* renderer, s32 yOffset, bool useClickTextColor) {
|
||
const s32 xPosition = getX() + m_maxWidth + 47;
|
||
const s32 yPosition = getY() + 45 - yOffset-1;
|
||
static constexpr s32 fontSize = 20;
|
||
|
||
static bool lastRunningInterpreter = false;
|
||
const auto textColor = determineValueTextColor(useClickTextColor, lastRunningInterpreter);
|
||
|
||
if (m_value != ult::INPROGRESS_SYMBOL) [[likely]] {
|
||
static const std::vector<std::string> specialChars = {ult::DIVIDER_SYMBOL};
|
||
renderer->drawStringWithColoredSections(m_value, false, specialChars, xPosition, yPosition, fontSize, textColor, textSeparatorColor);
|
||
} else {
|
||
drawThrobber(renderer, xPosition, yPosition, fontSize, textColor);
|
||
}
|
||
lastRunningInterpreter = ult::runningInterpreter.load(std::memory_order_acquire); // Relaxed ordering is sufficient
|
||
}
|
||
|
||
Color determineValueTextColor(bool useClickTextColor, bool lastRunningInterpreter) const {
|
||
// Check if selection value colors should be disabled
|
||
if (m_focused && ult::useSelectionValue) {
|
||
// Use selection colors when focused and selection colors are enabled
|
||
if (m_value == ult::DROPDOWN_SYMBOL || m_value == ult::OPTION_SYMBOL) {
|
||
return useClickTextColor ? (clickTextColor) :
|
||
(m_faint ? offTextColor : (useClickTextColor ? clickTextColor : (ult::useSelectionText ? selectedTextColor : defaultTextColor)));
|
||
}
|
||
|
||
const bool isRunning = ult::runningInterpreter.load(std::memory_order_acquire) || lastRunningInterpreter;
|
||
if (isRunning && (m_value.find(ult::DOWNLOAD_SYMBOL) != std::string::npos ||
|
||
m_value.find(ult::UNZIP_SYMBOL) != std::string::npos ||
|
||
m_value.find(ult::COPY_SYMBOL) != std::string::npos)) {
|
||
return m_faint ? offTextColor : (inprogressTextColor);
|
||
}
|
||
|
||
if (m_value == ult::INPROGRESS_SYMBOL) {
|
||
return m_faint ? offTextColor : (inprogressTextColor);
|
||
}
|
||
|
||
if (m_value == ult::CROSSMARK_SYMBOL) {
|
||
return m_faint ? offTextColor : (invalidTextColor);
|
||
}
|
||
|
||
// For normal values when focused, use selectedValueTextColor
|
||
return useClickTextColor ? clickTextColor : selectedValueTextColor;
|
||
}
|
||
|
||
// If focused but selection colors are disabled, or if not focused, use original logic
|
||
// Check for custom value color first
|
||
if (m_hasCustomValueColor) {
|
||
return m_customValueColor;
|
||
}
|
||
|
||
// Original logic for all other cases
|
||
if (m_value == ult::DROPDOWN_SYMBOL || m_value == ult::OPTION_SYMBOL) {
|
||
return (m_focused ? (useClickTextColor ? clickTextColor : (m_faint ? offTextColor : (ult::useSelectionText ? selectedTextColor : defaultTextColor))) :
|
||
(useClickTextColor ? clickTextColor : (m_faint ? offTextColor : defaultTextColor)));
|
||
}
|
||
|
||
const bool isRunning = ult::runningInterpreter.load(std::memory_order_acquire) || lastRunningInterpreter;
|
||
if (isRunning && (m_value.find(ult::DOWNLOAD_SYMBOL) != std::string::npos ||
|
||
m_value.find(ult::UNZIP_SYMBOL) != std::string::npos ||
|
||
m_value.find(ult::COPY_SYMBOL) != std::string::npos)) {
|
||
return m_faint ? offTextColor : (inprogressTextColor);
|
||
}
|
||
|
||
if (m_value == ult::INPROGRESS_SYMBOL) {
|
||
return m_faint ? offTextColor : (inprogressTextColor);
|
||
}
|
||
|
||
if (m_value == ult::CROSSMARK_SYMBOL) {
|
||
return m_faint ? offTextColor : (invalidTextColor);
|
||
}
|
||
|
||
return (m_faint ? offTextColor : (onTextColor));
|
||
}
|
||
|
||
void drawThrobber(gfx::Renderer* renderer, s32 xPosition, s32 yPosition, s32 fontSize, Color textColor) {
|
||
static size_t throbberCounter = 0;
|
||
const auto& throbberSymbol = ult::THROBBER_SYMBOLS[(throbberCounter / 10) % ult::THROBBER_SYMBOLS.size()];
|
||
throbberCounter = (throbberCounter + 1) % (10 * ult::THROBBER_SYMBOLS.size());
|
||
renderer->drawString(throbberSymbol, false, xPosition, yPosition, fontSize, textColor);
|
||
}
|
||
|
||
|
||
s64 determineKeyOnTouchRelease(bool useScriptKey) const {
|
||
const u64 touchDuration_ns = armTicksToNs(armGetSystemTick()) - m_touchStartTime_ns;
|
||
const float touchDurationInSeconds = static_cast<float>(touchDuration_ns) * 1e-9f; // More efficient than division
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
if (touchDurationInSeconds >= 0.7f) [[unlikely]] {
|
||
ult::longTouchAndRelease.store(true, std::memory_order_release);
|
||
return useScriptKey ? SCRIPT_KEY : STAR_KEY;
|
||
}
|
||
#endif
|
||
if (touchDurationInSeconds >= 0.3f) [[unlikely]] {
|
||
//if (useScriptKey)
|
||
ult::shortTouchAndRelease.store(true, std::memory_order_release);
|
||
return useScriptKey ? SCRIPT_KEY : SETTINGS_KEY;
|
||
}
|
||
return KEY_A;
|
||
}
|
||
|
||
|
||
void resetTextProperties() {
|
||
m_scrollText.clear();
|
||
m_ellipsisText.clear();
|
||
m_maxWidth = 0;
|
||
}
|
||
};
|
||
|
||
class MiniListItem : public ListItem {
|
||
public:
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
// Constructor for MiniListItem, with no `isMini` boolean.
|
||
MiniListItem(const std::string& text, const std::string& value = "", bool useScriptKey = false)
|
||
: ListItem(text, value, true, useScriptKey) { // Call the parent constructor with `isMini = true`
|
||
#else
|
||
MiniListItem(const std::string& text, const std::string& value = "")
|
||
: ListItem(text, value, true) { // Call the parent constructor with `isMini = true`
|
||
#endif
|
||
|
||
// Additional MiniListItem-specific initialization can go here, if necessary.
|
||
}
|
||
|
||
// Destructor if needed (inherits default behavior from ListItem)
|
||
virtual ~MiniListItem() {}
|
||
};
|
||
|
||
/**
|
||
* @brief A item that goes into a list (this version uses value and faint color sourcing)
|
||
*
|
||
*/
|
||
class ListItemV2 : public Element {
|
||
public:
|
||
u32 width, height;
|
||
u64 m_touchStartTime_ns; // Track the time when touch starts
|
||
|
||
/**
|
||
* @brief Constructor
|
||
*
|
||
* @param text Initial description text
|
||
*/
|
||
ListItemV2(const std::string& text, const std::string& value = "", Color valueColor = onTextColor, Color faintColor = offTextColor)
|
||
: Element(), m_text(text), m_value(value), m_valueColor{valueColor}, m_faintColor{faintColor} {
|
||
}
|
||
virtual ~ListItemV2() {}
|
||
|
||
|
||
virtual void draw(gfx::Renderer *renderer) override {
|
||
static float lastBottomBound;
|
||
bool useClickTextColor = false;
|
||
if (this->m_touched && Element::getInputMode() == InputMode::Touch) {
|
||
if (ult::touchInBounds) {
|
||
//renderer->drawRect(ELEMENT_BOUNDS(this), a(clickColor));
|
||
renderer->drawRect( this->getX()+4, this->getY(), this->getWidth()-8, this->getHeight(), a(clickColor));
|
||
useClickTextColor = true;
|
||
}
|
||
//renderer->drawRect(ELEMENT_BOUNDS(this), tsl::style::color::ColorClickAnimation);
|
||
}
|
||
|
||
// Calculate vertical offset to center the text
|
||
const s32 yOffset = (tsl::style::ListItemDefaultHeight - this->m_listItemHeight) / 2;
|
||
|
||
if (this->m_maxWidth == 0) {
|
||
if (this->m_value.length() > 0) {
|
||
//std::tie(width, height) = renderer->drawString(this->m_value, false, 0, 0, 20, a(tsl::style::color::ColorTransparent));
|
||
//auto valueWidth = renderer->getTextDimensions(this->m_value, false, 20).first;
|
||
width = renderer->getTextDimensions(this->m_value, false, 20).first;
|
||
this->m_maxWidth = this->getWidth() - width - 70 +4;
|
||
} else {
|
||
this->m_maxWidth = this->getWidth() - 40 -10 -12;
|
||
}
|
||
|
||
//std::tie(width, height) = renderer->drawString(this->m_text, false, 0, 0, 23, a(tsl::style::color::ColorTransparent));
|
||
//auto textWidth = renderer->getTextDimensions(this->m_text, false, 23).first;
|
||
width = renderer->getTextDimensions(this->m_text, false, 23).first;
|
||
this->m_trunctuated = width > this->m_maxWidth+20;
|
||
|
||
if (this->m_trunctuated) {
|
||
this->m_scrollText = this->m_text + " ";
|
||
//std::tie(width, height) = renderer->drawString(this->m_scrollText, false, 0, 0, 23, a(tsl::style::color::ColorTransparent));
|
||
//auto scrollWidth = renderer->getTextDimensions(this->m_scrollText, false, 23).first;
|
||
width = renderer->getTextDimensions(this->m_scrollText, false, 23).first;
|
||
this->m_scrollText += this->m_text;
|
||
this->m_textWidth = width;
|
||
|
||
this->m_ellipsisText = renderer->limitStringLength(this->m_text, false, 23, this->m_maxWidth);
|
||
} else {
|
||
this->m_textWidth = width;
|
||
}
|
||
}
|
||
|
||
if (lastBottomBound != this->getTopBound())
|
||
renderer->drawRect(this->getX()+4, this->getTopBound(), this->getWidth()+6 +4, 1, a(separatorColor));
|
||
renderer->drawRect(this->getX()+4, this->getBottomBound(), this->getWidth()+6 +4, 1, a(separatorColor));
|
||
|
||
lastBottomBound = this->getBottomBound();
|
||
|
||
|
||
if (this->m_trunctuated) {
|
||
if (this->m_focused) {
|
||
if (this->m_value.length() > 0)
|
||
renderer->enableScissoring(this->getX()+6, 97, this->m_maxWidth + 30 -3, tsl::cfg::FramebufferHeight-73-97);
|
||
else
|
||
renderer->enableScissoring(this->getX()+6, 97, this->m_maxWidth + 40 +9, tsl::cfg::FramebufferHeight-73-97);
|
||
renderer->drawString(this->m_scrollText, false, this->getX() + 20-1 - this->m_scrollOffset, this->getY() + 45 - yOffset, 23, a(selectedTextColor));
|
||
renderer->disableScissoring();
|
||
|
||
// Handle scrolling with frame rate compensation
|
||
const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
const u64 elapsed_ns = currentTime_ns - this->timeIn_ns;
|
||
|
||
// Frame rate compensation - cache calculations to reduce stutter
|
||
static u64 lastUpdateTime = 0;
|
||
static float cachedScrollOffset = 0.0f;
|
||
|
||
// Pre-compute constants as statics to avoid recalculation
|
||
static bool constantsInitialized = false;
|
||
static double totalCycleDuration;
|
||
static double delayDuration;
|
||
static double scrollDuration;
|
||
static double accelTime;
|
||
static double constantVelocityTime;
|
||
static double maxVelocity;
|
||
static double accelDistance;
|
||
static double constantVelocityDistance;
|
||
static double minScrollDistance;
|
||
static double invAccelTime; // 1/accelTime for multiplication instead of division
|
||
static double invDecelTime; // 1/decelTime for multiplication instead of division
|
||
static double invBillion; // 1/1000000000.0 for ns to seconds conversion
|
||
|
||
if (!constantsInitialized || minScrollDistance != static_cast<double>(this->m_textWidth)) {
|
||
// Constants for velocity-based scrolling
|
||
delayDuration = 2.0;
|
||
static constexpr double pauseDuration = 1.0;
|
||
maxVelocity = 166.0;
|
||
accelTime = 0.5;
|
||
static constexpr double decelTime = 0.5;
|
||
|
||
// Pre-calculate derived constants
|
||
minScrollDistance = static_cast<double>(this->m_textWidth);
|
||
accelDistance = 0.5 * maxVelocity * accelTime;
|
||
const double decelDistance = 0.5 * maxVelocity * decelTime;
|
||
constantVelocityDistance = std::max(0.0, minScrollDistance - accelDistance - decelDistance);
|
||
constantVelocityTime = constantVelocityDistance / maxVelocity;
|
||
scrollDuration = accelTime + constantVelocityTime + decelTime;
|
||
totalCycleDuration = delayDuration + scrollDuration + pauseDuration;
|
||
|
||
// Pre-calculate reciprocals for faster division
|
||
invAccelTime = 1.0 / accelTime;
|
||
invDecelTime = 1.0 / decelTime;
|
||
invBillion = 1.0 / 1000000000.0;
|
||
|
||
constantsInitialized = true;
|
||
}
|
||
|
||
// Fast ns to seconds conversion
|
||
const double elapsed_seconds = static_cast<double>(elapsed_ns) * invBillion;
|
||
|
||
// Update at consistent intervals regardless of frame rate
|
||
if (currentTime_ns - lastUpdateTime >= 8333333ULL) { // ~120 FPS update rate
|
||
// Use std::fmod for modulo - it's optimized and faster than loops
|
||
const double cyclePosition = std::fmod(elapsed_seconds, totalCycleDuration);
|
||
|
||
if (cyclePosition < delayDuration) {
|
||
// Delay phase - no scrolling
|
||
cachedScrollOffset = 0.0f;
|
||
} else if (cyclePosition < delayDuration + scrollDuration) {
|
||
// Scrolling phase - velocity-based movement
|
||
const double scrollTime = cyclePosition - delayDuration;
|
||
double distance;
|
||
|
||
if (scrollTime <= accelTime) {
|
||
// Acceleration phase - quadratic ease-in
|
||
const double t = scrollTime * invAccelTime; // Multiply instead of divide
|
||
const double smoothT = t * t;
|
||
distance = smoothT * accelDistance;
|
||
} else if (scrollTime <= accelTime + constantVelocityTime) {
|
||
// Constant velocity phase
|
||
const double constantTime = scrollTime - accelTime;
|
||
distance = accelDistance + (constantTime * maxVelocity);
|
||
} else {
|
||
// Deceleration phase - quadratic ease-out
|
||
const double decelStartTime = accelTime + constantVelocityTime;
|
||
const double t = (scrollTime - decelStartTime) * invDecelTime; // Multiply instead of divide
|
||
const double oneMinusT = 1.0 - t;
|
||
const double smoothT = 1.0 - oneMinusT * oneMinusT; // Avoid repeated calculation
|
||
distance = accelDistance + constantVelocityDistance + (smoothT * (minScrollDistance - accelDistance - constantVelocityDistance));
|
||
}
|
||
|
||
// Use branchless min with conditional move behavior
|
||
cachedScrollOffset = static_cast<float>(distance < minScrollDistance ? distance : minScrollDistance);
|
||
} else {
|
||
// Pause phase - stay at end
|
||
cachedScrollOffset = static_cast<float>(this->m_textWidth);
|
||
}
|
||
|
||
lastUpdateTime = currentTime_ns;
|
||
}
|
||
|
||
// Use cached value for consistent display
|
||
this->m_scrollOffset = cachedScrollOffset;
|
||
|
||
// Reset timer when cycle completes
|
||
if (elapsed_seconds >= totalCycleDuration) {
|
||
this->timeIn_ns = currentTime_ns;
|
||
}
|
||
} else {
|
||
renderer->drawString(this->m_ellipsisText, false, this->getX() + 20-1, this->getY() + 45 - yOffset, 23, a(!useClickTextColor ? defaultTextColor : clickTextColor));
|
||
}
|
||
} else {
|
||
// Render the text with special character handling
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
static const std::vector<std::string> specialChars = {ult::STAR_SYMBOL};
|
||
#else
|
||
static const std::vector<std::string> specialChars = {};
|
||
#endif
|
||
renderer->drawStringWithColoredSections(this->m_text, false, specialChars, this->getX() + 20-1, this->getY() + 45 - yOffset, 23,
|
||
(this->m_focused ? (!useClickTextColor ? selectedTextColor : clickTextColor) : (!useClickTextColor ? defaultTextColor : clickTextColor)),
|
||
(this->m_focused ? starColor : selectionStarColor)
|
||
);
|
||
}
|
||
|
||
|
||
// CUSTOM SECTION START (modification for submenu footer color)
|
||
//const std::string& value = this->m_value;
|
||
const s32 xPosition = this->getX() + this->m_maxWidth + 44 + 3;
|
||
const s32 yPosition = this->getY() + 45 - yOffset;
|
||
static constexpr s32 fontSize = 20;
|
||
//bool isFaint = ;
|
||
//bool isFocused = this->m_focused;
|
||
|
||
|
||
//static bool lastRunningInterpreter = ult::runningInterpreter.load(std::memory_order_acquire);
|
||
|
||
// Determine text color
|
||
const auto textColor = this->m_faint ? a(m_faintColor) : a(m_valueColor);
|
||
|
||
if (this->m_value != ult::INPROGRESS_SYMBOL) {
|
||
// Draw the string with the determined text color
|
||
renderer->drawString(this->m_value, false, xPosition, yPosition, fontSize, textColor);
|
||
} else {
|
||
static size_t throbberCounter = 0;
|
||
|
||
|
||
// Reset counter to prevent overflow (every full cycle)
|
||
if (throbberCounter >= 10 * ult::THROBBER_SYMBOLS.size()) {
|
||
throbberCounter = 0;
|
||
}
|
||
|
||
// Get current throbber symbol (changes every 10 frames)
|
||
const size_t symbolIndex = (throbberCounter / 10) % ult::THROBBER_SYMBOLS.size();
|
||
const std::string& currentSymbol = ult::THROBBER_SYMBOLS[symbolIndex];
|
||
|
||
// Instance-specific counter for independent throbber animation
|
||
++throbberCounter;
|
||
|
||
renderer->drawString(currentSymbol, false, xPosition, yPosition, fontSize, textColor);
|
||
}
|
||
//lastRunningInterpreter = ult::runningInterpreter.load(std::memory_order_acquire);
|
||
}
|
||
|
||
virtual void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {
|
||
this->setBoundaries(this->getX()+2+1, this->getY(), this->getWidth()+8+1, m_listItemHeight);
|
||
}
|
||
|
||
virtual bool onClick(u64 keys) override {
|
||
if (keys & KEY_A) {
|
||
this->triggerClickAnimation();
|
||
}
|
||
else if (keys & (KEY_UP | KEY_DOWN | KEY_LEFT | KEY_RIGHT))
|
||
this->m_clickAnimationProgress = 0;
|
||
|
||
return Element::onClick(keys);
|
||
}
|
||
|
||
virtual bool onTouch(TouchEvent event, s32 currX, s32 currY, s32 prevX, s32 prevY, s32 initialX, s32 initialY) override {
|
||
if (event == TouchEvent::Touch)
|
||
this->m_touched = this->inBounds(currX, currY);
|
||
|
||
if (event == TouchEvent::Release && this->m_touched) {
|
||
this->m_touched = false;
|
||
|
||
if (Element::getInputMode() == InputMode::Touch) {
|
||
const bool handled = this->onClick(KEY_A);
|
||
|
||
this->m_clickAnimationProgress = 0;
|
||
return handled;
|
||
}
|
||
}
|
||
|
||
|
||
return false;
|
||
}
|
||
|
||
|
||
virtual void setFocused(bool state) override {
|
||
this->m_scroll = false;
|
||
this->m_scrollOffset = 0;
|
||
this->timeIn_ns = armTicksToNs(armGetSystemTick());
|
||
Element::setFocused(state);
|
||
}
|
||
|
||
virtual Element* requestFocus(Element *oldFocus, FocusDirection direction) override {
|
||
return this;
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the left hand description text of the list item
|
||
*
|
||
* @param text Text
|
||
*/
|
||
inline void setText(const std::string& text) {
|
||
this->m_text = text;
|
||
this->m_scrollText = "";
|
||
this->m_ellipsisText = "";
|
||
this->m_maxWidth = 0;
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the right hand value text of the list item
|
||
*
|
||
* @param value Text
|
||
* @param faint Should the text be drawn in a glowing green or a faint gray
|
||
*/
|
||
inline void setValue(const std::string& value, bool faint = false) {
|
||
this->m_value = value;
|
||
this->m_faint = faint;
|
||
this->m_maxWidth = 0;
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the value color
|
||
*
|
||
* @param value_color color of the value
|
||
*/
|
||
inline void setValueColor(Color value_color) {
|
||
this->m_valueColor = value_color;
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the faint color
|
||
*
|
||
* @param faint_color color of the faint
|
||
*/
|
||
inline void setFaintColor(Color faint_color) {
|
||
this->m_faintColor = faint_color;
|
||
}
|
||
|
||
/**
|
||
* @brief Gets the left hand description text of the list item
|
||
*
|
||
* @return Text
|
||
*/
|
||
inline const std::string& getText() const {
|
||
return this->m_text;
|
||
}
|
||
|
||
/**
|
||
* @brief Gets the right hand value text of the list item
|
||
*
|
||
* @return Value
|
||
*/
|
||
inline const std::string& getValue() {
|
||
return this->m_value;
|
||
}
|
||
|
||
protected:
|
||
u64 timeIn_ns;
|
||
|
||
std::string m_text;
|
||
std::string m_value;
|
||
std::string m_scrollText;
|
||
std::string m_ellipsisText;
|
||
u32 m_listItemHeight = tsl::style::ListItemDefaultHeight;
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
bool m_useScriptKey = false;
|
||
#endif
|
||
Color m_valueColor;
|
||
Color m_faintColor;
|
||
|
||
bool m_scroll = false;
|
||
bool m_trunctuated = false;
|
||
bool m_faint = false;
|
||
|
||
bool m_touched = false;
|
||
|
||
u16 m_maxScroll = 0;
|
||
u16 m_scrollOffset = 0;
|
||
u32 m_maxWidth = 0;
|
||
u32 m_textWidth = 0;
|
||
u16 m_scrollAnimationCounter = 0;
|
||
};
|
||
|
||
/**
|
||
* @brief A toggleable list item that changes the state from On to Off when the A button gets pressed
|
||
*
|
||
*/
|
||
class ToggleListItem : public ListItem {
|
||
public:
|
||
/**
|
||
* @brief Constructor
|
||
*
|
||
* @param text Initial description text
|
||
* @param initialState Is the toggle set to On or Off initially
|
||
* @param onValue Value drawn if the toggle is on
|
||
* @param offValue Value drawn if the toggle is off
|
||
*/
|
||
ToggleListItem(const std::string& text, bool initialState, const std::string& onValue = ult::ON, const std::string& offValue = ult::OFF, bool isMini = false)
|
||
: ListItem(text, "", isMini), m_state(initialState), m_onValue(onValue), m_offValue(offValue) {
|
||
this->setState(this->m_state);
|
||
}
|
||
|
||
virtual ~ToggleListItem() {}
|
||
|
||
virtual bool onClick(u64 keys) override {
|
||
|
||
// Handle KEY_A for toggling
|
||
if (keys & KEY_A) {
|
||
this->m_state = !this->m_state;
|
||
|
||
this->setState(this->m_state);
|
||
this->m_stateChangedListener(this->m_state);
|
||
|
||
return ListItem::onClick(keys);
|
||
}
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
// Handle SCRIPT_KEY for executing script logic
|
||
else if (keys & SCRIPT_KEY) {
|
||
// Trigger the script key listener
|
||
if (this->m_scriptKeyListener) {
|
||
this->m_scriptKeyListener(this->m_state); // Pass the current state to the script key listener
|
||
}
|
||
return ListItem::onClick(keys);
|
||
}
|
||
#endif
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* @brief Gets the current state of the toggle
|
||
*
|
||
* @return State
|
||
*/
|
||
virtual inline bool getState() {
|
||
return this->m_state;
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the current state of the toggle. Updates the Value
|
||
*
|
||
* @param state State
|
||
*/
|
||
virtual inline void setState(bool state) {
|
||
this->m_state = state;
|
||
this->setValue(state ? this->m_onValue : this->m_offValue, !state);
|
||
}
|
||
|
||
/**
|
||
* @brief Adds a listener that gets called whenever the state of the toggle changes
|
||
*
|
||
* @param stateChangedListener Listener with the current state passed in as parameter
|
||
*/
|
||
void setStateChangedListener(std::function<void(bool)> stateChangedListener) {
|
||
this->m_stateChangedListener = stateChangedListener;
|
||
}
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
// Attach the script key listener for SCRIPT_KEY handling
|
||
void setScriptKeyListener(std::function<void(bool)> scriptKeyListener) {
|
||
this->m_scriptKeyListener = scriptKeyListener;
|
||
}
|
||
#endif
|
||
|
||
protected:
|
||
bool m_state = true;
|
||
std::string m_onValue, m_offValue;
|
||
|
||
std::function<void(bool)> m_stateChangedListener = [](bool){};
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
std::function<void(bool)> m_scriptKeyListener = nullptr; // Script key listener (with state)
|
||
#endif
|
||
};
|
||
|
||
class MiniToggleListItem : public ToggleListItem {
|
||
public:
|
||
// Constructor for MiniToggleListItem, with no `isMini` boolean.
|
||
MiniToggleListItem(const std::string& text, bool initialState, const std::string& onValue = ult::ON, const std::string& offValue = ult::OFF)
|
||
: ToggleListItem(text, initialState, onValue, offValue, true) {
|
||
}
|
||
|
||
// Destructor if needed (inherits default behavior from ListItem)
|
||
virtual ~MiniToggleListItem() {}
|
||
};
|
||
|
||
|
||
class DummyListItem : public ListItem {
|
||
public:
|
||
DummyListItem()
|
||
: ListItem("") { // Use an empty string for the base class constructor
|
||
// Set the properties to indicate it's a dummy item
|
||
this->m_text = "";
|
||
this->m_value = "";
|
||
this->m_maxWidth = 0;
|
||
this->width = 0;
|
||
this->height = 0;
|
||
m_isItem = false;
|
||
}
|
||
|
||
virtual ~DummyListItem() {}
|
||
|
||
// Override the draw method to do nothing
|
||
virtual void draw(gfx::Renderer* renderer) override {
|
||
// Intentionally left blank
|
||
}
|
||
|
||
// Override the layout method to set the dimensions to zero
|
||
virtual void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {
|
||
//this->setBoundaries(parentX, parentY, 0, 0); // Zero size
|
||
this->setBoundaries(this->getX(), this->getY(), 0, 0);
|
||
}
|
||
|
||
// Override the requestFocus method to allow this item to be focusable
|
||
virtual inline Element* requestFocus(Element* oldFocus, FocusDirection direction) override {
|
||
return this; // Allow this item to be focusable
|
||
}
|
||
|
||
//// Optionally override onClick and onTouch to handle interactions
|
||
//virtual bool onClick(u64 keys) override {
|
||
// return true; // Consume the click event
|
||
//}
|
||
//
|
||
//virtual bool onTouch(TouchEvent event, s32 currX, s32 currY, s32 prevX, s32 prevY, s32 initialX, s32 initialY) override {
|
||
// return true; // Consume the touch event
|
||
//}
|
||
};
|
||
|
||
|
||
class CategoryHeader : public Element {
|
||
public:
|
||
CategoryHeader(const std::string &title, bool hasSeparator = true)
|
||
: m_text(title), m_hasSeparator(hasSeparator), timeIn_ns(0),
|
||
m_scroll(false), m_truncated(false), m_scrollOffset(0.0f),
|
||
m_maxWidth(0), m_textWidth(0) {
|
||
ult::applyLangReplacements(m_text);
|
||
ult::convertComboToUnicode(m_text);
|
||
m_isItem = false;
|
||
}
|
||
|
||
virtual ~CategoryHeader() {}
|
||
|
||
virtual void draw(gfx::Renderer *renderer) override {
|
||
static const std::vector<std::string> specialChars = {""};
|
||
|
||
// Calculate widths if not done yet
|
||
if (!m_maxWidth) {
|
||
calculateWidths(renderer);
|
||
}
|
||
|
||
// Draw separator if needed
|
||
if (this->m_hasSeparator) {
|
||
renderer->drawRect(this->getX()+1+1, this->getBottomBound() - 29-4, 4, 22, (headerSeparatorColor));
|
||
}
|
||
|
||
// Determine text position
|
||
const int textX = m_hasSeparator ? (this->getX() + 15+1) : this->getX();
|
||
const int textY = this->getBottomBound() - 12-4;
|
||
|
||
// Handle scrolling text if truncated
|
||
if (m_truncated) {
|
||
if (!m_scroll) {
|
||
m_scroll = true;
|
||
timeIn_ns = armTicksToNs(armGetSystemTick());
|
||
}
|
||
|
||
// Calculate scissoring bounds that respect parent clipping
|
||
const int scissorX = textX;
|
||
const int scissorY = textY - 16;
|
||
const int scissorWidth = m_maxWidth;
|
||
const int scissorHeight = 24;
|
||
|
||
// Get parent bounds (you'll need to implement this based on your parent system)
|
||
// This assumes your parent has some way to get its visible bounds
|
||
if (Element* parent = this->getParent()) {
|
||
const int parentTop = parent->getY()-8; // or whatever method gets the top bound
|
||
const int parentBottom = parent->getBottomBound(); // or equivalent
|
||
const int parentLeft = parent->getX();
|
||
const int parentRight = parent->getX() + parent->getWidth();
|
||
|
||
// Clip scissor rectangle to parent bounds
|
||
const int clipLeft = std::max(scissorX, parentLeft);
|
||
const int clipRight = std::min(scissorX + scissorWidth, parentRight);
|
||
const int clipTop = std::max(scissorY, parentTop);
|
||
const int clipBottom = std::min(scissorY + scissorHeight, parentBottom);
|
||
|
||
// Only enable scissoring if there's a visible area
|
||
if (clipLeft < clipRight && clipTop < clipBottom) {
|
||
renderer->enableScissoring(clipLeft, clipTop,
|
||
clipRight - clipLeft,
|
||
clipBottom - clipTop);
|
||
|
||
renderer->drawStringWithColoredSections(m_scrollText, false, specialChars,
|
||
textX - static_cast<s32>(m_scrollOffset), textY, 16,
|
||
(headerTextColor), textSeparatorColor);
|
||
|
||
renderer->disableScissoring();
|
||
} else {
|
||
// Draw normal or ellipsis text
|
||
//const std::string& displayText = m_truncated ? m_ellipsisText : m_text;
|
||
renderer->drawStringWithColoredSections(m_text, false, specialChars,
|
||
textX, textY, 16, (headerTextColor), textSeparatorColor);
|
||
}
|
||
// If completely clipped, don't draw anything
|
||
} else {
|
||
// Draw normal or ellipsis text
|
||
//const std::string& displayText = m_truncated ? m_ellipsisText : m_text;
|
||
renderer->drawStringWithColoredSections(m_text, false, specialChars,
|
||
textX, textY, 16, (headerTextColor), textSeparatorColor);
|
||
}
|
||
|
||
handleScrolling();
|
||
} else {
|
||
// Draw normal or ellipsis text
|
||
//const std::string& displayText = m_truncated ? m_ellipsisText : m_text;
|
||
renderer->drawStringWithColoredSections(m_text, false, specialChars,
|
||
textX, textY, 16, (headerTextColor), textSeparatorColor);
|
||
}
|
||
}
|
||
|
||
virtual void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {
|
||
// Check if the CategoryHeader is part of a list and if it's the first entry in it, half it's height
|
||
if (List *list = static_cast<List*>(this->getParent()); list != nullptr) {
|
||
if (list->getIndexInList(this) == 0) {
|
||
this->setBoundaries(this->getX(), this->getY(), this->getWidth(), 29+4);
|
||
return;
|
||
}
|
||
}
|
||
this->setBoundaries(this->getX(), this->getY(), this->getWidth(), tsl::style::ListItemDefaultHeight *0.90);
|
||
}
|
||
|
||
virtual bool onClick(u64 keys) {
|
||
return false;
|
||
}
|
||
|
||
virtual Element* requestFocus(Element *oldFocus, FocusDirection direction) override {
|
||
return nullptr;
|
||
}
|
||
|
||
virtual void setFocused(bool state) override {}
|
||
|
||
inline void setText(const std::string &text) {
|
||
if (this->m_text != text) {
|
||
this->m_text = text;
|
||
ult::applyLangReplacements(m_text);
|
||
ult::convertComboToUnicode(m_text);
|
||
//resetTextProperties();
|
||
}
|
||
}
|
||
|
||
inline const std::string& getText() const {
|
||
return this->m_text;
|
||
}
|
||
|
||
private:
|
||
std::string m_text;
|
||
bool m_hasSeparator;
|
||
|
||
// Scrolling properties (matching ListItem)
|
||
u64 timeIn_ns;
|
||
std::string m_scrollText;
|
||
//std::string m_ellipsisText;
|
||
bool m_scroll;
|
||
bool m_truncated;
|
||
float m_scrollOffset;
|
||
u32 m_maxWidth;
|
||
u32 m_textWidth;
|
||
|
||
|
||
// Frame rate compensation - cache calculations to reduce stutter
|
||
u64 lastUpdateTime = 0;
|
||
float cachedScrollOffset = 0.0f;
|
||
|
||
// Pre-compute constants as statics to avoid recalculation
|
||
bool constantsInitialized = false;
|
||
double totalCycleDuration;
|
||
double delayDuration;
|
||
double scrollDuration;
|
||
double accelTime;
|
||
double constantVelocityTime;
|
||
double maxVelocity;
|
||
double accelDistance;
|
||
double constantVelocityDistance;
|
||
double minScrollDistance;
|
||
double invAccelTime;
|
||
double invDecelTime;
|
||
double invBillion;
|
||
|
||
void calculateWidths(gfx::Renderer* renderer) {
|
||
// Available width (accounting for separator and margins)
|
||
m_maxWidth = getWidth() - (m_hasSeparator ? 20-3 : 4);
|
||
|
||
// Get actual text width
|
||
const u32 width = renderer->getTextDimensions(m_text, false, 16).first;
|
||
m_truncated = width > m_maxWidth;
|
||
|
||
if (m_truncated) {
|
||
// Build scroll text: "text text"
|
||
m_scrollText.clear();
|
||
m_scrollText.reserve(m_text.size() * 2 + 8);
|
||
m_scrollText.append(m_text).append(" ");
|
||
m_textWidth = renderer->getTextDimensions(m_scrollText, false, 16).first;
|
||
m_scrollText.append(m_text);
|
||
|
||
// Create ellipsis text
|
||
//m_ellipsisText = renderer->limitStringLength(m_text, false, 16, m_maxWidth);
|
||
} else {
|
||
m_textWidth = width;
|
||
}
|
||
}
|
||
|
||
void handleScrolling() {
|
||
const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
const u64 elapsed_ns = currentTime_ns - timeIn_ns;
|
||
|
||
|
||
if (!constantsInitialized || minScrollDistance != static_cast<double>(m_textWidth)) {
|
||
// Constants for velocity-based scrolling (3 second pauses as requested)
|
||
delayDuration = 3.0; // 3 second pause at start
|
||
static constexpr double pauseDuration = 2.0; // 3 second pause at end
|
||
maxVelocity = 100.0; // Adjust for desired scroll speed
|
||
accelTime = 0.5;
|
||
static constexpr double decelTime = 0.5;
|
||
|
||
// Pre-calculate derived constants
|
||
minScrollDistance = static_cast<double>(m_textWidth);
|
||
accelDistance = 0.5 * maxVelocity * accelTime;
|
||
const double decelDistance = 0.5 * maxVelocity * decelTime;
|
||
constantVelocityDistance = std::max(0.0, minScrollDistance - accelDistance - decelDistance);
|
||
constantVelocityTime = constantVelocityDistance / maxVelocity;
|
||
scrollDuration = accelTime + constantVelocityTime + decelTime;
|
||
totalCycleDuration = delayDuration + scrollDuration + pauseDuration;
|
||
|
||
// Pre-calculate reciprocals for faster division
|
||
invAccelTime = 1.0 / accelTime;
|
||
invDecelTime = 1.0 / decelTime;
|
||
invBillion = 1.0 / 1000000000.0;
|
||
|
||
constantsInitialized = true;
|
||
}
|
||
|
||
// Fast ns to seconds conversion
|
||
const double elapsed_seconds = static_cast<double>(elapsed_ns) * invBillion;
|
||
|
||
// Update at consistent intervals regardless of frame rate
|
||
if (currentTime_ns - lastUpdateTime >= 8333333ULL) { // ~120 FPS update rate
|
||
// Use std::fmod for modulo - it's optimized and faster than loops
|
||
const double cyclePosition = std::fmod(elapsed_seconds, totalCycleDuration);
|
||
|
||
if (cyclePosition < delayDuration) {
|
||
// Delay phase - no scrolling (3 second pause)
|
||
cachedScrollOffset = 0.0f;
|
||
} else if (cyclePosition < delayDuration + scrollDuration) {
|
||
// Scrolling phase - velocity-based movement
|
||
const double scrollTime = cyclePosition - delayDuration;
|
||
double distance;
|
||
|
||
if (scrollTime <= accelTime) {
|
||
// Acceleration phase - quadratic ease-in
|
||
const double t = scrollTime * invAccelTime;
|
||
const double smoothT = t * t;
|
||
distance = smoothT * accelDistance;
|
||
} else if (scrollTime <= accelTime + constantVelocityTime) {
|
||
// Constant velocity phase
|
||
const double constantTime = scrollTime - accelTime;
|
||
distance = accelDistance + (constantTime * maxVelocity);
|
||
} else {
|
||
// Deceleration phase - quadratic ease-out
|
||
const double decelStartTime = accelTime + constantVelocityTime;
|
||
const double t = (scrollTime - decelStartTime) * invDecelTime;
|
||
const double oneMinusT = 1.0 - t;
|
||
const double smoothT = 1.0 - oneMinusT * oneMinusT;
|
||
distance = accelDistance + constantVelocityDistance + (smoothT * (minScrollDistance - accelDistance - constantVelocityDistance));
|
||
}
|
||
|
||
// Use branchless min
|
||
cachedScrollOffset = static_cast<float>(distance < minScrollDistance ? distance : minScrollDistance);
|
||
} else {
|
||
// Pause phase - stay at end (3 second pause)
|
||
cachedScrollOffset = static_cast<float>(m_textWidth);
|
||
}
|
||
|
||
lastUpdateTime = currentTime_ns;
|
||
}
|
||
|
||
// Use cached value for consistent display
|
||
m_scrollOffset = cachedScrollOffset;
|
||
|
||
// Reset timer when cycle completes
|
||
if (elapsed_seconds >= totalCycleDuration) {
|
||
timeIn_ns = currentTime_ns;
|
||
}
|
||
}
|
||
|
||
//void resetTextProperties() {
|
||
// m_scrollText.clear();
|
||
// m_ellipsisText.clear();
|
||
// m_maxWidth = 0;
|
||
//}
|
||
};
|
||
|
||
|
||
/**
|
||
* @brief A customizable analog trackbar going from 0% to 100% (like the brightness slider)
|
||
*
|
||
*/
|
||
class TrackBar : public Element {
|
||
public:
|
||
/**
|
||
* @brief Constructor
|
||
*
|
||
* @param icon Icon shown next to the track bar
|
||
* @param usingStepTrackbar Whether this is a step trackbar
|
||
* @param usingNamedStepTrackbar Whether this is a named step trackbar
|
||
* @param useV2Style Whether to use V2 visual style (label + value instead of icon)
|
||
* @param label Label text for V2 style
|
||
* @param units Units text for V2 style
|
||
*/
|
||
TrackBar(const char icon[3], bool usingStepTrackbar=false, bool usingNamedStepTrackbar = false,
|
||
bool useV2Style = false, const std::string& label = "", const std::string& units = "")
|
||
: m_icon(icon), m_usingStepTrackbar(usingStepTrackbar), m_usingNamedStepTrackbar(usingNamedStepTrackbar),
|
||
m_useV2Style(useV2Style), m_label(label), m_units(units) {
|
||
m_isItem = true;
|
||
}
|
||
|
||
virtual ~TrackBar() {}
|
||
|
||
virtual Element* requestFocus(Element *oldFocus, FocusDirection direction) {
|
||
return this;
|
||
}
|
||
|
||
virtual bool handleInput(u64 keysDown, u64 keysHeld, const HidTouchState &touchPos, HidAnalogStickState leftJoyStick, HidAnalogStickState rightJoyStick) override {
|
||
if (keysHeld & KEY_LEFT && keysHeld & KEY_RIGHT)
|
||
return true;
|
||
|
||
if (keysHeld & KEY_LEFT) {
|
||
if (this->m_value > 0) {
|
||
this->m_value--;
|
||
this->m_valueChangedListener(this->m_value);
|
||
return true;
|
||
}
|
||
}
|
||
|
||
if (keysHeld & KEY_RIGHT) {
|
||
if (this->m_value < 100) {
|
||
this->m_value++;
|
||
this->m_valueChangedListener(this->m_value);
|
||
return true;
|
||
}
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
virtual bool onTouch(TouchEvent event, s32 currX, s32 currY, s32 prevX, s32 prevY, s32 initialX, s32 initialY) override {
|
||
const u16 trackBarWidth = this->getWidth() - 95;
|
||
const u16 handlePos = (trackBarWidth * (this->m_value - 0)) / (100 - 0);
|
||
const s32 circleCenterX = this->getX() + 59 + handlePos;
|
||
const s32 circleCenterY = this->getY() + 40 + 16 - 1;
|
||
static constexpr s32 circleRadius = 16;
|
||
|
||
const bool touchInCircle = (std::abs(initialX - circleCenterX) <= circleRadius) && (std::abs(initialY - circleCenterY) <= circleRadius);
|
||
|
||
|
||
if (event == TouchEvent::Release) {
|
||
this->m_interactionLocked = false;
|
||
touchInSliderBounds = false;
|
||
return false;
|
||
}
|
||
|
||
if (!this->m_interactionLocked && (touchInCircle || touchInSliderBounds)) {
|
||
touchInSliderBounds = true;
|
||
if (currX > this->getLeftBound() + 50 && currX < this->getRightBound() && currY > this->getTopBound() && currY < this->getBottomBound()) {
|
||
s16 newValue = (static_cast<float>(currX - (this->getX() + 60)) / static_cast<float>(this->getWidth() - 95)) * 100;
|
||
|
||
if (newValue < 0) {
|
||
newValue = 0;
|
||
} else if (newValue > 100) {
|
||
newValue = 100;
|
||
}
|
||
|
||
if (newValue != this->m_value) {
|
||
this->m_value = newValue;
|
||
this->m_valueChangedListener(this->getProgress());
|
||
}
|
||
|
||
return true;
|
||
}
|
||
}
|
||
else
|
||
this->m_interactionLocked = true;
|
||
|
||
return false;
|
||
}
|
||
|
||
|
||
// Define drawBar function outside the draw method
|
||
void drawBar(gfx::Renderer *renderer, s32 x, s32 y, u16 width, Color& color, bool isRounded = true) {
|
||
if (isRounded) {
|
||
renderer->drawUniformRoundedRect(x, y, width, 7, a(color));
|
||
} else {
|
||
renderer->drawRect(x, y, width, 7, a(color));
|
||
}
|
||
}
|
||
|
||
virtual void draw(gfx::Renderer *renderer) override {
|
||
static float lastBottomBound;
|
||
|
||
s32 xPos = this->getX() + 59;
|
||
s32 yPos = this->getY() + 40 + 16 - 1;
|
||
s32 width = this->getWidth() - 95;
|
||
u16 handlePos = width * (this->m_value) / (100);
|
||
|
||
if (!m_usingNamedStepTrackbar) {
|
||
yPos -= 11;
|
||
}
|
||
|
||
s32 iconOffset = 0;
|
||
|
||
if (!m_useV2Style && m_icon[0] != '\0') {
|
||
s32 iconWidth = 23;//tsl::gfx::calculateStringWidth(m_icon, 23);
|
||
iconOffset = 14 + iconWidth;
|
||
xPos += iconOffset;
|
||
width -= iconOffset;
|
||
handlePos = (width) * (this->m_value) / (100);
|
||
}
|
||
|
||
// Draw track bar background
|
||
drawBar(renderer, xPos, yPos-3, width, trackBarEmptyColor, !m_usingNamedStepTrackbar);
|
||
|
||
if (!this->m_focused) {
|
||
drawBar(renderer, xPos, yPos-3, handlePos, trackBarFullColor, !m_usingNamedStepTrackbar);
|
||
renderer->drawCircle(xPos + handlePos, yPos, 16, true, a(trackBarSliderBorderColor));
|
||
renderer->drawCircle(xPos + handlePos, yPos, 13, true, a((m_unlockedTrackbar || touchInSliderBounds) ? trackBarSliderMalleableColor : trackBarSliderColor));
|
||
} else {
|
||
touchInSliderBounds = false;
|
||
if (m_unlockedTrackbar != ult::unlockedSlide.load(std::memory_order_acquire))
|
||
ult::unlockedSlide.store(m_unlockedTrackbar, std::memory_order_release);
|
||
drawBar(renderer, xPos, yPos-3, handlePos, trackBarFullColor, !m_usingNamedStepTrackbar);
|
||
renderer->drawCircle(xPos + x + handlePos, yPos +y, 16, true, a(highlightColor));
|
||
renderer->drawCircle(xPos + x + handlePos, yPos +y, 12, true, a((ult::allowSlide.load(std::memory_order_acquire) || m_unlockedTrackbar) ? trackBarSliderMalleableColor : trackBarSliderColor));
|
||
}
|
||
|
||
// Draw icon (original style) or label + value (V2 style)
|
||
if (m_useV2Style) {
|
||
// V2 Style: Draw label and value
|
||
std::string labelPart = this->m_label;
|
||
ult::removeTag(labelPart);
|
||
|
||
std::string valuePart;
|
||
if (!m_usingNamedStepTrackbar) {
|
||
valuePart = (this->m_units == "%" || this->m_units == "°C" || this->m_units == "°F") ?
|
||
ult::to_string(this->m_value) + this->m_units :
|
||
ult::to_string(this->m_value) + (this->m_units.empty() ? "" : " ") + this->m_units;
|
||
} else {
|
||
valuePart = this->m_selection;
|
||
}
|
||
|
||
const auto valueWidth = renderer->getTextDimensions(valuePart, false, 16).first;
|
||
|
||
renderer->drawString(labelPart, false, this->getX() + 59, this->getY() + 14 + 16, 16,
|
||
((!this->m_focused || !ult::useSelectionText) ? (defaultTextColor) : (selectedTextColor)));
|
||
|
||
renderer->drawString(valuePart, false, this->getWidth() -17 - valueWidth, this->getY() + 14 + 16, 16, (this->m_focused && ult::useSelectionValue) ? selectedValueTextColor : onTextColor);
|
||
} else {
|
||
// Original Style: Draw icon
|
||
if (m_icon[0] != '\0')
|
||
renderer->drawString(this->m_icon, false, this->getX()+42, this->getY() + 50+2, 23, a(tsl::style::color::ColorText));
|
||
}
|
||
|
||
//renderer->drawRect(this->getX(), this->getY(), this->getWidth(), 1, a(tsl::style::color::ColorFrame));
|
||
//renderer->drawRect(this->getX(), this->getBottomBound(), this->getWidth(), 1, a(tsl::style::color::ColorFrame));
|
||
|
||
//u16 handlePos = (this->getWidth() - 95) * static_cast<float>(this->m_value) / 100;
|
||
//renderer->drawCircle(this->getX() + 60, this->getY() + 42, 2, true, a(tsl::style::color::ColorHighlight));
|
||
//renderer->drawCircle(this->getX() + 60 + this->getWidth() - 95, this->getY() + 42, 2, true, a(tsl::style::color::ColorFrame));
|
||
//renderer->drawRect(this->getX() + 60 + handlePos, this->getY() + 40, this->getWidth() - 95 - handlePos, 5, a(tsl::style::color::ColorFrame));
|
||
//renderer->drawRect(this->getX() + 60, this->getY() + 40, handlePos, 5, a(tsl::style::color::ColorHighlight));
|
||
//
|
||
//renderer->drawCircle(this->getX() + 62 + handlePos, this->getY() + 42, 18, true, a(tsl::style::color::ColorHandle));
|
||
//renderer->drawCircle(this->getX() + 62 + handlePos, this->getY() + 42, 18, false, a(tsl::style::color::ColorFrame));
|
||
|
||
|
||
if (lastBottomBound != this->getTopBound())
|
||
renderer->drawRect(this->getX() + 4+20-1, this->getTopBound(), this->getWidth() + 6 + 10+20 +4, 1, a(separatorColor));
|
||
renderer->drawRect(this->getX() + 4+20-1, this->getBottomBound(), this->getWidth() + 6 + 10+20 +4, 1, a(separatorColor));
|
||
lastBottomBound = this->getBottomBound();
|
||
}
|
||
|
||
virtual void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {
|
||
this->setBoundaries(this->getX() - 16 , this->getY(), this->getWidth()+20+4, tsl::style::TrackBarDefaultHeight );
|
||
}
|
||
|
||
virtual void drawFocusBackground(gfx::Renderer *renderer) {
|
||
// No background drawn here in HOS
|
||
}
|
||
|
||
virtual void drawHighlight(gfx::Renderer *renderer) override {
|
||
|
||
// Get current time using ARM system tick for animation timing
|
||
const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
|
||
// High precision time calculation - matches standard cosine wave timing
|
||
const double time_seconds = static_cast<double>(currentTime_ns) / 1000000000.0;
|
||
|
||
// Standard cosine wave calculation with high precision
|
||
progress = (std::cos(2.0 * ult::_M_PI * std::fmod(time_seconds, 1.0) - ult::_M_PI / 2) + 1.0) / 2.0;
|
||
|
||
// High precision floating point color interpolation
|
||
highlightColor = {
|
||
static_cast<u8>(highlightColor2.r + (highlightColor1.r - highlightColor2.r) * progress + 0.5),
|
||
static_cast<u8>(highlightColor2.g + (highlightColor1.g - highlightColor2.g) * progress + 0.5),
|
||
static_cast<u8>(highlightColor2.b + (highlightColor1.b - highlightColor2.b) * progress + 0.5),
|
||
0xF
|
||
};
|
||
|
||
// Initialize position offsets
|
||
x = 0;
|
||
y = 0;
|
||
|
||
if (this->m_highlightShaking) {
|
||
//const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
t_ns = currentTime_ns - this->m_highlightShakingStartTime; // Changed
|
||
if (t_ns >= 100000000) // 100ms in nanoseconds
|
||
this->m_highlightShaking = false;
|
||
else {
|
||
amplitude = std::rand() % 5 + 5;
|
||
|
||
switch (this->m_highlightShakingDirection) {
|
||
case FocusDirection::Up:
|
||
y -= shakeAnimation(t_ns, amplitude); // Changed parameter
|
||
break;
|
||
case FocusDirection::Down:
|
||
y += shakeAnimation(t_ns, amplitude); // Changed parameter
|
||
break;
|
||
case FocusDirection::Left:
|
||
x -= shakeAnimation(t_ns, amplitude); // Changed parameter
|
||
break;
|
||
case FocusDirection::Right:
|
||
x += shakeAnimation(t_ns, amplitude); // Changed parameter
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
|
||
x = std::clamp(x, -amplitude, amplitude);
|
||
y = std::clamp(y, -amplitude, amplitude);
|
||
}
|
||
}
|
||
|
||
if (ult::useSelectionBG) {
|
||
if (ult::expandedMemory)
|
||
renderer->drawRectMultiThreaded(this->getX() + x +19, this->getY() + y, this->getWidth()-11-4, this->getHeight(), aWithOpacity(selectionBGColor)); // CUSTOM MODIFICATION
|
||
else
|
||
renderer->drawRect(this->getX() + x +19, this->getY() + y, this->getWidth()-11-4, this->getHeight(), aWithOpacity(selectionBGColor)); // CUSTOM MODIFICATION
|
||
|
||
|
||
//renderer->drawRect(this->getX() + x +19, this->getY() + y, this->getWidth()-11-4, this->getHeight(), a(selectionBGColor)); // CUSTOM MODIFICATION
|
||
}
|
||
|
||
renderer->drawBorderedRoundedRect(this->getX() + x +19, this->getY() + y, this->getWidth()-11, this->getHeight(), 5, 5, a(highlightColor));
|
||
|
||
if (!ult::onTrackBar.load(std::memory_order_acquire))
|
||
ult::onTrackBar.store(true, std::memory_order_release);
|
||
}
|
||
|
||
/**
|
||
* @brief Gets the current value of the trackbar
|
||
*
|
||
* @return State
|
||
*/
|
||
virtual inline u8 getProgress() {
|
||
return this->m_value;
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the current state of the toggle. Updates the Value
|
||
*
|
||
* @param state State
|
||
*/
|
||
virtual void setProgress(u8 value) {
|
||
this->m_value = value;
|
||
}
|
||
|
||
/**
|
||
* @brief Adds a listener that gets called whenever the state of the toggle changes
|
||
*
|
||
* @param stateChangedListener Listener with the current state passed in as parameter
|
||
*/
|
||
void setValueChangedListener(std::function<void(u8)> valueChangedListener) {
|
||
this->m_valueChangedListener = valueChangedListener;
|
||
}
|
||
|
||
protected:
|
||
const char *m_icon = nullptr;
|
||
s16 m_value = 0;
|
||
bool m_interactionLocked = false;
|
||
|
||
std::function<void(u8)> m_valueChangedListener = [](u8){};
|
||
|
||
bool m_usingStepTrackbar = false;
|
||
bool m_usingNamedStepTrackbar = false;
|
||
bool m_unlockedTrackbar = true;
|
||
bool touchInSliderBounds = false;
|
||
|
||
// V2 Style properties
|
||
bool m_useV2Style = false;
|
||
std::string m_label;
|
||
std::string m_units;
|
||
std::string m_selection; // Used for named step trackbars
|
||
};
|
||
|
||
|
||
/**
|
||
* @brief A customizable analog trackbar going from 0% to 100% but using discrete steps (Like the volume slider)
|
||
*
|
||
*/
|
||
class StepTrackBar : public TrackBar {
|
||
public:
|
||
/**
|
||
* @brief Constructor
|
||
*
|
||
* @param icon Icon shown next to the track bar
|
||
* @param numSteps Number of steps the track bar has
|
||
* @param usingNamedStepTrackbar Whether this is a named step trackbar
|
||
* @param useV2Style Whether to use V2 visual style (label + value instead of icon)
|
||
* @param label Label text for V2 style
|
||
* @param units Units text for V2 style
|
||
*/
|
||
StepTrackBar(const char icon[3], size_t numSteps, bool usingNamedStepTrackbar = false,
|
||
bool useV2Style = false, const std::string& label = "", const std::string& units = "")
|
||
: TrackBar(icon, true, usingNamedStepTrackbar, useV2Style, label, units), m_numSteps(numSteps) { }
|
||
|
||
virtual ~StepTrackBar() {}
|
||
|
||
virtual bool handleInput(u64 keysDown, u64 keysHeld, const HidTouchState &touchPos, HidAnalogStickState leftJoyStick, HidAnalogStickState rightJoyStick) override {
|
||
static u32 tick = 0;
|
||
|
||
if (keysHeld & KEY_LEFT && keysHeld & KEY_RIGHT) {
|
||
tick = 0;
|
||
return true;
|
||
}
|
||
|
||
if (keysHeld & (KEY_LEFT | KEY_RIGHT)) {
|
||
if ((tick == 0 || tick > 20) && (tick % 3) == 0) {
|
||
if (keysHeld & KEY_LEFT && this->m_value > 0) {
|
||
this->m_value = std::max(this->m_value - (100 / (this->m_numSteps - 1)), 0);
|
||
} else if (keysHeld & KEY_RIGHT && this->m_value < 100) {
|
||
this->m_value = std::min(this->m_value + (100 / (this->m_numSteps - 1)), 100);
|
||
} else {
|
||
return false;
|
||
}
|
||
this->m_valueChangedListener(this->getProgress());
|
||
}
|
||
tick++;
|
||
return true;
|
||
} else {
|
||
tick = 0;
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
virtual bool onTouch(TouchEvent event, s32 currX, s32 currY, s32 prevX, s32 prevY, s32 initialX, s32 initialY) override {
|
||
if (this->inBounds(initialX, initialY)) {
|
||
if (currY > this->getTopBound() && currY < this->getBottomBound()) {
|
||
s16 newValue = (static_cast<float>(currX - (this->getX() + 60)) / static_cast<float>(this->getWidth() - 95)) * 100;
|
||
|
||
if (newValue < 0) {
|
||
newValue = 0;
|
||
} else if (newValue > 100) {
|
||
newValue = 100;
|
||
} else {
|
||
newValue = std::round(newValue / (100.0F / (this->m_numSteps - 1))) * (100.0F / (this->m_numSteps - 1));
|
||
}
|
||
|
||
if (newValue != this->m_value) {
|
||
this->m_value = newValue;
|
||
this->m_valueChangedListener(this->getProgress());
|
||
}
|
||
|
||
return true;
|
||
}
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* @brief Gets the current value of the trackbar
|
||
*
|
||
* @return State
|
||
*/
|
||
virtual inline u8 getProgress() override {
|
||
return this->m_value / (100 / (this->m_numSteps - 1));
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the current state of the toggle. Updates the Value
|
||
*
|
||
* @param state State
|
||
*/
|
||
virtual void setProgress(u8 value) override {
|
||
value = std::min(value, u8(this->m_numSteps - 1));
|
||
this->m_value = value * (100 / (this->m_numSteps - 1));
|
||
}
|
||
|
||
protected:
|
||
u8 m_numSteps = 1;
|
||
};
|
||
|
||
|
||
/**
|
||
* @brief A customizable trackbar with multiple discrete steps with specific names. Name gets displayed above the bar
|
||
*
|
||
*/
|
||
class NamedStepTrackBar : public StepTrackBar {
|
||
public:
|
||
/**
|
||
* @brief Constructor
|
||
*
|
||
* @param icon Icon shown next to the track bar
|
||
* @param stepDescriptions Step names displayed above the track bar
|
||
* @param useV2Style Whether to use V2 visual style (label + value instead of icon)
|
||
* @param label Label text for V2 style
|
||
*/
|
||
NamedStepTrackBar(const char icon[3], std::initializer_list<std::string> stepDescriptions,
|
||
bool useV2Style = false, const std::string& label = "")
|
||
: StepTrackBar(icon, stepDescriptions.size(), true, useV2Style, label, ""),
|
||
m_stepDescriptions(stepDescriptions.begin(), stepDescriptions.end()) {
|
||
this->m_usingNamedStepTrackbar = true;
|
||
// Initialize selection with first step
|
||
if (!m_stepDescriptions.empty()) {
|
||
this->m_selection = m_stepDescriptions[0];
|
||
}
|
||
}
|
||
|
||
virtual ~NamedStepTrackBar() {}
|
||
|
||
virtual bool handleInput(u64 keysDown, u64 keysHeld, const HidTouchState &touchPos, HidAnalogStickState leftJoyStick, HidAnalogStickState rightJoyStick) override {
|
||
// Store previous value to update selection
|
||
const u8 prevProgress = this->getProgress();
|
||
|
||
// Call parent input handling
|
||
const bool result = StepTrackBar::handleInput(keysDown, keysHeld, touchPos, leftJoyStick, rightJoyStick);
|
||
|
||
// Update selection if progress changed
|
||
if (result && this->getProgress() != prevProgress) {
|
||
const u8 currentIndex = this->getProgress();
|
||
if (currentIndex < m_stepDescriptions.size()) {
|
||
this->m_selection = m_stepDescriptions[currentIndex];
|
||
}
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
virtual bool onTouch(TouchEvent event, s32 currX, s32 currY, s32 prevX, s32 prevY, s32 initialX, s32 initialY) override {
|
||
// Store previous value to update selection
|
||
const u8 prevProgress = this->getProgress();
|
||
|
||
// Call parent touch handling
|
||
const bool result = StepTrackBar::onTouch(event, currX, currY, prevX, prevY, initialX, initialY);
|
||
|
||
// Update selection if progress changed
|
||
if (result && this->getProgress() != prevProgress) {
|
||
const u8 currentIndex = this->getProgress();
|
||
if (currentIndex < m_stepDescriptions.size()) {
|
||
this->m_selection = m_stepDescriptions[currentIndex];
|
||
}
|
||
}
|
||
|
||
return result;
|
||
}
|
||
|
||
virtual void setProgress(u8 value) override {
|
||
StepTrackBar::setProgress(value);
|
||
|
||
// Update selection when progress is set programmatically
|
||
const u8 currentIndex = this->getProgress();
|
||
if (currentIndex < m_stepDescriptions.size()) {
|
||
this->m_selection = m_stepDescriptions[currentIndex];
|
||
}
|
||
}
|
||
|
||
virtual void draw(gfx::Renderer *renderer) override {
|
||
// TrackBar width excluding the handle areas
|
||
u16 trackBarWidth = this->getWidth() - 95;
|
||
|
||
// Base X and Y coordinates
|
||
u16 baseX = this->getX() + 59;
|
||
const u16 baseY = this->getY() + 44; // 50 - 3
|
||
|
||
s32 iconOffset = 0;
|
||
|
||
if (!m_useV2Style && m_icon[0] != '\0') {
|
||
s32 iconWidth = 23;//tsl::gfx::calculateStringWidth(m_icon, 23);
|
||
iconOffset = 14 + iconWidth;
|
||
baseX += iconOffset;
|
||
trackBarWidth -= iconOffset;
|
||
}
|
||
|
||
// Calculate the spacing between each step
|
||
const float stepSpacing = static_cast<float>(trackBarWidth) / (this->m_numSteps - 1);
|
||
|
||
// Calculate the halfway point index
|
||
const u8 halfNumSteps = (this->m_numSteps - 1) / 2;
|
||
// Draw step rectangles
|
||
u16 stepX;
|
||
for (u8 i = 0; i < this->m_numSteps; i++) {
|
||
stepX = baseX + std::round(i * stepSpacing);
|
||
|
||
// Subtract 1 from the X position for steps on the right side of the center
|
||
if (i > halfNumSteps) {
|
||
stepX -= 1;
|
||
}
|
||
// Adjust the last step to avoid overshooting
|
||
if (i == this->m_numSteps - 1) {
|
||
stepX = baseX + trackBarWidth -1;
|
||
}
|
||
|
||
renderer->drawRect(stepX, baseY, 1, 8, a(trackBarEmptyColor));
|
||
}
|
||
|
||
const u8 currentDescIndex = std::clamp(this->m_value / (100 / (this->m_numSteps - 1)), 0, this->m_numSteps - 1);
|
||
|
||
// Update selection for current index
|
||
if (currentDescIndex < m_stepDescriptions.size()) {
|
||
this->m_selection = m_stepDescriptions[currentDescIndex];
|
||
}
|
||
|
||
// Only draw the step description above the bar if not using V2 style (V2 style shows it on the right)
|
||
if (!m_useV2Style) {
|
||
const auto descWidth = renderer->getTextDimensions(this->m_stepDescriptions[currentDescIndex].c_str(), false, 15).first;
|
||
renderer->drawString(this->m_stepDescriptions[currentDescIndex].c_str(), false, ((baseX +1) + (trackBarWidth) / 2) - (descWidth / 2), this->getY() + 20 + 6, 15, (this->m_focused && ult::useSelectionValue) ? selectedValueTextColor : onTextColor);
|
||
}
|
||
|
||
// Draw the parent trackbar
|
||
StepTrackBar::draw(renderer);
|
||
}
|
||
|
||
protected:
|
||
std::vector<std::string> m_stepDescriptions;
|
||
|
||
};
|
||
|
||
|
||
|
||
/**
|
||
* @brief A customizable analog trackbar going from minValue to maxValue
|
||
*
|
||
*/
|
||
class TrackBarV2 : public Element {
|
||
public:
|
||
u64 lastUpdate_ns;
|
||
|
||
Color highlightColor = {0xf, 0xf, 0xf, 0xf};
|
||
float progress;
|
||
float counter = 0.0;
|
||
s32 x, y;
|
||
s32 amplitude;
|
||
u32 descWidth, descHeight;
|
||
|
||
// Add setScriptKeyListener function
|
||
void setScriptKeyListener(std::function<void()> listener) {
|
||
m_scriptKeyListener = std::move(listener);
|
||
}
|
||
|
||
|
||
// Ensure the order of initialization matches the order of declaration
|
||
TrackBarV2(std::string label, std::string packagePath = "", s16 minValue = 0, s16 maxValue = 100, std::string units = "",
|
||
std::function<bool(std::vector<std::vector<std::string>>&&, const std::string&, const std::string&)> executeCommands = nullptr,
|
||
std::function<std::vector<std::vector<std::string>>(const std::vector<std::vector<std::string>>&, const std::string&, size_t, const std::string&)> sourceReplacementFunc = nullptr,
|
||
std::vector<std::vector<std::string>> cmd = {}, const std::string& selCmd = "", bool usingStepTrackbar = false, bool usingNamedStepTrackbar = false, s16 numSteps = -1, bool unlockedTrackbar = false, bool executeOnEveryTick = false)
|
||
: m_label(label), m_packagePath(packagePath), m_minValue(minValue), m_maxValue(maxValue), m_units(units),
|
||
interpretAndExecuteCommands(executeCommands), getSourceReplacement(sourceReplacementFunc), commands(std::move(cmd)), selectedCommand(selCmd),
|
||
m_usingStepTrackbar(usingStepTrackbar), m_usingNamedStepTrackbar(usingNamedStepTrackbar), m_numSteps(numSteps), m_unlockedTrackbar(unlockedTrackbar), m_executeOnEveryTick(executeOnEveryTick) {
|
||
m_isItem = true;
|
||
|
||
if ((!usingStepTrackbar && !usingNamedStepTrackbar) || numSteps == -1) {
|
||
m_numSteps = (maxValue - minValue)+1;
|
||
}
|
||
|
||
bool loadedValue = false;
|
||
//if (!m_packagePath.empty()) {
|
||
// std::string initialIndex = ult::parseValueFromIniSection(m_packagePath + "config.ini", m_label, "index");
|
||
//
|
||
// if (!initialIndex.empty()) {
|
||
// m_index = static_cast<s16>(ult::stoi(initialIndex)); // convert initializedValue to s16
|
||
// }
|
||
// if (!m_usingNamedStepTrackbar) {
|
||
// std::string initialValue = ult::parseValueFromIniSection(m_packagePath + "config.ini", m_label, "value");
|
||
//
|
||
// if (!initialValue.empty()) {
|
||
// m_value = static_cast<s16>(ult::stoi(initialValue)); // convert initializedValue to s16
|
||
// loadedValue = true;
|
||
// }
|
||
// }
|
||
//}
|
||
|
||
if (!m_packagePath.empty()) {
|
||
// Load config INI once and extract both values
|
||
auto configIniData = ult::getParsedDataFromIniFile(m_packagePath + "config.ini");
|
||
auto sectionIt = configIniData.find(m_label);
|
||
|
||
if (sectionIt != configIniData.end()) {
|
||
// Get index value
|
||
auto indexIt = sectionIt->second.find("index");
|
||
if (indexIt != sectionIt->second.end() && !indexIt->second.empty()) {
|
||
m_index = static_cast<s16>(ult::stoi(indexIt->second)); // convert initializedValue to s16
|
||
}
|
||
|
||
// Get value if not using named step trackbar
|
||
if (!m_usingNamedStepTrackbar) {
|
||
auto valueIt = sectionIt->second.find("value");
|
||
if (valueIt != sectionIt->second.end() && !valueIt->second.empty()) {
|
||
m_value = static_cast<s16>(ult::stoi(valueIt->second)); // convert initializedValue to s16
|
||
loadedValue = true;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
if (m_index > m_numSteps -1) m_index = m_numSteps - 1;
|
||
else if (m_index < 0) m_index = 0;
|
||
|
||
if (!loadedValue)
|
||
m_value = minValue + m_index * (static_cast<float>(maxValue - minValue) / (m_numSteps - 1));
|
||
|
||
if (m_value > maxValue) m_value = maxValue;
|
||
else if (m_value < minValue) m_value = minValue;
|
||
|
||
lastUpdate_ns = armTicksToNs(armGetSystemTick());
|
||
}
|
||
|
||
virtual ~TrackBarV2() {}
|
||
|
||
virtual Element* requestFocus(Element *oldFocus, FocusDirection direction) {
|
||
return this;
|
||
}
|
||
|
||
|
||
inline void updateAndExecute(bool updateIni = true) {
|
||
if (m_packagePath.empty()) {
|
||
return;
|
||
}
|
||
|
||
const std::string indexStr = ult::to_string(m_index);
|
||
const std::string valueStr = m_usingNamedStepTrackbar ? m_selection : ult::to_string(m_value);
|
||
|
||
if (updateIni) {
|
||
const std::string configPath = m_packagePath + "config.ini";
|
||
ult::setIniFileValue(configPath, m_label, "index", indexStr);
|
||
ult::setIniFileValue(configPath, m_label, "value", valueStr);
|
||
}
|
||
bool success = false;
|
||
|
||
// Placeholder replacement - cache lengths once
|
||
static const std::string valuePlaceholder = "{value}";
|
||
static const std::string indexPlaceholder = "{index}";
|
||
static const size_t valuePlaceholderLen = valuePlaceholder.length();
|
||
static const size_t indexPlaceholderLen = indexPlaceholder.length();
|
||
const size_t valueStrLen = valueStr.length();
|
||
const size_t indexStrLen = indexStr.length();
|
||
|
||
size_t tryCount = 0;
|
||
while (!success) {
|
||
if (interpretAndExecuteCommands) {
|
||
if (tryCount > 3)
|
||
break;
|
||
auto modifiedCmds = getSourceReplacement(commands, valueStr, m_index, m_packagePath);
|
||
|
||
for (auto& cmd : modifiedCmds) {
|
||
for (auto& arg : cmd) {
|
||
// Replace value placeholder
|
||
for (size_t pos = 0; (pos = arg.find(valuePlaceholder, pos)) != std::string::npos; pos += valueStrLen) {
|
||
arg.replace(pos, valuePlaceholderLen, valueStr);
|
||
}
|
||
|
||
// Replace index placeholder if needed
|
||
if (m_usingNamedStepTrackbar) {
|
||
for (size_t pos = 0; (pos = arg.find(indexPlaceholder, pos)) != std::string::npos; pos += indexStrLen) {
|
||
arg.replace(pos, indexPlaceholderLen, indexStr);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
success = interpretAndExecuteCommands(std::move(modifiedCmds), m_packagePath, selectedCommand);
|
||
ult::resetPercentages();
|
||
|
||
if (success)
|
||
break;
|
||
tryCount++;
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
virtual inline bool handleInput(u64 keysDown, u64 keysHeld, const HidTouchState &touchPos, HidAnalogStickState leftJoyStick, HidAnalogStickState rightJoyStick) override {
|
||
static bool holding = false;
|
||
static u64 holdStartTime_ns;
|
||
static u64 prevKeysHeld = 0;
|
||
|
||
const u64 keysReleased = prevKeysHeld & ~keysHeld;
|
||
prevKeysHeld = keysHeld;
|
||
|
||
const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
const u64 elapsed_ns = currentTime_ns - lastUpdate_ns;
|
||
|
||
static bool wasLastHeld = false;
|
||
|
||
m_keyRHeld = (keysHeld & KEY_R) != 0;
|
||
|
||
if ((keysHeld & KEY_R)) {
|
||
//auto currentFocus = currentGui->getFocusedElement();
|
||
if (keysDown & KEY_UP && !(keysHeld & ~KEY_UP & ~KEY_R & ALL_KEYS_MASK))
|
||
this->shakeHighlight(FocusDirection::Up);
|
||
else if (keysDown & KEY_DOWN && !(keysHeld & ~KEY_DOWN & ~KEY_R & ALL_KEYS_MASK))
|
||
this->shakeHighlight(FocusDirection::Down);
|
||
else if (keysDown & KEY_LEFT && !(keysHeld & ~KEY_LEFT & ~KEY_R & ALL_KEYS_MASK)){
|
||
this->shakeHighlight(FocusDirection::Left);
|
||
}
|
||
else if (keysDown & KEY_RIGHT && !(keysHeld & ~KEY_RIGHT & ~KEY_R & ALL_KEYS_MASK)) {
|
||
this->shakeHighlight(FocusDirection::Right);
|
||
}
|
||
|
||
return true;
|
||
}
|
||
|
||
// Check if KEY_A is pressed to toggle ult::allowSlide
|
||
if ((keysDown & KEY_A) && !(keysHeld & ~KEY_A & ALL_KEYS_MASK)) {
|
||
if (!m_unlockedTrackbar) {
|
||
//ult::allowSlide = !ult::allowSlide;
|
||
//ult::allowSlide.fetch_xor(true, std::memory_order_acq_rel);
|
||
ult::atomicToggle(ult::allowSlide);
|
||
holding = false; // Reset holding state when KEY_A is pressed
|
||
}
|
||
if (m_unlockedTrackbar || (!m_unlockedTrackbar && !ult::allowSlide.load(std::memory_order_acquire))) {
|
||
updateAndExecute();
|
||
triggerClick = true;
|
||
}
|
||
return true;
|
||
}
|
||
|
||
// Handle SCRIPT_KEY press
|
||
if ((keysDown & SCRIPT_KEY) && !(keysHeld & ~SCRIPT_KEY & ALL_KEYS_MASK)) {
|
||
if (m_scriptKeyListener) {
|
||
m_scriptKeyListener();
|
||
}
|
||
return true;
|
||
}
|
||
|
||
|
||
// Allow sliding only if KEY_A has been pressed
|
||
if (ult::allowSlide.load(std::memory_order_acquire) || m_unlockedTrackbar) {
|
||
|
||
if (((keysReleased & KEY_LEFT) || (keysReleased & KEY_RIGHT)) ||
|
||
(wasLastHeld && !((keysHeld & KEY_LEFT) || (keysHeld & KEY_RIGHT)))) {
|
||
|
||
wasLastHeld = false;
|
||
updateAndExecute();
|
||
lastUpdate_ns = armTicksToNs(armGetSystemTick());
|
||
|
||
holding = false;
|
||
return true;
|
||
}
|
||
|
||
if (keysDown & KEY_LEFT && keysDown & KEY_RIGHT)
|
||
return true;
|
||
|
||
if (keysDown & KEY_LEFT) {
|
||
if (this->m_value > m_minValue) {
|
||
this->m_index--;
|
||
this->m_value--;
|
||
this->m_valueChangedListener(this->m_value);
|
||
updateAndExecute(false);
|
||
lastUpdate_ns = armTicksToNs(armGetSystemTick());
|
||
return true;
|
||
}
|
||
}
|
||
|
||
if (keysDown & KEY_RIGHT) {
|
||
if (this->m_value < m_maxValue) {
|
||
this->m_index++;
|
||
this->m_value++;
|
||
this->m_valueChangedListener(this->m_value);
|
||
updateAndExecute(false);
|
||
lastUpdate_ns = armTicksToNs(armGetSystemTick());
|
||
return true;
|
||
}
|
||
}
|
||
|
||
|
||
|
||
if (keysHeld & KEY_LEFT && keysHeld & KEY_RIGHT)
|
||
return true;
|
||
|
||
// Check if the button is being held down
|
||
if (((keysHeld & KEY_LEFT) || (keysHeld & KEY_RIGHT))) {
|
||
|
||
if (!holding) {
|
||
holding = true;
|
||
holdStartTime_ns = armTicksToNs(armGetSystemTick());
|
||
}
|
||
|
||
const u64 holdDuration_ns = currentTime_ns - holdStartTime_ns;
|
||
|
||
// Define the duration boundaries in nanoseconds
|
||
static constexpr u64 initialInterval_ns = 67000000ULL; // 67ms in nanoseconds
|
||
static constexpr u64 shortInterval_ns = 10000000ULL; // 10ms in nanoseconds
|
||
static constexpr u64 transitionPoint_ns = 2000000000ULL; // 2000ms in nanoseconds
|
||
|
||
// Calculate transition factor (t) from 0 to 1 based on how far we are from the transition point
|
||
const float t = std::min(1.0f, static_cast<float>(holdDuration_ns) / static_cast<float>(transitionPoint_ns));
|
||
|
||
// Smooth transition between intervals
|
||
const u64 currentInterval_ns = static_cast<u64>((initialInterval_ns - shortInterval_ns) * (1.0f - t) + shortInterval_ns);
|
||
|
||
if (elapsed_ns >= currentInterval_ns) {
|
||
if (keysHeld & KEY_LEFT) {
|
||
if (this->m_value > m_minValue) {
|
||
this->m_index--;
|
||
this->m_value--;
|
||
this->m_valueChangedListener(this->m_value);
|
||
if (m_executeOnEveryTick) {
|
||
updateAndExecute(false);
|
||
}
|
||
lastUpdate_ns = armTicksToNs(armGetSystemTick());
|
||
wasLastHeld = true;
|
||
return true;
|
||
}
|
||
}
|
||
|
||
if (keysHeld & KEY_RIGHT) {
|
||
if (this->m_value < m_maxValue) {
|
||
this->m_index++;
|
||
this->m_value++;
|
||
this->m_valueChangedListener(this->m_value);
|
||
if (m_executeOnEveryTick) {
|
||
updateAndExecute(false);
|
||
}
|
||
lastUpdate_ns = armTicksToNs(armGetSystemTick());
|
||
wasLastHeld = true;
|
||
return true;
|
||
}
|
||
}
|
||
}
|
||
} else {
|
||
holding = false; // Reset holding state if no relevant key is held
|
||
}
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
|
||
virtual bool onTouch(TouchEvent event, s32 currX, s32 currY, s32 prevX, s32 prevY, s32 initialX, s32 initialY) override {
|
||
const u16 trackBarWidth = this->getWidth() - 95;
|
||
const u16 handlePos = (trackBarWidth * (this->m_value - m_minValue)) / (m_maxValue - m_minValue);
|
||
const s32 circleCenterX = this->getX() + 59 + handlePos;
|
||
const s32 circleCenterY = this->getY() + 40 + 16 - 1;
|
||
static constexpr s32 circleRadius = 16;
|
||
|
||
const bool touchInCircle = (std::abs(initialX - circleCenterX) <= circleRadius) && (std::abs(initialY - circleCenterY) <= circleRadius);
|
||
|
||
if (!m_unlockedTrackbar && !ult::allowSlide.load(std::memory_order_acquire))
|
||
return false;
|
||
|
||
if ((touchInCircle || touchInSliderBounds)) {
|
||
|
||
touchInSliderBounds = true;
|
||
|
||
//s16 newIndex = static_cast<s16>((currX - (this->getX() + 59)) / static_cast<float>(this->getWidth() - 95) * (m_numSteps - 1));
|
||
|
||
// Clamp the index within valid range
|
||
const s16 newIndex = std::max(static_cast<s16>(0), std::min(static_cast<s16>((currX - (this->getX() + 59)) / static_cast<float>(this->getWidth() - 95) * (m_numSteps - 1)), static_cast<s16>(m_numSteps - 1)));
|
||
|
||
const s16 newValue = m_minValue + newIndex * (static_cast<float>(m_maxValue - m_minValue) / (m_numSteps - 1));
|
||
|
||
if (newValue != this->m_value || newIndex != this->m_index) {
|
||
this->m_value = newValue;
|
||
this->m_index = newIndex;
|
||
this->m_valueChangedListener(this->getProgress());
|
||
if (m_executeOnEveryTick) {
|
||
updateAndExecute(false);
|
||
}
|
||
} else {
|
||
if (event == TouchEvent::Release) {
|
||
updateAndExecute();
|
||
if (event == TouchEvent::Release)
|
||
touchInSliderBounds = false;
|
||
}
|
||
}
|
||
|
||
return true;
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
// Define drawBar function outside the draw method
|
||
void drawBar(gfx::Renderer *renderer, s32 x, s32 y, u16 width, Color& color, bool isRounded = true) {
|
||
if (isRounded) {
|
||
renderer->drawUniformRoundedRect(x, y, width, 7, a(color));
|
||
} else {
|
||
renderer->drawRect(x, y, width, 7, a(color));
|
||
}
|
||
}
|
||
|
||
virtual void draw(gfx::Renderer *renderer) override {
|
||
static float lastBottomBound;
|
||
const u16 handlePos = (this->getWidth() - 95) * (this->m_value - m_minValue) / (m_maxValue - m_minValue);
|
||
const s32 xPos = this->getX() + 59;
|
||
const s32 yPos = this->getY() + 40 + 16 - 1;
|
||
const s32 width = this->getWidth() - 95;
|
||
|
||
|
||
// Draw track bar background
|
||
drawBar(renderer, xPos, yPos-3, width, trackBarEmptyColor, !m_usingNamedStepTrackbar);
|
||
|
||
|
||
// Determine visual appearance: look locked if KEY_R is held and trackbar is unlocked
|
||
const bool shouldAppearLocked = m_unlockedTrackbar && m_keyRHeld;
|
||
const bool visuallyUnlocked = (m_unlockedTrackbar && !m_keyRHeld) || touchInSliderBounds;
|
||
|
||
if (!this->m_focused) {
|
||
drawBar(renderer, xPos, yPos-3, handlePos, trackBarFullColor, !m_usingNamedStepTrackbar);
|
||
renderer->drawCircle(xPos + handlePos, yPos, 16, true, a(trackBarSliderBorderColor));
|
||
renderer->drawCircle(xPos + handlePos, yPos, 13, true, a(visuallyUnlocked ? trackBarSliderMalleableColor : trackBarSliderColor));
|
||
} else {
|
||
touchInSliderBounds = false;
|
||
if (m_unlockedTrackbar != ult::unlockedSlide.load(std::memory_order_acquire))
|
||
ult::unlockedSlide.store(m_unlockedTrackbar, std::memory_order_release);
|
||
drawBar(renderer, xPos, yPos-3, handlePos, trackBarFullColor, !m_usingNamedStepTrackbar);
|
||
renderer->drawCircle(xPos + x + handlePos, yPos +y, 16, true, a(highlightColor));
|
||
// For focused state, check both allowSlide and visual unlock state
|
||
const bool focusedVisuallyUnlocked = (ult::allowSlide.load(std::memory_order_acquire) || m_unlockedTrackbar) && !shouldAppearLocked;
|
||
renderer->drawCircle(xPos + x + handlePos, yPos +y, 12, true, a(focusedVisuallyUnlocked ? trackBarSliderMalleableColor : trackBarSliderColor));
|
||
}
|
||
|
||
std::string labelPart = this->m_label;
|
||
ult::removeTag(labelPart);
|
||
|
||
static std::string valuePart;
|
||
if (!m_usingNamedStepTrackbar)
|
||
valuePart = (this->m_units == "%" || this->m_units == "°C" || this->m_units == "°F") ? ult::to_string(this->m_value) + this->m_units : ult::to_string(this->m_value) + (this->m_units.empty() ? "" : " ") + this->m_units;
|
||
else
|
||
valuePart = this->m_selection;
|
||
|
||
const auto valueWidth = renderer->getTextDimensions(valuePart, false, 16).first;
|
||
|
||
renderer->drawString(labelPart, false, xPos, this->getY() + 14 + 16, 16, ((!this->m_focused || !ult::useSelectionText) ? (defaultTextColor) : (selectedTextColor)));
|
||
//renderer->drawString(valuePart, false, this->getWidth() -17 - valueWidth, this->getY() + 14 + 16, 16, (onTextColor));
|
||
renderer->drawString(valuePart, false, this->getWidth() -17 - valueWidth, this->getY() + 14 + 16, 16,
|
||
(this->m_focused && ult::useSelectionValue) ? selectedValueTextColor : onTextColor);
|
||
|
||
|
||
if (lastBottomBound != this->getTopBound())
|
||
renderer->drawRect(this->getX() + 4+20-1, this->getTopBound(), this->getWidth() + 6 + 10+20 +4, 1, a(separatorColor));
|
||
renderer->drawRect(this->getX() + 4+20-1, this->getBottomBound(), this->getWidth() + 6 + 10+20 +4, 1, a(separatorColor));
|
||
lastBottomBound = this->getBottomBound();
|
||
}
|
||
|
||
|
||
virtual void layout(u16 parentX, u16 parentY, u16 parentWidth, u16 parentHeight) override {
|
||
this->setBoundaries(this->getX() - 16 , this->getY(), this->getWidth()+20+4, tsl::style::TrackBarDefaultHeight );
|
||
}
|
||
|
||
virtual void drawFocusBackground(gfx::Renderer *renderer) {
|
||
// No background drawn here in HOS
|
||
}
|
||
|
||
virtual void drawHighlight(gfx::Renderer *renderer) override {
|
||
|
||
// Calculate progress using ARM ticks instead of chrono
|
||
const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
const double timeInSeconds = static_cast<double>(currentTime_ns) / 1000000000.0;
|
||
progress = ((std::cos(2.0 * ult::_M_PI * std::fmod(timeInSeconds, 1.0) - ult::_M_PI / 2) + 1.0) / 2.0);
|
||
|
||
static u64 clickStartTime_ns;
|
||
static bool clickActive = false;
|
||
|
||
Color clickColor1 = highlightColor1;
|
||
Color clickColor2 = clickColor;
|
||
|
||
// Activate `clickStartTime_ns` when `triggerClick` is set to true
|
||
if (triggerClick && !clickActive) {
|
||
clickStartTime_ns = currentTime_ns;
|
||
clickActive = true;
|
||
// Within the cycle, perform the highlight effect
|
||
if (progress >= 0.5) {
|
||
clickColor1 = clickColor;
|
||
clickColor2 = highlightColor2;
|
||
}
|
||
}
|
||
static auto lastLabel = m_label;
|
||
|
||
if (lastLabel != m_label) {
|
||
clickActive = false;
|
||
triggerClick = false;
|
||
}
|
||
lastLabel = m_label;
|
||
|
||
if (clickActive) {
|
||
const u64 elapsedTime_ns = currentTime_ns - clickStartTime_ns;
|
||
if (elapsedTime_ns < 500000000ULL) { // 500ms in nanoseconds
|
||
// High precision floating point color interpolation for click colors
|
||
highlightColor = {
|
||
static_cast<u8>((clickColor1.r - clickColor2.r) * progress + clickColor2.r + 0.5),
|
||
static_cast<u8>((clickColor1.g - clickColor2.g) * progress + clickColor2.g + 0.5),
|
||
static_cast<u8>((clickColor1.b - clickColor2.b) * progress + clickColor2.b + 0.5),
|
||
0xF
|
||
};
|
||
} else {
|
||
// End the effect after one cycle
|
||
clickActive = false;
|
||
triggerClick = false;
|
||
}
|
||
|
||
} else {
|
||
// Determine if should appear locked: unlocked trackbar with KEY_R held
|
||
const bool shouldAppearLocked = m_unlockedTrackbar && m_keyRHeld;
|
||
|
||
if ((ult::allowSlide.load(std::memory_order_acquire) || m_unlockedTrackbar) && !shouldAppearLocked) {
|
||
// High precision floating point color interpolation for unlocked trackbar
|
||
highlightColor = {
|
||
static_cast<u8>((highlightColor1.r - highlightColor2.r) * progress + highlightColor2.r + 0.5),
|
||
static_cast<u8>((highlightColor1.g - highlightColor2.g) * progress + highlightColor2.g + 0.5),
|
||
static_cast<u8>((highlightColor1.b - highlightColor2.b) * progress + highlightColor2.b + 0.5),
|
||
0xF
|
||
};
|
||
} else {
|
||
// High precision floating point color interpolation for locked trackbar (or visually locked)
|
||
highlightColor = {
|
||
static_cast<u8>((highlightColor3.r - highlightColor4.r) * progress + highlightColor4.r + 0.5),
|
||
static_cast<u8>((highlightColor3.g - highlightColor4.g) * progress + highlightColor4.g + 0.5),
|
||
static_cast<u8>((highlightColor3.b - highlightColor4.b) * progress + highlightColor4.b + 0.5),
|
||
0xF
|
||
};
|
||
}
|
||
}
|
||
|
||
x = 0;
|
||
y = 0;
|
||
|
||
if (this->m_highlightShaking) {
|
||
//const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
t_ns = currentTime_ns - this->m_highlightShakingStartTime;
|
||
if (t_ns >= 100000000ULL) // 100ms in nanoseconds
|
||
this->m_highlightShaking = false;
|
||
else {
|
||
amplitude = std::rand() % 5 + 5;
|
||
|
||
switch (this->m_highlightShakingDirection) {
|
||
case FocusDirection::Up:
|
||
y -= shakeAnimation(t_ns, amplitude);
|
||
break;
|
||
case FocusDirection::Down:
|
||
y += shakeAnimation(t_ns, amplitude);
|
||
break;
|
||
case FocusDirection::Left:
|
||
x -= shakeAnimation(t_ns, amplitude);
|
||
break;
|
||
case FocusDirection::Right:
|
||
x += shakeAnimation(t_ns, amplitude);
|
||
break;
|
||
default:
|
||
break;
|
||
}
|
||
|
||
x = std::clamp(x, -amplitude, amplitude);
|
||
y = std::clamp(y, -amplitude, amplitude);
|
||
}
|
||
}
|
||
|
||
if (ult::useSelectionBG) {
|
||
if (ult::expandedMemory)
|
||
renderer->drawRectMultiThreaded(this->getX() + x +19, this->getY() + y, this->getWidth()-11-4, this->getHeight(), aWithOpacity(selectionBGColor)); // CUSTOM MODIFICATION
|
||
else
|
||
renderer->drawRect(this->getX() + x +19, this->getY() + y, this->getWidth()-11-4, this->getHeight(), aWithOpacity(selectionBGColor)); // CUSTOM MODIFICATION
|
||
|
||
//renderer->drawRect(this->getX() + x +19, this->getY() + y, this->getWidth()-11-4, this->getHeight(), a(selectionBGColor)); // CUSTOM MODIFICATION
|
||
}
|
||
|
||
renderer->drawBorderedRoundedRect(this->getX() + x +19, this->getY() + y, this->getWidth()-11, this->getHeight(), 5, 5, a(highlightColor));
|
||
|
||
ult::onTrackBar.store(true, std::memory_order_release);
|
||
|
||
if (clickActive) {
|
||
const u64 elapsedTime_ns = currentTime_ns - clickStartTime_ns;
|
||
|
||
// Handle click animation progress and animColor rendering
|
||
auto clickAnimationProgress = tsl::style::ListItemHighlightLength * (1.0f - (static_cast<float>(elapsedTime_ns) / 500000000.0f)); // 500ms in nanoseconds
|
||
|
||
// Ensure progress does not go below 0
|
||
if (clickAnimationProgress < 0.0f) {
|
||
clickAnimationProgress = 0.0f;
|
||
}
|
||
|
||
if (clickAnimationProgress > 0.0f) {
|
||
const u8 saturation = tsl::style::ListItemHighlightSaturation * (float(clickAnimationProgress) / float(tsl::style::ListItemHighlightLength));
|
||
|
||
Color animColor = {0xF, 0xF, 0xF, 0xF};
|
||
if (invertBGClickColor) {
|
||
animColor.r = 15 - saturation;
|
||
animColor.g = 15 - saturation;
|
||
animColor.b = 15 - saturation;
|
||
} else {
|
||
animColor.r = saturation;
|
||
animColor.g = saturation;
|
||
animColor.b = saturation;
|
||
}
|
||
animColor.a = selectionBGColor.a;
|
||
renderer->drawRect(this->getX() +22, this->getY(), this->getWidth() -22, this->getHeight(), aWithOpacity(animColor));
|
||
}
|
||
}
|
||
}
|
||
|
||
virtual inline u8 getProgress() {
|
||
return this->m_value;
|
||
}
|
||
|
||
virtual void setProgress(u8 value) {
|
||
this->m_value = value;
|
||
}
|
||
|
||
void setValueChangedListener(std::function<void(u8)> valueChangedListener) {
|
||
this->m_valueChangedListener = valueChangedListener;
|
||
}
|
||
|
||
protected:
|
||
std::string m_label;
|
||
std::string m_packagePath;
|
||
std::string m_selection;
|
||
s16 m_value = 0;
|
||
s16 m_minValue;
|
||
s16 m_maxValue;
|
||
std::string m_units;
|
||
bool m_interactionLocked = false;
|
||
bool m_keyRHeld = false; // Track KEY_R state for visual appearance
|
||
|
||
std::function<void(u8)> m_valueChangedListener = [](u8) {};
|
||
|
||
// New member variables to store the function and its parameters
|
||
std::function<bool(std::vector<std::vector<std::string>>&&, const std::string&, const std::string&)> interpretAndExecuteCommands;
|
||
std::function<std::vector<std::vector<std::string>>(const std::vector<std::vector<std::string>>&, const std::string&, size_t, const std::string&)> getSourceReplacement;
|
||
std::vector<std::vector<std::string>> commands;
|
||
std::string selectedCommand;
|
||
|
||
bool m_usingStepTrackbar = false;
|
||
bool m_usingNamedStepTrackbar = false;
|
||
s16 m_numSteps = 2;
|
||
s16 m_index = 0;
|
||
bool m_unlockedTrackbar = false;
|
||
bool m_executeOnEveryTick = false;
|
||
bool touchInSliderBounds = false;
|
||
bool triggerClick = false;
|
||
std::function<void()> m_scriptKeyListener;
|
||
|
||
};
|
||
|
||
|
||
/**
|
||
* @brief A customizable analog trackbar going from 0% to 100% but using discrete steps (Like the volume slider)
|
||
*
|
||
*/
|
||
class StepTrackBarV2 : public TrackBarV2 {
|
||
public:
|
||
|
||
/**
|
||
* @brief Constructor
|
||
*
|
||
* @param icon Icon shown next to the track bar
|
||
* @param numSteps Number of steps the track bar has
|
||
*/
|
||
StepTrackBarV2(std::string label, std::string packagePath, size_t numSteps, s16 minValue, s16 maxValue, std::string units,
|
||
std::function<bool(std::vector<std::vector<std::string>>&&, const std::string&, const std::string&)> executeCommands = nullptr,
|
||
std::function<std::vector<std::vector<std::string>>(const std::vector<std::vector<std::string>>&, const std::string&, size_t, const std::string&)> sourceReplacementFunc = nullptr,
|
||
std::vector<std::vector<std::string>> cmd = {}, const std::string& selCmd = "", bool usingNamedStepTrackbar = false, bool unlockedTrackbar = false, bool executeOnEveryTick = false)
|
||
: TrackBarV2(label, packagePath, minValue, maxValue, units, executeCommands, sourceReplacementFunc, cmd, selCmd, !usingNamedStepTrackbar, usingNamedStepTrackbar, numSteps, unlockedTrackbar, executeOnEveryTick) {}
|
||
|
||
virtual ~StepTrackBarV2() {}
|
||
|
||
virtual inline bool handleInput(u64 keysDown, u64 keysHeld, const HidTouchState &touchPos, HidAnalogStickState leftJoyStick, HidAnalogStickState rightJoyStick) override {
|
||
static u32 tick = 0;
|
||
static bool holding = false;
|
||
static u64 prevKeysHeld = 0;
|
||
const u64 keysReleased = prevKeysHeld & ~keysHeld;
|
||
prevKeysHeld = keysHeld;
|
||
|
||
static bool wasLastHeld = false;
|
||
|
||
// ADD THIS LINE: Update KEY_R state for visual appearance
|
||
m_keyRHeld = (keysHeld & KEY_R) != 0;
|
||
|
||
if ((keysHeld & KEY_R)) {
|
||
//auto currentFocus = currentGui->getFocusedElement();
|
||
if (keysDown & KEY_UP && !(keysHeld & ~KEY_UP & ~KEY_R & ALL_KEYS_MASK))
|
||
this->shakeHighlight(FocusDirection::Up);
|
||
else if (keysDown & KEY_DOWN && !(keysHeld & ~KEY_DOWN & ~KEY_R & ALL_KEYS_MASK))
|
||
this->shakeHighlight(FocusDirection::Down);
|
||
else if (keysDown & KEY_LEFT && !(keysHeld & ~KEY_LEFT & ~KEY_R & ALL_KEYS_MASK)){
|
||
this->shakeHighlight(FocusDirection::Left);
|
||
}
|
||
else if (keysDown & KEY_RIGHT && !(keysHeld & ~KEY_RIGHT & ~KEY_R & ALL_KEYS_MASK)) {
|
||
this->shakeHighlight(FocusDirection::Right);
|
||
}
|
||
return true;
|
||
}
|
||
|
||
// Check if KEY_A is pressed to toggle ult::allowSlide
|
||
if ((keysDown & KEY_A) && !(keysHeld & ~KEY_A & ALL_KEYS_MASK)) {
|
||
if (!m_unlockedTrackbar) {
|
||
ult::atomicToggle(ult::allowSlide);
|
||
holding = false; // Reset holding state when KEY_A is pressed
|
||
}
|
||
if (m_unlockedTrackbar || (!m_unlockedTrackbar && !ult::allowSlide.load(std::memory_order_acquire))) {
|
||
updateAndExecute();
|
||
triggerClick = true;
|
||
}
|
||
return true;
|
||
}
|
||
|
||
// Handle SCRIPT_KEY press
|
||
if ((keysDown & SCRIPT_KEY) && !(keysHeld & ~SCRIPT_KEY & ALL_KEYS_MASK)) {
|
||
if (m_scriptKeyListener) {
|
||
m_scriptKeyListener();
|
||
}
|
||
return true;
|
||
}
|
||
|
||
if (ult::allowSlide.load(std::memory_order_acquire) || m_unlockedTrackbar) {
|
||
if (((keysReleased & KEY_LEFT) || (keysReleased & KEY_RIGHT)) ||
|
||
(wasLastHeld && !(keysHeld & (KEY_LEFT | KEY_RIGHT)))) {
|
||
updateAndExecute();
|
||
holding = false;
|
||
wasLastHeld = false;
|
||
tick = 0;
|
||
return true;
|
||
}
|
||
|
||
if (keysHeld & KEY_LEFT && keysHeld & KEY_RIGHT) {
|
||
tick = 0;
|
||
return true;
|
||
}
|
||
|
||
if (keysHeld & (KEY_LEFT | KEY_RIGHT)) {
|
||
if (!holding) {
|
||
holding = true;
|
||
tick = 0;
|
||
}
|
||
|
||
if ((tick == 0 || tick > 20) && (tick % 3) == 0) {
|
||
const float stepSize = static_cast<float>(m_maxValue - m_minValue) / (this->m_numSteps - 1);
|
||
if (keysHeld & KEY_LEFT && this->m_index > 0) {
|
||
this->m_index--;
|
||
this->m_value = static_cast<s16>(std::round(m_minValue + m_index * stepSize));
|
||
} else if (keysHeld & KEY_RIGHT && this->m_index < this->m_numSteps-1) {
|
||
this->m_index++;
|
||
this->m_value = static_cast<s16>(std::round(m_minValue + m_index * stepSize));
|
||
} else {
|
||
return false;
|
||
}
|
||
this->m_valueChangedListener(this->getProgress());
|
||
if (m_executeOnEveryTick)
|
||
updateAndExecute(false);
|
||
wasLastHeld = true;
|
||
}
|
||
tick++;
|
||
return true;
|
||
} else {
|
||
holding = false;
|
||
tick = 0;
|
||
}
|
||
}
|
||
|
||
return false;
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Gets the current value of the trackbar
|
||
*
|
||
* @return State
|
||
*/
|
||
virtual inline u8 getProgress() override {
|
||
return this->m_value / (100 / (this->m_numSteps - 1));
|
||
}
|
||
|
||
/**
|
||
* @brief Sets the current state of the toggle. Updates the Value
|
||
*
|
||
* @param state State
|
||
*/
|
||
virtual void setProgress(u8 value) override {
|
||
value = std::min(value, u8(this->m_numSteps - 1));
|
||
this->m_value = value * (100 / (this->m_numSteps - 1));
|
||
}
|
||
|
||
//protected:
|
||
//u8 m_numSteps = 1;
|
||
|
||
};
|
||
|
||
|
||
/**
|
||
* @brief A customizable trackbar with multiple discrete steps with specific names. Name gets displayed above the bar
|
||
*
|
||
*/
|
||
class NamedStepTrackBarV2 : public StepTrackBarV2 {
|
||
public:
|
||
u16 trackBarWidth, stepWidth, currentDescIndex;
|
||
u32 descWidth, descHeight;
|
||
|
||
/**
|
||
* @brief Constructor
|
||
*
|
||
* @param icon Icon shown next to the track bar
|
||
* @param stepDescriptions Step names displayed above the track bar
|
||
*/
|
||
NamedStepTrackBarV2(std::string label, std::string packagePath, std::vector<std::string>& stepDescriptions,
|
||
std::function<bool(std::vector<std::vector<std::string>>&&, const std::string&, const std::string&)> executeCommands = nullptr,
|
||
std::function<std::vector<std::vector<std::string>>(const std::vector<std::vector<std::string>>&, const std::string&, size_t, const std::string&)> sourceReplacementFunc = nullptr,
|
||
std::vector<std::vector<std::string>> cmd = {}, const std::string& selCmd = "", bool unlockedTrackbar = false, bool executeOnEveryTick = false)
|
||
: StepTrackBarV2(label, packagePath, stepDescriptions.size(), 0, (stepDescriptions.size()-1), "", executeCommands, sourceReplacementFunc, cmd, selCmd, true, unlockedTrackbar, executeOnEveryTick), m_stepDescriptions(stepDescriptions) {
|
||
//usingNamedStepTrackbar = true;
|
||
//logMessage("on initialization");
|
||
|
||
// Initialize the selection with the current index
|
||
if (!m_stepDescriptions.empty() && m_index >= 0 && m_index < static_cast<s16>(m_stepDescriptions.size())) {
|
||
this->m_selection = m_stepDescriptions[m_index];
|
||
currentDescIndex = m_index;
|
||
}
|
||
}
|
||
|
||
virtual ~NamedStepTrackBarV2() {}
|
||
|
||
virtual void draw(gfx::Renderer *renderer) override {
|
||
// Cache frequently used values
|
||
const u16 trackBarWidth = this->getWidth() - 95;
|
||
const u16 baseX = this->getX() + 59;
|
||
const u16 baseY = this->getY() + 44; // 50 - 3
|
||
const u8 numSteps = this->m_numSteps;
|
||
const u8 halfNumSteps = (numSteps - 1) / 2;
|
||
const u16 lastStepX = baseX + trackBarWidth - 1;
|
||
|
||
// Pre-calculate step spacing
|
||
const float stepSpacing = static_cast<float>(trackBarWidth) / (numSteps - 1);
|
||
|
||
// Cache color for multiple drawRect calls
|
||
const auto stepColor = a(trackBarEmptyColor);
|
||
|
||
// Draw step rectangles - optimized loop
|
||
u16 stepX;
|
||
for (u8 i = 0; i < numSteps; i++) {
|
||
|
||
if (i == numSteps - 1) {
|
||
// Last step - avoid overshooting
|
||
stepX = lastStepX;
|
||
} else {
|
||
stepX = baseX + static_cast<u16>(std::round(i * stepSpacing));
|
||
// Adjust for steps on right side of center
|
||
if (i > halfNumSteps) {
|
||
stepX -= 1;
|
||
}
|
||
}
|
||
|
||
renderer->drawRect(stepX, baseY, 1, 8, stepColor);
|
||
}
|
||
|
||
// Update selection (only if index changed - optional optimization)
|
||
if (currentDescIndex != this->m_index) {
|
||
currentDescIndex = this->m_index;
|
||
this->m_selection = this->m_stepDescriptions[currentDescIndex];
|
||
}
|
||
|
||
// Draw the parent trackbar
|
||
StepTrackBarV2::draw(renderer);
|
||
}
|
||
|
||
|
||
protected:
|
||
std::vector<std::string> m_stepDescriptions;
|
||
|
||
};
|
||
|
||
}
|
||
|
||
// Global state and event system
|
||
static inline Event notificationEvent;
|
||
static inline std::mutex notificationJsonMutex;
|
||
static inline std::atomic<uint32_t> notificationGeneration{0};
|
||
|
||
class NotificationPrompt {
|
||
public:
|
||
NotificationPrompt()
|
||
: enabled_(true),
|
||
is_active_(false),
|
||
//pending_event_fire_(false),
|
||
generation_(notificationGeneration.load(std::memory_order_acquire))
|
||
{}
|
||
|
||
~NotificationPrompt() {
|
||
shutdown(); // safe cleanup
|
||
}
|
||
|
||
enum class PromptState {
|
||
Inactive,
|
||
SlidingIn,
|
||
Visible,
|
||
SlidingOut
|
||
};
|
||
|
||
struct NotificationData {
|
||
std::string text;
|
||
std::string fileName;
|
||
size_t fontSize = 28;
|
||
s32 promptWidth = 448;
|
||
s32 promptHeight = 88;
|
||
u32 durationMs = 2500;
|
||
u32 priority = 20;
|
||
u64 arrivalNs = 0;
|
||
|
||
NotificationData() = default;
|
||
|
||
NotificationData(const std::string& t, const std::string& f = "",
|
||
size_t fs = 28, s32 w = 448, s32 h = 88,
|
||
u32 dur = 2500, u32 prio = 20)
|
||
: text(t), fileName(f), fontSize(fs), promptWidth(w), promptHeight(h),
|
||
durationMs(dur), priority(prio), arrivalNs(0) {}
|
||
};
|
||
|
||
struct NotificationCompare {
|
||
bool operator()(const NotificationData& a, const NotificationData& b) const {
|
||
if (a.priority == b.priority) {
|
||
return a.arrivalNs > b.arrivalNs; // FIFO
|
||
}
|
||
return a.priority > b.priority; // Max-heap
|
||
}
|
||
};
|
||
|
||
struct NotificationState {
|
||
std::string activeText;
|
||
std::string fileName;
|
||
size_t fontSize = 28;
|
||
s32 promptWidth = 448;
|
||
s32 promptHeight = 88;
|
||
PromptState state = PromptState::Inactive;
|
||
u64 expireNs = 0;
|
||
u64 stateStartNs = 0;
|
||
|
||
NotificationState() = default;
|
||
bool isTextEmpty() const { return activeText.empty(); }
|
||
};
|
||
|
||
// ---------------- Public Methods ----------------
|
||
|
||
void show(const std::string& msg, size_t fontSize = 26, u32 priority = 20,
|
||
const std::string& fileName = "", u32 durationMs = 2500,
|
||
s32 promptWidth = 448, s32 promptHeight = 88)
|
||
{
|
||
if (msg.empty()) return;
|
||
|
||
// Quick reject using atomics (fast-path)
|
||
if (!enabled_.load(std::memory_order_acquire)) return;
|
||
if (!ult::useNotifications) return;
|
||
if (generation_ != notificationGeneration.load(std::memory_order_acquire)) return;
|
||
|
||
NotificationData data;
|
||
data.text = msg;
|
||
data.fileName = fileName;
|
||
data.fontSize = std::clamp(fontSize, size_t(8), size_t(48));
|
||
data.promptWidth = std::clamp(promptWidth, s32(100), s32(1280));
|
||
data.promptHeight = std::clamp(promptHeight, s32(50), s32(720));
|
||
data.durationMs = std::clamp(durationMs, 500u, 30000u);
|
||
data.priority = priority;
|
||
data.arrivalNs = armTicksToNs(armGetSystemTick());
|
||
|
||
std::lock_guard<std::mutex> lg(state_mutex_);
|
||
|
||
// Re-check under lock to avoid TOCTOU
|
||
if (!enabled_.load(std::memory_order_acquire)) return;
|
||
if (generation_ != notificationGeneration.load(std::memory_order_acquire)) return;
|
||
if (pending_queue_.size() >= MAX_NOTIFS) return;
|
||
|
||
pending_queue_.push(data);
|
||
|
||
if (!is_active_) {
|
||
startNext_NoLock();
|
||
//pending_event_fire_.store(true, std::memory_order_release);
|
||
eventFire(¬ificationEvent);
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (isRendering) {
|
||
isRendering = false;
|
||
wasRendering = true;
|
||
|
||
leventSignal(&renderingStopEvent);
|
||
}
|
||
#endif
|
||
}
|
||
}
|
||
|
||
|
||
void draw(gfx::Renderer* renderer, bool promptOnly = false) {
|
||
if (ult::launchingOverlay.load(std::memory_order_acquire) ||
|
||
generation_ != notificationGeneration.load(std::memory_order_acquire)) return;
|
||
if (!enabled_.load(std::memory_order_acquire)) return;
|
||
|
||
NotificationState copy;
|
||
{
|
||
std::lock_guard<std::mutex> lg(state_mutex_);
|
||
if (current_state_.state == PromptState::Inactive || current_state_.activeText.empty()) return;
|
||
copy = current_state_;
|
||
}
|
||
|
||
const u64 now = armTicksToNs(armGetSystemTick());
|
||
const u64 elapsedMs = (now - copy.stateStartNs) / 1'000'000ULL;
|
||
|
||
s32 x = 0, y = 0;
|
||
switch (copy.state) {
|
||
case PromptState::SlidingIn: {
|
||
const float t = std::min(1.0f, float(elapsedMs) / SLIDE_DURATION_MS);
|
||
x = ult::useRightAlignment ?
|
||
(tsl::cfg::FramebufferWidth - copy.promptWidth + static_cast<s32>((1.0f - t) * copy.promptWidth)) :
|
||
static_cast<s32>(-copy.promptWidth + t * copy.promptWidth);
|
||
break;
|
||
}
|
||
case PromptState::Visible:
|
||
x = ult::useRightAlignment ? (tsl::cfg::FramebufferWidth - copy.promptWidth) : 0;
|
||
break;
|
||
case PromptState::SlidingOut: {
|
||
const float t = std::min(1.0f, float(elapsedMs) / SLIDE_DURATION_MS);
|
||
x = ult::useRightAlignment ?
|
||
(tsl::cfg::FramebufferWidth - copy.promptWidth + static_cast<s32>(t * copy.promptWidth)) :
|
||
static_cast<s32>(-t * copy.promptWidth);
|
||
break;
|
||
}
|
||
default: return;
|
||
}
|
||
|
||
const s32 scissorX = std::max(0, x);
|
||
const s32 scissorW = std::min(copy.promptWidth, tsl::cfg::FramebufferWidth - scissorX);
|
||
|
||
if (scissorX >= 0 && scissorW > 0 && copy.promptHeight > 0) {
|
||
renderer->enableScissoring(scissorX, y, scissorW, copy.promptHeight);
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
renderer->drawRect(x, y, copy.promptWidth, copy.promptHeight, defaultBackgroundColor);
|
||
#else
|
||
if (!promptOnly && ult::expandedMemory)
|
||
renderer->drawRectMultiThreaded(x, y, copy.promptWidth, copy.promptHeight, defaultBackgroundColor);
|
||
else
|
||
renderer->drawRect(x, y, copy.promptWidth, copy.promptHeight, defaultBackgroundColor);
|
||
#endif
|
||
|
||
if (!copy.activeText.empty()) {
|
||
std::vector<std::string> lines;
|
||
const std::string& text = copy.activeText;
|
||
|
||
size_t start = 0;
|
||
while (start < text.size() && lines.size() < 8) {
|
||
// Look for escaped "\n"
|
||
const size_t pos = text.find("\\n", start);
|
||
|
||
if (pos == std::string::npos) {
|
||
// No more "\n", take the rest
|
||
lines.emplace_back(text.substr(start));
|
||
break;
|
||
} else {
|
||
// Extract line up to the escape sequence
|
||
lines.emplace_back(text.substr(start, pos - start));
|
||
start = pos + 2; // Skip past "\n"
|
||
}
|
||
}
|
||
|
||
const auto fm = tsl::gfx::FontManager::getFontMetricsForCharacter('A', copy.fontSize);
|
||
const s32 startY = y + (copy.promptHeight - static_cast<int>(lines.size()) * fm.lineHeight) / 2 + fm.ascent;
|
||
|
||
for (size_t i = 0; i < lines.size(); ++i) {
|
||
const std::string& line = lines[i];
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
// Check if line contains "Ultrahand" (case insensitive)
|
||
const bool hasUltrahand = (line.find(ult::CAPITAL_ULTRAHAND_PROJECT_NAME) != std::string::npos);
|
||
|
||
if (hasUltrahand) {
|
||
// Draw line with dynamic Ultrahand effect
|
||
drawUltrahandLine(renderer, line, x, startY + static_cast<int>(i) * fm.lineHeight,
|
||
copy.fontSize, copy.promptWidth);
|
||
} else {
|
||
// Draw normal line
|
||
const auto [lw, lh] = renderer->getNotificationTextDimensions(line, false, copy.fontSize);
|
||
renderer->drawNotificationString(
|
||
line, false,
|
||
x + (copy.promptWidth - lw) / 2,
|
||
startY + static_cast<int>(i) * fm.lineHeight,
|
||
copy.fontSize, defaultTextColor
|
||
);
|
||
}
|
||
#else
|
||
// Draw normal line
|
||
const auto [lw, lh] = renderer->getNotificationTextDimensions(line, false, copy.fontSize);
|
||
renderer->drawNotificationString(
|
||
line, false,
|
||
x + (copy.promptWidth - lw) / 2,
|
||
startY + static_cast<int>(i) * fm.lineHeight,
|
||
copy.fontSize, defaultTextColor
|
||
);
|
||
#endif
|
||
}
|
||
}
|
||
|
||
if (!ult::useRightAlignment) {
|
||
renderer->drawRect(x + copy.promptWidth - 1, y, 1, copy.promptHeight, edgeSeparatorColor);
|
||
renderer->drawRect(x, y + copy.promptHeight - 1, copy.promptWidth, 1, edgeSeparatorColor);
|
||
} else {
|
||
renderer->drawRect(x, y, 1, copy.promptHeight, edgeSeparatorColor);
|
||
renderer->drawRect(x, y + copy.promptHeight - 1, copy.promptWidth, 1, edgeSeparatorColor);
|
||
}
|
||
|
||
renderer->disableScissoring();
|
||
}
|
||
}
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
void drawUltrahandLine(gfx::Renderer* renderer, const std::string& line, s32 x, s32 y,
|
||
u32 fontSize, s32 promptWidth) {
|
||
// Find position of "Ultrahand" in the line (case insensitive)
|
||
size_t ultrahandPos = std::string::npos;
|
||
std::string ultrahandToReplace;
|
||
|
||
// Check for "Ultrahand" first
|
||
ultrahandPos = line.find(ult::CAPITAL_ULTRAHAND_PROJECT_NAME);
|
||
if (ultrahandPos != std::string::npos) {
|
||
ultrahandToReplace = ult::CAPITAL_ULTRAHAND_PROJECT_NAME;
|
||
}
|
||
|
||
if (ultrahandPos == std::string::npos) {
|
||
// Fallback to normal drawing if not found
|
||
const auto [lw, lh] = renderer->getNotificationTextDimensions(line, false, fontSize);
|
||
renderer->drawNotificationString(line, false, x + (promptWidth - lw) / 2, y, fontSize, defaultTextColor);
|
||
return;
|
||
}
|
||
|
||
// Split the line into parts
|
||
const std::string before = line.substr(0, ultrahandPos);
|
||
const std::string ultra = ult::SPLIT_PROJECT_NAME_1;
|
||
const std::string hand = ult::SPLIT_PROJECT_NAME_2;
|
||
const std::string after = line.substr(ultrahandPos + ultrahandToReplace.length());
|
||
|
||
// Calculate individual part widths to get accurate total width
|
||
s32 beforeWidth = 0, ultraWidth = 0, handWidth = 0, afterWidth = 0;
|
||
|
||
if (!before.empty()) {
|
||
const auto [bw, bh] = renderer->getNotificationTextDimensions(before, false, fontSize);
|
||
beforeWidth = bw;
|
||
}
|
||
|
||
if (!after.empty()) {
|
||
const auto [aw, ah] = renderer->getNotificationTextDimensions(after, false, fontSize);
|
||
afterWidth = aw;
|
||
}
|
||
|
||
const auto [hw, hh] = renderer->getNotificationTextDimensions(hand, false, fontSize);
|
||
handWidth = hw;
|
||
|
||
// For "Ultra" width, we need to handle dynamic vs static rendering differently
|
||
if (ult::useDynamicLogo) {
|
||
// Calculate width by measuring each character (since dynamic rendering might have slight differences)
|
||
for (const char letter : ultra) {
|
||
const std::string letterStr(1, letter);
|
||
const auto [lw, lh] = renderer->getNotificationTextDimensions(letterStr, false, fontSize);
|
||
ultraWidth += lw;
|
||
}
|
||
} else {
|
||
const auto [uw, uh] = renderer->getNotificationTextDimensions(ultra, false, fontSize);
|
||
ultraWidth = uw;
|
||
}
|
||
|
||
// Calculate total width and starting position for centering
|
||
const s32 totalWidth = beforeWidth + ultraWidth + handWidth + afterWidth;
|
||
s32 currentX = x + (promptWidth - totalWidth) / 2;
|
||
|
||
// Draw each part in sequence
|
||
|
||
// Draw "before" part
|
||
if (!before.empty()) {
|
||
renderer->drawNotificationString(before, false, currentX, y, fontSize, defaultTextColor);
|
||
currentX += beforeWidth;
|
||
}
|
||
|
||
// Draw dynamic "Ultra" part
|
||
if (ult::useDynamicLogo) {
|
||
const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
const double currentTimeCount = static_cast<double>(currentTime_ns) / 1000000000.0;
|
||
const double timeBase = std::fmod(currentTimeCount, cycleDuration);
|
||
const double waveScale = 2.0 * ult::_M_PI / cycleDuration;
|
||
static constexpr double phaseShift = ult::_M_PI / 2.0;
|
||
|
||
float countOffset = 0;
|
||
for (const char letter : ultra) {
|
||
const double wavePhase = waveScale * (timeBase + static_cast<double>(countOffset));
|
||
const double rawProgress = std::cos(wavePhase - phaseShift);
|
||
|
||
const double normalizedProgress = (rawProgress + 1.0) * 0.5;
|
||
const double smoothedProgress = normalizedProgress * normalizedProgress * (3.0 - 2.0 * normalizedProgress);
|
||
const double ultraSmoothProgress = smoothedProgress * smoothedProgress * (3.0 - 2.0 * smoothedProgress);
|
||
|
||
const double blend = std::max(0.0, std::min(1.0, ultraSmoothProgress));
|
||
|
||
const tsl::Color highlightColor = {
|
||
static_cast<u8>(dynamicLogoRGB1.r + (dynamicLogoRGB2.r - dynamicLogoRGB1.r) * blend + 0.5),
|
||
static_cast<u8>(dynamicLogoRGB1.g + (dynamicLogoRGB2.g - dynamicLogoRGB1.g) * blend + 0.5),
|
||
static_cast<u8>(dynamicLogoRGB1.b + (dynamicLogoRGB2.b - dynamicLogoRGB1.b) * blend + 0.5),
|
||
15
|
||
};
|
||
|
||
const std::string letterStr(1, letter);
|
||
const auto [letterWidth, letterHeight] = renderer->drawNotificationString(letterStr, false, currentX, y, fontSize, highlightColor);
|
||
currentX += letterWidth;
|
||
countOffset -= static_cast<float>(cycleDuration / 8.0);
|
||
}
|
||
} else {
|
||
renderer->drawNotificationString(ultra, false, currentX, y, fontSize, logoColor1);
|
||
currentX += ultraWidth;
|
||
}
|
||
|
||
// Draw static "hand" part
|
||
renderer->drawNotificationString(hand, false, currentX, y, fontSize, logoColor2);
|
||
currentX += handWidth;
|
||
|
||
// Draw "after" part
|
||
if (!after.empty()) {
|
||
renderer->drawNotificationString(after, false, currentX, y, fontSize, defaultTextColor);
|
||
}
|
||
}
|
||
#endif
|
||
|
||
void update() {
|
||
if (!isActive()) {
|
||
return;
|
||
}
|
||
std::lock_guard<std::mutex> lg(state_mutex_);
|
||
// Optional extra safety: skip if already inactive and queue empty
|
||
if (ult::launchingOverlay.load(std::memory_order_acquire) ||
|
||
(!is_active_ && current_state_.activeText.empty() && pending_queue_.empty())) {
|
||
return;
|
||
}
|
||
|
||
//if (pending_event_fire_.load(std::memory_order_acquire) &&
|
||
// !ult::launchingOverlay.load(std::memory_order_acquire) &&
|
||
// generation_ == notificationGeneration.load(std::memory_order_acquire))
|
||
//{
|
||
// pending_event_fire_.store(false, std::memory_order_release);
|
||
//
|
||
// if ((current_state_.activeText.empty() || current_state_.state == PromptState::Inactive) && pending_queue_.empty()) {
|
||
// eventFire(¬ificationEvent);
|
||
// }
|
||
//}
|
||
|
||
//std::lock_guard<std::mutex> lg(state_mutex_);
|
||
if (generation_ != notificationGeneration.load(std::memory_order_acquire) ||
|
||
!enabled_.load(std::memory_order_acquire))
|
||
{
|
||
current_state_ = NotificationState{};
|
||
is_active_ = false;
|
||
return;
|
||
}
|
||
|
||
const u64 now = armTicksToNs(armGetSystemTick());
|
||
const u64 elapsedMs = (current_state_.stateStartNs == 0) ? 0 : (now - current_state_.stateStartNs) / 1'000'000ULL;
|
||
|
||
switch (current_state_.state) {
|
||
case PromptState::SlidingIn:
|
||
if (elapsedMs >= SLIDE_DURATION_MS) {
|
||
current_state_.state = PromptState::Visible;
|
||
current_state_.stateStartNs = now;
|
||
}
|
||
break;
|
||
case PromptState::Visible:
|
||
if (now >= current_state_.expireNs) {
|
||
current_state_.state = PromptState::SlidingOut;
|
||
current_state_.stateStartNs = now;
|
||
}
|
||
break;
|
||
case PromptState::SlidingOut:
|
||
if (elapsedMs >= SLIDE_DURATION_MS) {
|
||
const std::string fileToDelete = current_state_.fileName;
|
||
// Delete the JSON file safely
|
||
if (!fileToDelete.empty()) {
|
||
std::lock_guard<std::mutex> lg(notificationJsonMutex);
|
||
const std::string fullPath = ult::NOTIFICATIONS_PATH + fileToDelete;
|
||
remove(fullPath.c_str()); // ignore errors for now
|
||
}
|
||
|
||
current_state_ = NotificationState{};
|
||
const bool hadNext = startNext_NoLock();
|
||
if (!hadNext) {
|
||
is_active_ = false;
|
||
}
|
||
}
|
||
break;
|
||
default: break;
|
||
}
|
||
}
|
||
|
||
bool isActive() const {
|
||
if (!ult::useNotifications) return false;
|
||
if (generation_ != notificationGeneration.load(std::memory_order_acquire)) return false;
|
||
std::lock_guard<std::mutex> lg(state_mutex_);
|
||
if (is_active_) return true;
|
||
if (!pending_queue_.empty()) return true;
|
||
//if (pending_event_fire_.load(std::memory_order_acquire)) return true;
|
||
if (!current_state_.activeText.empty() && current_state_.state != PromptState::Inactive) return true;
|
||
return false;
|
||
}
|
||
|
||
void shutdown() {
|
||
enabled_.store(false, std::memory_order_release);
|
||
notificationGeneration.fetch_add(1, std::memory_order_acq_rel);
|
||
generation_ = notificationGeneration.load(std::memory_order_acquire);
|
||
|
||
std::lock_guard<std::mutex> lg(state_mutex_);
|
||
while (!pending_queue_.empty()) pending_queue_.pop();
|
||
current_state_ = NotificationState{};
|
||
is_active_ = false;
|
||
//pending_event_fire_.store(false, std::memory_order_release);
|
||
}
|
||
|
||
void forceShutdown() {
|
||
enabled_.store(false, std::memory_order_release);
|
||
//pending_event_fire_.store(false, std::memory_order_release);
|
||
}
|
||
|
||
//void forceCompleteTransition() {
|
||
// std::lock_guard<std::mutex> lg(state_mutex_);
|
||
// current_state_ = NotificationState{};
|
||
// while (!pending_queue_.empty()) pending_queue_.pop();
|
||
// is_active_ = false;
|
||
// //pending_event_fire_.store(false, std::memory_order_release);
|
||
//}
|
||
//
|
||
//void freezeState() {
|
||
// generation_++;
|
||
// enabled_.store(false, std::memory_order_release);
|
||
// {
|
||
// std::lock_guard<std::mutex> lg(state_mutex_);
|
||
// is_active_ = false;
|
||
// }
|
||
// //pending_event_fire_.store(false, std::memory_order_release);
|
||
//}
|
||
|
||
private:
|
||
static constexpr size_t MAX_NOTIFS = 30;
|
||
static constexpr u32 SLIDE_DURATION_MS = 200;
|
||
static constexpr double cycleDuration = 1.6;
|
||
|
||
mutable std::mutex state_mutex_;
|
||
NotificationState current_state_;
|
||
std::priority_queue<NotificationData, std::vector<NotificationData>, NotificationCompare> pending_queue_;
|
||
|
||
std::atomic<bool> enabled_{true};
|
||
bool is_active_{false}; // protected by mutex
|
||
//std::atomic<bool> pending_event_fire_{false};
|
||
uint32_t generation_{0};
|
||
|
||
bool startNext_NoLock() {
|
||
if (pending_queue_.empty()) return false;
|
||
const NotificationData next = pending_queue_.top();
|
||
pending_queue_.pop();
|
||
|
||
const u64 now = armTicksToNs(armGetSystemTick());
|
||
current_state_.activeText = next.text;
|
||
current_state_.fileName = next.fileName;
|
||
current_state_.fontSize = next.fontSize;
|
||
current_state_.promptWidth = next.promptWidth;
|
||
current_state_.promptHeight = next.promptHeight;
|
||
current_state_.state = PromptState::SlidingIn;
|
||
current_state_.stateStartNs = now;
|
||
current_state_.expireNs = now + static_cast<u64>(SLIDE_DURATION_MS) * 1'000'000ULL
|
||
+ static_cast<u64>(next.durationMs) * 1'000'000ULL;
|
||
is_active_ = true;
|
||
return true;
|
||
}
|
||
};
|
||
|
||
// Optional: pointer to global notification
|
||
static inline NotificationPrompt* notification = nullptr;
|
||
|
||
|
||
// GUI
|
||
|
||
/**
|
||
* @brief The top level Gui class
|
||
* @note The main menu and every sub menu are a separate Gui. Create your own Gui class that extends from this one to create your own menus
|
||
*
|
||
*/
|
||
class Gui {
|
||
public:
|
||
Gui() {
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
#else
|
||
{
|
||
#if INITIALIZE_IN_GUI_DIRECTIVE // for different project structures
|
||
tsl::initializeThemeVars();
|
||
|
||
// Load the bitmap file into memory
|
||
ult::loadWallpaperFileWhenSafe();
|
||
#endif
|
||
}
|
||
#endif
|
||
}
|
||
|
||
virtual ~Gui() {
|
||
if (this->m_topElement != nullptr)
|
||
delete this->m_topElement;
|
||
|
||
if (this->m_bottomElement != nullptr)
|
||
delete this->m_bottomElement;
|
||
}
|
||
|
||
/**
|
||
* @brief Creates all elements present in this Gui
|
||
* @note Implement this function and let it return a heap allocated element used as the top level element. This is usually some kind of frame e.g \ref OverlayFrame
|
||
*
|
||
* @return Top level element
|
||
*/
|
||
virtual elm::Element* createUI() = 0;
|
||
|
||
/**
|
||
* @brief Called once per frame to update values
|
||
*
|
||
*/
|
||
virtual void update() {}
|
||
|
||
/**
|
||
* @brief Called once per frame with the latest HID inputs
|
||
*
|
||
* @param keysDown Buttons pressed in the last frame
|
||
* @param keysHeld Buttons held down longer than one frame
|
||
* @param touchInput Last touch position
|
||
* @param leftJoyStick Left joystick position
|
||
* @param rightJoyStick Right joystick position
|
||
* @return Weather or not the input has been consumed
|
||
*/
|
||
virtual inline bool handleInput(u64 keysDown, u64 keysHeld, const HidTouchState &touchPos, HidAnalogStickState leftJoyStick, HidAnalogStickState rightJoyStick) {
|
||
return false;
|
||
}
|
||
|
||
/**
|
||
* @brief Gets the top level element
|
||
*
|
||
* @return Top level element
|
||
*/
|
||
elm::Element* getTopElement() {
|
||
return this->m_topElement;
|
||
}
|
||
|
||
/**
|
||
* @brief Gets the bottom level element
|
||
*
|
||
* @return Bottom level element
|
||
*/
|
||
elm::Element* getBottomElement() {
|
||
return this->m_bottomElement;
|
||
}
|
||
|
||
/**
|
||
* @brief Get the currently focused element
|
||
*
|
||
* @return Focused element
|
||
*/
|
||
elm::Element* getFocusedElement() {
|
||
return this->m_focusedElement;
|
||
}
|
||
|
||
/**
|
||
* @brief Requests focus to a element
|
||
* @note Use this function when focusing a element outside of a element's requestFocus function
|
||
*
|
||
* @param element Element to focus
|
||
* @param direction Focus direction
|
||
*/
|
||
inline void requestFocus(elm::Element *element, FocusDirection direction, bool shake = true) {
|
||
elm::Element *oldFocus = this->m_focusedElement;
|
||
|
||
if (element != nullptr) {
|
||
this->m_focusedElement = element->requestFocus(oldFocus, direction);
|
||
|
||
if (oldFocus != nullptr)
|
||
oldFocus->setFocused(false);
|
||
|
||
if (this->m_focusedElement != nullptr) {
|
||
this->m_focusedElement->setFocused(true);
|
||
}
|
||
}
|
||
|
||
if (shake && oldFocus == this->m_focusedElement && this->m_focusedElement != nullptr)
|
||
this->m_focusedElement->shakeHighlight(direction);
|
||
}
|
||
|
||
/**
|
||
* @brief Removes focus from a element
|
||
*
|
||
* @param element Element to remove focus from. Pass nullptr to remove the focus unconditionally
|
||
*/
|
||
inline void removeFocus(elm::Element* element = nullptr) {
|
||
if (element == nullptr || element == this->m_focusedElement) {
|
||
if (this->m_focusedElement != nullptr) {
|
||
this->m_focusedElement->setFocused(false);
|
||
this->m_focusedElement = nullptr;
|
||
}
|
||
}
|
||
}
|
||
|
||
inline void restoreFocus() {
|
||
this->m_initialFocusSet = false;
|
||
}
|
||
|
||
protected:
|
||
constexpr static inline auto a = &gfx::Renderer::a;
|
||
constexpr static inline auto aWithOpacity = &gfx::Renderer::aWithOpacity;
|
||
|
||
private:
|
||
elm::Element *m_focusedElement = nullptr;
|
||
elm::Element *m_topElement = nullptr;
|
||
elm::Element *m_bottomElement = nullptr;
|
||
|
||
bool m_initialFocusSet = false;
|
||
|
||
friend class Overlay;
|
||
friend class gfx::Renderer;
|
||
|
||
//// Function to recursively find the bottom element
|
||
//void findBottomElement(elm::Element* currentElement) {
|
||
// // Base case: if the current element has no children, it is the bottom element
|
||
// if (currentElement->getChildren().empty()) {
|
||
// m_bottomElement = currentElement;
|
||
// return;
|
||
// }
|
||
//
|
||
// // Recursive case: traverse through all children elements
|
||
// for (elm::Element* child : currentElement->getChildren()) {
|
||
// findBottomElement(child);
|
||
// }
|
||
//}
|
||
|
||
/**
|
||
* @brief Draws the Gui
|
||
*
|
||
* @param renderer
|
||
*/
|
||
void draw(gfx::Renderer *renderer) {
|
||
if (this->m_topElement != nullptr)
|
||
this->m_topElement->draw(renderer);
|
||
}
|
||
|
||
inline bool initialFocusSet() {
|
||
return this->m_initialFocusSet;
|
||
}
|
||
|
||
inline void markInitialFocusSet() {
|
||
this->m_initialFocusSet = true;
|
||
}
|
||
|
||
};
|
||
|
||
|
||
// Overlay
|
||
|
||
/**
|
||
* @brief The top level Overlay class
|
||
* @note Every Tesla overlay should have exactly one Overlay class initializing services and loading the default Gui
|
||
*/
|
||
class Overlay {
|
||
protected:
|
||
/**
|
||
* @brief Constructor
|
||
* @note Called once when the Overlay gets loaded
|
||
*/
|
||
Overlay() {}
|
||
public:
|
||
|
||
/**
|
||
* @brief Deconstructor
|
||
* @note Called once when the Overlay exits
|
||
*
|
||
*/
|
||
virtual ~Overlay() {}
|
||
|
||
|
||
/**
|
||
* @brief Initializes services
|
||
* @note Called once at the start to initializes services. You have a sm session available during this call, no need to initialize sm yourself
|
||
*/
|
||
virtual void initServices() {}
|
||
|
||
/**
|
||
* @brief Exits services
|
||
* @note Make sure to exit all services you initialized in \ref Overlay::initServices() here to prevent leaking handles
|
||
*/
|
||
virtual void exitServices() {}
|
||
|
||
/**
|
||
* @brief Called before overlay changes from invisible to visible state
|
||
*
|
||
*/
|
||
virtual void onShow() {}
|
||
|
||
/**
|
||
* @brief Called before overlay changes from visible to invisible state
|
||
*
|
||
*/
|
||
virtual void onHide() {}
|
||
|
||
/**
|
||
* @brief Loads the default Gui
|
||
* @note This function should return the initial Gui to load using the \ref Gui::initially<T>(Args.. args) function
|
||
* e.g `return initially<GuiMain>();`
|
||
*
|
||
* @return Default Gui
|
||
*/
|
||
virtual std::unique_ptr<tsl::Gui> loadInitialGui() = 0;
|
||
|
||
/**
|
||
* @brief Gets a reference to the current Gui on top of the Gui stack
|
||
*
|
||
* @return Current Gui reference
|
||
*/
|
||
std::unique_ptr<tsl::Gui>& getCurrentGui() {
|
||
return this->m_guiStack.top();
|
||
}
|
||
|
||
/**
|
||
* @brief Shows the Gui
|
||
*
|
||
*/
|
||
void show() {
|
||
|
||
if (this->m_disableNextAnimation) {
|
||
this->m_animationCounter = MAX_ANIMATION_COUNTER;
|
||
this->m_disableNextAnimation = false;
|
||
}
|
||
else {
|
||
this->m_fadeInAnimationPlaying = true;
|
||
this->m_animationCounter = 0;
|
||
}
|
||
|
||
ult::isHidden.store(false);
|
||
this->onShow();
|
||
|
||
//if (auto& currGui = this->getCurrentGui(); currGui != nullptr) // TESTING DISABLED (EFFECTS NEED TO BE VERIFIED)
|
||
// currGui->restoreFocus();
|
||
}
|
||
|
||
/**
|
||
* @brief Hides the Gui
|
||
*
|
||
*/
|
||
void hide(bool useNoFade = false) {
|
||
if (useNoFade) {
|
||
// Immediately hide overlay
|
||
ult::isHidden.store(true);
|
||
this->m_shouldHide = true;
|
||
return;
|
||
}
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (FullMode && !deactivateOriginalFooter) {
|
||
|
||
if (this->m_disableNextAnimation) {
|
||
this->m_animationCounter = 0;
|
||
this->m_disableNextAnimation = false;
|
||
}
|
||
else {
|
||
this->m_fadeOutAnimationPlaying = true;
|
||
this->m_animationCounter = MAX_ANIMATION_COUNTER;
|
||
}
|
||
ult::isHidden.store(true);
|
||
this->onHide();
|
||
}
|
||
#else
|
||
|
||
if (this->m_disableNextAnimation) {
|
||
this->m_animationCounter = 0;
|
||
this->m_disableNextAnimation = false;
|
||
}
|
||
else {
|
||
this->m_fadeOutAnimationPlaying = true;
|
||
this->m_animationCounter = MAX_ANIMATION_COUNTER;
|
||
}
|
||
ult::isHidden.store(true);
|
||
this->onHide();
|
||
#endif
|
||
}
|
||
|
||
/**
|
||
* @brief Returns whether fade animation is playing
|
||
*
|
||
* @return whether fade animation is playing
|
||
*/
|
||
bool fadeAnimationPlaying() {
|
||
return this->m_fadeInAnimationPlaying || this->m_fadeOutAnimationPlaying;
|
||
}
|
||
|
||
/**
|
||
* @brief Closes the Gui
|
||
* @note This makes the Tesla overlay exit and return back to the Tesla-Menu
|
||
*
|
||
*/
|
||
void close(bool forceClose = false) {
|
||
if (!forceClose && notification && notification->isActive()) {
|
||
this->closeAfter();
|
||
this->hide(true);
|
||
return;
|
||
}
|
||
|
||
this->m_shouldClose = true;
|
||
}
|
||
|
||
/**
|
||
* @brief Closes the Gui
|
||
* @note This makes the Tesla overlay exit and return back to the Tesla-Menu
|
||
*
|
||
*/
|
||
void closeAfter() {
|
||
this->m_shouldCloseAfter = true;
|
||
|
||
}
|
||
|
||
/**
|
||
* @brief Gets the Overlay instance
|
||
*
|
||
* @return Overlay instance
|
||
*/
|
||
static inline Overlay* const get() {
|
||
return Overlay::s_overlayInstance;
|
||
}
|
||
|
||
/**
|
||
* @brief Creates the initial Gui of an Overlay and moves the object to the Gui stack
|
||
*
|
||
* @tparam T
|
||
* @tparam Args
|
||
* @param args
|
||
* @return constexpr std::unique_ptr<T>
|
||
*/
|
||
template<typename T, typename ... Args>
|
||
constexpr inline std::unique_ptr<T> initially(Args&&... args) {
|
||
return std::make_unique<T>(args...);
|
||
}
|
||
|
||
private:
|
||
using GuiPtr = std::unique_ptr<tsl::Gui>;
|
||
std::stack<GuiPtr, std::list<GuiPtr>> m_guiStack;
|
||
static inline Overlay *s_overlayInstance = nullptr;
|
||
|
||
bool m_fadeInAnimationPlaying = false, m_fadeOutAnimationPlaying = false;
|
||
u8 m_animationCounter = 0;
|
||
static constexpr int MAX_ANIMATION_COUNTER = 5; // Define the maximum animation counter value
|
||
|
||
bool m_shouldHide = false;
|
||
bool m_shouldClose = false;
|
||
bool m_shouldCloseAfter = false;
|
||
|
||
bool m_disableNextAnimation = false;
|
||
|
||
bool m_closeOnExit;
|
||
|
||
static inline std::atomic<bool> isNavigatingBackwards{false};
|
||
bool justNavigated = false;
|
||
|
||
/**
|
||
* @brief Initializes the Renderer
|
||
*
|
||
*/
|
||
void initScreen() {
|
||
gfx::Renderer::get().init();
|
||
}
|
||
|
||
/**
|
||
* @brief Exits the Renderer
|
||
*
|
||
*/
|
||
void exitScreen() {
|
||
gfx::Renderer::get().exit();
|
||
}
|
||
|
||
/**
|
||
* @brief Weather or not the Gui should get hidden
|
||
*
|
||
* @return should hide
|
||
*/
|
||
bool shouldHide() {
|
||
return this->m_shouldHide;
|
||
}
|
||
|
||
/**
|
||
* @brief Weather or not hte Gui should get closed
|
||
*
|
||
* @return should close
|
||
*/
|
||
bool shouldClose() {
|
||
return this->m_shouldClose;
|
||
}
|
||
|
||
/**
|
||
* @brief Weather or not hte Gui should get closed after
|
||
*
|
||
* @return should close after
|
||
*/
|
||
bool shouldCloseAfter() {
|
||
return this->m_shouldCloseAfter;
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Quadratic ease-in-out function
|
||
*
|
||
* @param t Normalized time (0 to 1)
|
||
* @return Eased value
|
||
*/
|
||
float calculateEaseInOut(float t) {
|
||
if (t < 0.5) {
|
||
return 2 * t * t;
|
||
} else {
|
||
return -1 + (4 - 2 * t) * t;
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief Handles fade in and fade out animations of the Overlay
|
||
*
|
||
*/
|
||
void animationLoop() {
|
||
|
||
|
||
if (this->m_fadeInAnimationPlaying) {
|
||
if (this->m_animationCounter < MAX_ANIMATION_COUNTER) {
|
||
this->m_animationCounter++;
|
||
}
|
||
|
||
if (this->m_animationCounter >= MAX_ANIMATION_COUNTER) {
|
||
this->m_fadeInAnimationPlaying = false;
|
||
}
|
||
}
|
||
|
||
if (this->m_fadeOutAnimationPlaying) {
|
||
if (this->m_animationCounter > 0) {
|
||
this->m_animationCounter--;
|
||
}
|
||
|
||
if (this->m_animationCounter == 0) {
|
||
this->m_fadeOutAnimationPlaying = false;
|
||
this->m_shouldHide = true;
|
||
}
|
||
}
|
||
|
||
// Calculate and set the opacity using an easing function
|
||
//float opacity = calculateEaseInOut(static_cast<float>(this->m_animationCounter) / MAX_ANIMATION_COUNTER);
|
||
gfx::Renderer::setOpacity(calculateEaseInOut(static_cast<float>(this->m_animationCounter) / MAX_ANIMATION_COUNTER));
|
||
}
|
||
|
||
|
||
|
||
/**
|
||
* @brief Overlay Main loop
|
||
*
|
||
*/
|
||
void loop(bool promptOnly = false) {
|
||
// Early exit check - avoid all work if shutting down
|
||
if (ult::launchingOverlay.load(std::memory_order_acquire)) {
|
||
return;
|
||
}
|
||
|
||
// Static state management - made atomic for thread safety
|
||
static std::atomic<bool> screenshotStacksAdded{false};
|
||
static std::atomic<bool> notificationCacheNeedsClearing{false};
|
||
|
||
auto& renderer = gfx::Renderer::get();
|
||
renderer.startFrame();
|
||
|
||
// Handle main UI rendering
|
||
if (!promptOnly) {
|
||
// Screenshot management - thread-safe
|
||
if (!screenshotStacksAdded.load(std::memory_order_acquire)) {
|
||
renderer.addScreenshotStacks();
|
||
screenshotStacksAdded.store(true, std::memory_order_release);
|
||
}
|
||
|
||
this->animationLoop();
|
||
this->getCurrentGui()->update();
|
||
this->getCurrentGui()->draw(&renderer);
|
||
|
||
// Mark that we may need to clear notification cache later
|
||
notificationCacheNeedsClearing.store(true, std::memory_order_release);
|
||
} else {
|
||
// Prompt-only mode - clean up screenshots
|
||
if (screenshotStacksAdded.load(std::memory_order_acquire) && !screenshotsAreDisabled) {
|
||
renderer.removeScreenshotStacks();
|
||
screenshotStacksAdded.store(false, std::memory_order_release);
|
||
}
|
||
renderer.clearScreen();
|
||
}
|
||
|
||
// Notification handling - safer approach with consistent ordering
|
||
{
|
||
if (notification) {
|
||
|
||
if (notification->isActive()) {
|
||
// Snapshot pointer to avoid it becoming null mid-use
|
||
notification->update();
|
||
notification->draw(&renderer, promptOnly);
|
||
notificationCacheNeedsClearing.store(true, std::memory_order_release);
|
||
} else {
|
||
if (notificationCacheNeedsClearing.exchange(false, std::memory_order_acq_rel)) {
|
||
tsl::gfx::FontManager::clearNotificationCache();
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (wasRendering) {
|
||
wasRendering = false;
|
||
isRendering = true;
|
||
leventClear(&renderingStopEvent);
|
||
}
|
||
#endif
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
renderer.endFrame();
|
||
}
|
||
|
||
// Calculate transition using ease-in-out curve instead of linear
|
||
float easeInOutCubic(float t) {
|
||
return t < 0.5f ? 4.0f * t * t * t : 1.0f - pow(-2.0f * t + 2.0f, 3.0f) / 2.0f;
|
||
}
|
||
|
||
|
||
|
||
void handleInput(u64 keysDown, u64 keysHeld, bool touchDetected, const HidTouchState &touchPos, HidAnalogStickState joyStickPosLeft, HidAnalogStickState joyStickPosRight) {
|
||
if (!ult::internalTouchReleased.load(std::memory_order_acquire) || ult::launchingOverlay.load(std::memory_order_acquire))
|
||
return;
|
||
|
||
// Static variables to maintain state between function calls
|
||
static HidTouchState initialTouchPos = { 0 };
|
||
static HidTouchState oldTouchPos = { 0 };
|
||
static bool oldTouchDetected = false;
|
||
static elm::TouchEvent touchEvent, oldTouchEvent;
|
||
|
||
static u64 buttonPressTime_ns = 0, lastKeyEventTime_ns = 0, keyEventInterval_ns = 67000000ULL;
|
||
static bool singlePressHandled = false;
|
||
static constexpr u64 CLICK_THRESHOLD_NS = 340000000ULL; // 340ms in nanoseconds
|
||
|
||
|
||
static bool hasScrolled = false;
|
||
static void* lastGuiPtr = nullptr; // Use void* instead
|
||
|
||
auto& currentGui = this->getCurrentGui();
|
||
|
||
// Return early if current GUI is not available or internal touch is not released
|
||
if (!currentGui) {
|
||
|
||
elm::Element::setInputMode(InputMode::Controller);
|
||
|
||
oldTouchPos = { 0 };
|
||
initialTouchPos = { 0 };
|
||
touchEvent = elm::TouchEvent::None;
|
||
ult::stillTouching.store(false, std::memory_order_release);
|
||
ult::interruptedTouch.store(false, std::memory_order_release);
|
||
return;
|
||
}
|
||
|
||
// Retrieve current focus and top/bottom elements of the GUI
|
||
auto currentFocus = currentGui->getFocusedElement();
|
||
auto topElement = currentGui->getTopElement();
|
||
|
||
|
||
const bool interpreterIsRunning = ult::runningInterpreter.load(std::memory_order_acquire);
|
||
#if !IS_STATUS_MONITOR_DIRECTIVE
|
||
if (interpreterIsRunning) {
|
||
if (keysDown & KEY_UP && !(keysHeld & ~KEY_UP & ALL_KEYS_MASK))
|
||
currentFocus->shakeHighlight(FocusDirection::Up);
|
||
else if (keysDown & KEY_DOWN && !(keysHeld & ~KEY_DOWN & ALL_KEYS_MASK))
|
||
currentFocus->shakeHighlight(FocusDirection::Down);
|
||
else if (keysDown & KEY_LEFT && !(keysHeld & ~KEY_LEFT & ALL_KEYS_MASK))
|
||
currentFocus->shakeHighlight(FocusDirection::Left);
|
||
else if (keysDown & KEY_RIGHT && !(keysHeld & ~KEY_RIGHT & ALL_KEYS_MASK))
|
||
currentFocus->shakeHighlight(FocusDirection::Right);
|
||
}
|
||
#endif
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (FullMode && !deactivateOriginalFooter) {
|
||
if (ult::simulatedSelect.load(std::memory_order_acquire)) {
|
||
ult::simulatedSelect.store(false, std::memory_order_release);
|
||
keysDown |= KEY_A;
|
||
}
|
||
if (ult::simulatedBack.load(std::memory_order_acquire)) {
|
||
ult::simulatedBack.store(false, std::memory_order_release);
|
||
keysDown |= KEY_B;
|
||
}
|
||
|
||
if (!overrideBackButton) {
|
||
if (keysDown & KEY_B && !(keysHeld & ~KEY_B & ALL_KEYS_MASK)) {
|
||
if (!currentGui->handleInput(KEY_B,0,{},{},{})) {
|
||
this->goBack();
|
||
//ult::simulatedBackComplete = true;
|
||
}
|
||
return;
|
||
}
|
||
}
|
||
} else {
|
||
if (ult::simulatedSelect.load(std::memory_order_acquire))
|
||
ult::simulatedSelect.store(false, std::memory_order_release);
|
||
if (ult::simulatedBack.load(std::memory_order_acquire))
|
||
ult::simulatedBack.store(false, std::memory_order_release);
|
||
}
|
||
#else
|
||
if (ult::simulatedSelect.load(std::memory_order_acquire)) {
|
||
ult::simulatedSelect.store(false, std::memory_order_release);
|
||
keysDown |= KEY_A;
|
||
}
|
||
|
||
if (ult::simulatedBack.load(std::memory_order_acquire)) {
|
||
ult::simulatedBack.store(false, std::memory_order_release);
|
||
keysDown |= KEY_B;
|
||
//ult::simulatedBack = false;
|
||
}
|
||
|
||
if (!overrideBackButton) {
|
||
if (keysDown & KEY_B && !(keysHeld & ~KEY_B & ALL_KEYS_MASK)) {
|
||
if (!currentGui->handleInput(KEY_B,0,{},{},{})) {
|
||
this->goBack();
|
||
//ult::simulatedBackComplete = true;
|
||
}
|
||
return;
|
||
}
|
||
}
|
||
#endif
|
||
|
||
// Reset touch state when GUI changes
|
||
if (currentGui.get() != lastGuiPtr) { // or just currentGui != lastGuiPtr if it's not a smart pointer
|
||
hasScrolled = false;
|
||
oldTouchEvent = elm::TouchEvent::None;
|
||
oldTouchDetected = false;
|
||
oldTouchPos = { 0 };
|
||
initialTouchPos = { 0 };
|
||
lastGuiPtr = currentGui.get(); // or just currentGui
|
||
}
|
||
|
||
const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
|
||
if (!currentFocus && !ult::simulatedBack.load(std::memory_order_acquire) && !ult::stillTouching.load(std::memory_order_acquire) && !oldTouchDetected && !interpreterIsRunning) {
|
||
if (!topElement) return;
|
||
|
||
if (!currentGui->initialFocusSet() || keysDown & (KEY_UP | KEY_DOWN | KEY_LEFT | KEY_RIGHT)) {
|
||
currentGui->requestFocus(topElement, FocusDirection::None);
|
||
currentGui->markInitialFocusSet();
|
||
}
|
||
}
|
||
if (isNavigatingBackwards.load(std::memory_order_acquire) && !currentFocus && topElement && keysDown & (KEY_UP | KEY_DOWN | KEY_LEFT | KEY_RIGHT)) {
|
||
currentGui->requestFocus(topElement, FocusDirection::None);
|
||
currentGui->markInitialFocusSet();
|
||
isNavigatingBackwards.store(false, std::memory_order_release);
|
||
|
||
// Reset navigation timing to prevent fast scrolling
|
||
buttonPressTime_ns = currentTime_ns;
|
||
lastKeyEventTime_ns = buttonPressTime_ns;
|
||
singlePressHandled = false;
|
||
}
|
||
|
||
|
||
|
||
if (!currentFocus && !touchDetected && (!oldTouchDetected || oldTouchEvent == elm::TouchEvent::Scroll)) {
|
||
if (!isNavigatingBackwards.load(std::memory_order_acquire) &&
|
||
!ult::shortTouchAndRelease.load(std::memory_order_acquire) &&
|
||
!ult::longTouchAndRelease.load(std::memory_order_acquire) &&
|
||
!ult::simulatedSelect.load(std::memory_order_acquire) &&
|
||
!ult::simulatedBack.load(std::memory_order_acquire) &&
|
||
!ult::simulatedNextPage.load(std::memory_order_acquire)
|
||
&& topElement) {
|
||
|
||
if (oldTouchEvent == elm::TouchEvent::Scroll) {
|
||
hasScrolled = true;
|
||
}
|
||
if (!hasScrolled) {
|
||
currentGui->removeFocus();
|
||
currentGui->requestFocus(topElement, FocusDirection::None);
|
||
}
|
||
}
|
||
else if (ult::longTouchAndRelease.load(std::memory_order_acquire)) {
|
||
ult::longTouchAndRelease.store(false, std::memory_order_release);
|
||
hasScrolled = true;
|
||
} else if (ult::shortTouchAndRelease.load(std::memory_order_acquire)) { // cant be handled correctly without knowing where it is going after release
|
||
ult::shortTouchAndRelease.store(false, std::memory_order_release);
|
||
hasScrolled = true;
|
||
}
|
||
}
|
||
|
||
bool handled = false;
|
||
elm::Element* parentElement = currentFocus;
|
||
|
||
while (!handled && parentElement) {
|
||
handled = parentElement->onClick(keysDown) || parentElement->handleInput(keysDown, keysHeld, touchPos, joyStickPosLeft, joyStickPosRight);
|
||
parentElement = parentElement->getParent();
|
||
}
|
||
|
||
if (currentGui != this->getCurrentGui()) return;
|
||
|
||
handled |= currentGui->handleInput(keysDown, keysHeld, touchPos, joyStickPosLeft, joyStickPosRight);
|
||
|
||
|
||
// Navigational boundary cases for handling wrapping
|
||
static bool lastDirectionPressed = true;
|
||
const bool directionPressed = ((keysHeld & KEY_UP) || (keysHeld & KEY_DOWN) || (keysHeld & KEY_LEFT) || (keysHeld & KEY_RIGHT));
|
||
|
||
if (!directionPressed && lastDirectionPressed)
|
||
tsl::elm::s_directionalKeyReleased.store(true, std::memory_order_release);
|
||
else if (directionPressed && lastDirectionPressed)
|
||
tsl::elm::s_directionalKeyReleased.store(false, std::memory_order_release);
|
||
|
||
lastDirectionPressed = directionPressed;
|
||
|
||
const float currentScrollVelocity = tsl::elm::s_currentScrollVelocity.load(std::memory_order_acquire);
|
||
|
||
if (hasScrolled) {
|
||
const bool singleArrowKeyPress = ((keysHeld & KEY_UP) != 0) + ((keysHeld & KEY_DOWN) != 0) + ((keysHeld & KEY_LEFT) != 0) + ((keysHeld & KEY_RIGHT) != 0) == 1 && !(keysHeld & ~((currentScrollVelocity != 0.0f ? KEY_A | KEY_UP : KEY_UP) | KEY_DOWN | KEY_LEFT | KEY_RIGHT) & ALL_KEYS_MASK);
|
||
|
||
if (singleArrowKeyPress) {
|
||
// const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
buttonPressTime_ns = currentTime_ns;
|
||
lastKeyEventTime_ns = currentTime_ns;
|
||
hasScrolled = false;
|
||
isNavigatingBackwards.store(false, std::memory_order_release);
|
||
}
|
||
} else {
|
||
if (!touchDetected && !oldTouchDetected && !handled && currentFocus && !ult::stillTouching.load(std::memory_order_acquire) && !interpreterIsRunning) {
|
||
static bool shouldShake = true;
|
||
const bool singleArrowKeyPress = ((keysHeld & KEY_UP) != 0) + ((keysHeld & KEY_DOWN) != 0) + ((keysHeld & KEY_LEFT) != 0) + ((keysHeld & KEY_RIGHT) != 0) == 1 && !(keysHeld & ~((currentScrollVelocity != 0.0f ? KEY_A | KEY_UP: KEY_UP) | KEY_DOWN | KEY_LEFT | KEY_RIGHT) & ALL_KEYS_MASK);
|
||
|
||
if (singleArrowKeyPress) {
|
||
//const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
|
||
if (keysDown) {
|
||
buttonPressTime_ns = currentTime_ns;
|
||
lastKeyEventTime_ns = currentTime_ns;
|
||
singlePressHandled = false;
|
||
// Immediate single press action
|
||
if (keysHeld & KEY_UP && !(keysHeld & ~KEY_UP & ALL_KEYS_MASK))
|
||
currentGui->requestFocus(topElement, FocusDirection::Up, shouldShake);
|
||
else if (keysHeld & KEY_DOWN && !(keysHeld & ~KEY_DOWN & ALL_KEYS_MASK)) {
|
||
currentGui->requestFocus(currentFocus->getParent(), FocusDirection::Down, shouldShake);
|
||
//isTopElement = false;
|
||
}
|
||
else if (keysHeld & KEY_LEFT && !(keysHeld & ~KEY_LEFT & ALL_KEYS_MASK))
|
||
currentGui->requestFocus(currentFocus->getParent(), FocusDirection::Left, shouldShake);
|
||
else if (keysHeld & KEY_RIGHT && !(keysHeld & ~KEY_RIGHT & ALL_KEYS_MASK))
|
||
currentGui->requestFocus(currentFocus->getParent(), FocusDirection::Right, shouldShake);
|
||
}
|
||
|
||
if (keysHeld & ~KEY_DOWN & ~KEY_UP & ~KEY_LEFT & ~KEY_RIGHT & ALL_KEYS_MASK) // reset
|
||
buttonPressTime_ns = currentTime_ns;
|
||
|
||
const u64 durationSincePress_ns = currentTime_ns - buttonPressTime_ns;
|
||
const u64 durationSinceLastEvent_ns = currentTime_ns - lastKeyEventTime_ns;
|
||
|
||
if (!singlePressHandled && durationSincePress_ns >= CLICK_THRESHOLD_NS) {
|
||
singlePressHandled = true;
|
||
}
|
||
|
||
if (!tsl::elm::isTableScrolling.load(std::memory_order_acquire)) {
|
||
// Calculate transition factor (t) from 0 to 1 based on how far we are from the transition point
|
||
static constexpr u64 transitionPoint_ns = 2000000000ULL; // 2000ms in nanoseconds
|
||
static constexpr u64 initialInterval_ns = 67000000ULL; // 67ms in nanoseconds
|
||
static constexpr u64 shortInterval_ns = 10000000ULL; // 10ms in nanoseconds
|
||
|
||
const float t = (durationSincePress_ns >= transitionPoint_ns) ? 1.0f :
|
||
(float)durationSincePress_ns / (float)transitionPoint_ns;
|
||
// Smooth transition between intervals using linear interpolation
|
||
keyEventInterval_ns = ((1.0f - t) * initialInterval_ns + t * shortInterval_ns);
|
||
} else {
|
||
// Table scrolling - faster timing
|
||
static constexpr u64 transitionPoint_ns = 200000000ULL; // 300ms (faster transition)
|
||
static constexpr u64 initialInterval_ns = 33000000ULL; // 33ms (faster initial)
|
||
static constexpr u64 shortInterval_ns = 5000000ULL; // 5ms (faster sustained)
|
||
|
||
const float t = (durationSincePress_ns >= transitionPoint_ns) ? 1.0f :
|
||
(float)durationSincePress_ns / (float)transitionPoint_ns;
|
||
// Smooth transition between intervals using linear interpolation
|
||
keyEventInterval_ns = ((1.0f - t) * initialInterval_ns + t * shortInterval_ns);
|
||
}
|
||
|
||
|
||
|
||
if (singlePressHandled && durationSinceLastEvent_ns >= keyEventInterval_ns) {
|
||
lastKeyEventTime_ns = currentTime_ns;
|
||
if (keysHeld & KEY_UP && !(keysHeld & ~((currentScrollVelocity != 0.0f ? KEY_A | KEY_UP: KEY_UP)) & ALL_KEYS_MASK))
|
||
currentGui->requestFocus(topElement, FocusDirection::Up, false);
|
||
else if (keysHeld & KEY_DOWN && !(keysHeld & ~((currentScrollVelocity != 0.0f ? KEY_A | KEY_DOWN: KEY_DOWN)) & ALL_KEYS_MASK)) {
|
||
currentGui->requestFocus(currentFocus->getParent(), FocusDirection::Down, false);
|
||
//isTopElement = false;
|
||
}
|
||
else if (keysHeld & KEY_LEFT && !(keysHeld & ~KEY_LEFT & ALL_KEYS_MASK))
|
||
currentGui->requestFocus(currentFocus->getParent(), FocusDirection::Left, false);
|
||
else if (keysHeld & KEY_RIGHT && !(keysHeld & ~KEY_RIGHT & ALL_KEYS_MASK))
|
||
currentGui->requestFocus(currentFocus->getParent(), FocusDirection::Right, false);
|
||
}
|
||
#if !IS_STATUS_MONITOR_DIRECTIVE
|
||
} else {
|
||
buttonPressTime_ns = lastKeyEventTime_ns = currentTime_ns;
|
||
#endif
|
||
}
|
||
}
|
||
}
|
||
|
||
#if !IS_STATUS_MONITOR_DIRECTIVE
|
||
if (!touchDetected && !interpreterIsRunning && topElement) {
|
||
#else
|
||
if (!disableJumpTo && !touchDetected && !interpreterIsRunning && topElement) {
|
||
#endif
|
||
// Shared constants used by ZL/ZR buttons
|
||
static constexpr u64 INITIAL_HOLD_THRESHOLD_NS = 400000000ULL;
|
||
static constexpr u64 HOLD_THRESHOLD_NS = 300000000ULL; // 300ms to start continuous
|
||
static constexpr u64 RAPID_CLICK_WINDOW_NS = 500000000ULL; // 500ms window for rapid clicking
|
||
static constexpr u64 RAPID_MODE_TIMEOUT_NS = 1000000000ULL; // 1s timeout to exit rapid mode
|
||
// Acceleration timing constants
|
||
static constexpr u64 ACCELERATION_POINT_NS = 1500000000ULL; // 1.5s transition point
|
||
static constexpr u64 INITIAL_INTERVAL_NS = 67000000ULL; // 67ms initial interval
|
||
static constexpr u64 FAST_INTERVAL_NS = 10000000ULL; // 10ms fast interval
|
||
|
||
//const u64 currentTime_ns = armTicksToNs(armGetSystemTick());
|
||
|
||
// Detect PHYSICAL key states (whether key is actually pressed)
|
||
const bool lKeyPressed = (keysHeld & KEY_L);
|
||
const bool rKeyPressed = (keysHeld & KEY_R);
|
||
const bool zlKeyPressed = (keysHeld & KEY_ZL);
|
||
const bool zrKeyPressed = (keysHeld & KEY_ZR);
|
||
|
||
// Detect if other keys are pressed (for preventing timer resets)
|
||
const bool notlKeyPressed = (keysHeld & ~KEY_L & ALL_KEYS_MASK);
|
||
const bool notrKeyPressed = (keysHeld & ~KEY_R & ALL_KEYS_MASK);
|
||
const bool notzlKeyPressed = (keysHeld & ~KEY_ZL & ALL_KEYS_MASK);
|
||
const bool notzrKeyPressed = (keysHeld & ~KEY_ZR & ALL_KEYS_MASK);
|
||
|
||
// Handle L button (simple jump to top on release, but not if held too long)
|
||
{
|
||
static bool lKeyWasPressed = false;
|
||
static bool lWasIsolated = false; // Track if L was isolated when first pressed
|
||
static u64 lButtonPressStart_ns = 0;
|
||
|
||
if (lKeyPressed) {
|
||
|
||
if (!lKeyWasPressed) {
|
||
// L key physically pressed for the first time (start timer)
|
||
lButtonPressStart_ns = currentTime_ns;
|
||
lWasIsolated = !notlKeyPressed; // Remember if it started isolated
|
||
}
|
||
// Don't reset timer if other keys are pressed after L was already held
|
||
lKeyWasPressed = true;
|
||
} else {
|
||
if (lKeyWasPressed) {
|
||
// L key physically released - only jump to top if was isolated when first pressed and not held too long
|
||
if (lWasIsolated && !(keysHeld & ~KEY_L & ALL_KEYS_MASK)) { // Was isolated initially and no other keys held at release
|
||
const u64 holdDuration = currentTime_ns - lButtonPressStart_ns;
|
||
if (holdDuration < INITIAL_HOLD_THRESHOLD_NS) {
|
||
jumpToTop.store(true, std::memory_order_release);
|
||
currentGui->requestFocus(topElement, FocusDirection::None);
|
||
}
|
||
}
|
||
}
|
||
lKeyWasPressed = false;
|
||
lWasIsolated = false;
|
||
}
|
||
}
|
||
|
||
// Handle R button (simple jump to bottom on release, but not if held too long)
|
||
{
|
||
static bool rKeyWasPressed = false;
|
||
static bool rWasIsolated = false; // Track if R was isolated when first pressed
|
||
static u64 rButtonPressStart_ns = 0;
|
||
|
||
if (rKeyPressed) {
|
||
if (!rKeyWasPressed) {
|
||
// R key physically pressed for the first time (start timer)
|
||
rButtonPressStart_ns = currentTime_ns;
|
||
rWasIsolated = !notrKeyPressed; // Remember if it started isolated
|
||
}
|
||
// Don't reset timer if other keys are pressed after R was already held
|
||
rKeyWasPressed = true;
|
||
} else {
|
||
if (rKeyWasPressed) {
|
||
// R key physically released - only jump to bottom if was isolated when first pressed and not held too long
|
||
if (rWasIsolated && !(keysHeld & ~KEY_R & ALL_KEYS_MASK)) { // Was isolated initially and no other keys held at release
|
||
const u64 holdDuration = currentTime_ns - rButtonPressStart_ns;
|
||
if (holdDuration < INITIAL_HOLD_THRESHOLD_NS) {
|
||
jumpToBottom.store(true, std::memory_order_release);
|
||
currentGui->requestFocus(topElement, FocusDirection::None);
|
||
}
|
||
}
|
||
}
|
||
rKeyWasPressed = false;
|
||
rWasIsolated = false;
|
||
}
|
||
}
|
||
|
||
// Handle ZL button (skip up with hold)
|
||
{
|
||
static u64 zlLastClickTime_ns = 0;
|
||
static bool zlKeyWasPressed = false;
|
||
static bool zlWasIsolated = false; // Track if ZL was isolated when first pressed
|
||
static bool zlInRapidClickMode = false;
|
||
static u64 zlFirstClickPressStart_ns = 0; // Track timing for first clicks only
|
||
|
||
// Check if we should exit rapid click mode due to timeout
|
||
if (zlInRapidClickMode && (currentTime_ns - zlLastClickTime_ns) > RAPID_MODE_TIMEOUT_NS) {
|
||
zlInRapidClickMode = false;
|
||
}
|
||
|
||
if (zlKeyPressed) {
|
||
if (!zlKeyWasPressed) {
|
||
// ZL key physically pressed for the first time
|
||
const u64 timeSinceLastClick = currentTime_ns - zlLastClickTime_ns;
|
||
|
||
zlWasIsolated = !notzlKeyPressed; // Remember if it started isolated
|
||
|
||
// Track press start time for first clicks (when not in rapid mode)
|
||
if (!zlInRapidClickMode) {
|
||
zlFirstClickPressStart_ns = currentTime_ns;
|
||
}
|
||
|
||
// Enter rapid click mode if clicking within window
|
||
if (timeSinceLastClick <= RAPID_CLICK_WINDOW_NS) {
|
||
zlInRapidClickMode = true;
|
||
}
|
||
|
||
// Only trigger immediately if in rapid click mode AND was isolated initially
|
||
if (zlInRapidClickMode && zlWasIsolated) {
|
||
skipUp.store(true, std::memory_order_release);
|
||
currentGui->requestFocus(topElement, FocusDirection::None);
|
||
zlLastClickTime_ns = currentTime_ns;
|
||
}
|
||
}
|
||
|
||
// Check for hold behavior - ONLY if in rapid click mode AND was isolated initially
|
||
if (zlInRapidClickMode && zlWasIsolated) {
|
||
static u64 zlButtonPressStart_ns = 0;
|
||
static u64 zlLastHoldTrigger_ns = 0;
|
||
static bool zlHoldTriggered = false;
|
||
|
||
// Initialize on new press
|
||
if (!zlKeyWasPressed) {
|
||
zlButtonPressStart_ns = currentTime_ns;
|
||
zlLastHoldTrigger_ns = currentTime_ns;
|
||
zlHoldTriggered = false;
|
||
}
|
||
|
||
const u64 holdDuration = currentTime_ns - zlButtonPressStart_ns;
|
||
|
||
if (holdDuration >= HOLD_THRESHOLD_NS) {
|
||
// Calculate dynamic interval based on hold duration (accelerating)
|
||
const float t = (holdDuration >= ACCELERATION_POINT_NS) ? 1.0f :
|
||
(float)holdDuration / (float)ACCELERATION_POINT_NS;
|
||
const u64 currentInterval = ((1.0f - t) * INITIAL_INTERVAL_NS + t * FAST_INTERVAL_NS);
|
||
|
||
const u64 timeSinceLastHoldTrigger = currentTime_ns - zlLastHoldTrigger_ns;
|
||
|
||
if (!zlHoldTriggered || timeSinceLastHoldTrigger >= currentInterval) {
|
||
// Trigger skip
|
||
skipUp.store(true, std::memory_order_release);
|
||
currentGui->requestFocus(topElement, FocusDirection::None);
|
||
zlHoldTriggered = true;
|
||
zlLastHoldTrigger_ns = currentTime_ns;
|
||
zlLastClickTime_ns = currentTime_ns; // Keep rapid mode active
|
||
}
|
||
}
|
||
}
|
||
|
||
zlKeyWasPressed = true;
|
||
} else {
|
||
if (zlKeyWasPressed) {
|
||
// ZL key physically released - only trigger if was isolated initially and no other keys held at release
|
||
if (!zlInRapidClickMode && zlWasIsolated && !(keysHeld & ~KEY_ZL & ALL_KEYS_MASK)) {
|
||
const u64 holdDuration = currentTime_ns - zlFirstClickPressStart_ns;
|
||
|
||
// Only trigger if not held too long
|
||
if (holdDuration < INITIAL_HOLD_THRESHOLD_NS) {
|
||
skipUp.store(true, std::memory_order_release);
|
||
currentGui->requestFocus(topElement, FocusDirection::None);
|
||
zlLastClickTime_ns = currentTime_ns;
|
||
zlInRapidClickMode = true; // Enter rapid mode after first release
|
||
}
|
||
}
|
||
}
|
||
zlKeyWasPressed = false;
|
||
zlWasIsolated = false;
|
||
}
|
||
}
|
||
|
||
// Handle ZR button (skip down with hold)
|
||
{
|
||
static u64 zrLastClickTime_ns = 0;
|
||
static bool zrKeyWasPressed = false;
|
||
static bool zrWasIsolated = false; // Track if ZR was isolated when first pressed
|
||
static bool zrInRapidClickMode = false;
|
||
static u64 zrFirstClickPressStart_ns = 0; // Track timing for first clicks only
|
||
|
||
// Check if we should exit rapid click mode due to timeout
|
||
if (zrInRapidClickMode && (currentTime_ns - zrLastClickTime_ns) > RAPID_MODE_TIMEOUT_NS) {
|
||
zrInRapidClickMode = false;
|
||
}
|
||
|
||
if (zrKeyPressed) {
|
||
if (!zrKeyWasPressed) {
|
||
// ZR key physically pressed for the first time
|
||
const u64 timeSinceLastClick = currentTime_ns - zrLastClickTime_ns;
|
||
|
||
zrWasIsolated = !notzrKeyPressed; // Remember if it started isolated
|
||
|
||
// Track press start time for first clicks (when not in rapid mode)
|
||
if (!zrInRapidClickMode) {
|
||
zrFirstClickPressStart_ns = currentTime_ns;
|
||
}
|
||
|
||
// Enter rapid click mode if clicking within window
|
||
if (timeSinceLastClick <= RAPID_CLICK_WINDOW_NS) {
|
||
zrInRapidClickMode = true;
|
||
}
|
||
|
||
// Only trigger immediately if in rapid click mode AND was isolated initially
|
||
if (zrInRapidClickMode && zrWasIsolated) {
|
||
skipDown.store(true, std::memory_order_release);
|
||
currentGui->requestFocus(topElement, FocusDirection::None);
|
||
zrLastClickTime_ns = currentTime_ns;
|
||
}
|
||
}
|
||
|
||
// Check for hold behavior - ONLY if in rapid click mode AND was isolated initially
|
||
if (zrInRapidClickMode && zrWasIsolated) {
|
||
static u64 zrButtonPressStart_ns = 0;
|
||
static u64 zrLastHoldTrigger_ns = 0;
|
||
static bool zrHoldTriggered = false;
|
||
|
||
// Initialize on new press
|
||
if (!zrKeyWasPressed) {
|
||
zrButtonPressStart_ns = currentTime_ns;
|
||
zrLastHoldTrigger_ns = currentTime_ns;
|
||
zrHoldTriggered = false;
|
||
}
|
||
|
||
const u64 holdDuration = currentTime_ns - zrButtonPressStart_ns;
|
||
|
||
if (holdDuration >= HOLD_THRESHOLD_NS) {
|
||
// Calculate dynamic interval based on hold duration (accelerating)
|
||
const float t = (holdDuration >= ACCELERATION_POINT_NS) ? 1.0f :
|
||
(float)holdDuration / (float)ACCELERATION_POINT_NS;
|
||
const u64 currentInterval = ((1.0f - t) * INITIAL_INTERVAL_NS + t * FAST_INTERVAL_NS);
|
||
|
||
const u64 timeSinceLastHoldTrigger = currentTime_ns - zrLastHoldTrigger_ns;
|
||
|
||
if (!zrHoldTriggered || timeSinceLastHoldTrigger >= currentInterval) {
|
||
// Trigger skip
|
||
skipDown.store(true, std::memory_order_release);
|
||
currentGui->requestFocus(topElement, FocusDirection::None);
|
||
zrHoldTriggered = true;
|
||
zrLastHoldTrigger_ns = currentTime_ns;
|
||
zrLastClickTime_ns = currentTime_ns; // Keep rapid mode active
|
||
}
|
||
}
|
||
}
|
||
|
||
zrKeyWasPressed = true;
|
||
} else {
|
||
if (zrKeyWasPressed) {
|
||
// ZR key physically released - only trigger if was isolated initially and no other keys held at release
|
||
if (!zrInRapidClickMode && zrWasIsolated && !(keysHeld & ~KEY_ZR & ALL_KEYS_MASK)) {
|
||
const u64 holdDuration = currentTime_ns - zrFirstClickPressStart_ns;
|
||
|
||
// Only trigger if not held too long
|
||
if (holdDuration < INITIAL_HOLD_THRESHOLD_NS) {
|
||
skipDown.store(true, std::memory_order_release);
|
||
currentGui->requestFocus(topElement, FocusDirection::None);
|
||
zrLastClickTime_ns = currentTime_ns;
|
||
zrInRapidClickMode = true; // Enter rapid mode after first release
|
||
}
|
||
}
|
||
}
|
||
zrKeyWasPressed = false;
|
||
zrWasIsolated = false;
|
||
}
|
||
}
|
||
}
|
||
|
||
//if (keysDown & KEY_ZL) {
|
||
// //while (tsl::notification && tsl::notification->isActive()) {
|
||
// // tsl::notification->update(true, true); // No file ops, allow state transitions
|
||
// // svcSleepThread(10'000'000); // 1ms sleep
|
||
// //}
|
||
// if (notification)
|
||
// notification->forceShutdown();
|
||
//}
|
||
|
||
|
||
if (!touchDetected && oldTouchDetected && currentGui && topElement) {
|
||
topElement->onTouch(elm::TouchEvent::Release, oldTouchPos.x, oldTouchPos.y, oldTouchPos.x, oldTouchPos.y, initialTouchPos.x, initialTouchPos.y);
|
||
}
|
||
|
||
// Cache common calculations
|
||
// Use consistent edge padding equal to halfGap (matching drawing code)
|
||
const float edgePadding = ult::halfGap.load(std::memory_order_acquire) - 5;
|
||
const float buttonStartX = edgePadding;
|
||
|
||
// Calculate button positions matching the drawing code
|
||
const float backLeftEdge = buttonStartX + ult::layerEdge;
|
||
const float backRightEdge = backLeftEdge + ult::backWidth.load(std::memory_order_acquire);
|
||
const float selectLeftEdge = backRightEdge;
|
||
const float selectRightEdge = selectLeftEdge + ult::selectWidth.load(std::memory_order_acquire);
|
||
const float nextPageLeftEdge = ult::noClickableItems.load(std::memory_order_acquire) ? backRightEdge : selectRightEdge;
|
||
const float nextPageRightEdge = nextPageLeftEdge + ult::nextPageWidth.load(std::memory_order_acquire);
|
||
|
||
const float menuRightEdge = 245.0f + ult::layerEdge - 13;
|
||
const u32 footerY = cfg::FramebufferHeight - 73U + 1;
|
||
static std::vector<bool> lastSimulatedTouch = {false, false, false, false};
|
||
|
||
// Touch region calculations
|
||
const bool backTouched = (touchPos.x >= backLeftEdge && touchPos.x < backRightEdge && touchPos.y > footerY) &&
|
||
(initialTouchPos.x >= backLeftEdge && initialTouchPos.x < backRightEdge && initialTouchPos.y > footerY);
|
||
|
||
const bool selectTouched = !ult::noClickableItems.load(std::memory_order_acquire) &&
|
||
(touchPos.x >= selectLeftEdge && touchPos.x < selectRightEdge && touchPos.y > footerY) &&
|
||
(initialTouchPos.x >= selectLeftEdge && initialTouchPos.x < selectRightEdge && initialTouchPos.y > footerY);
|
||
|
||
const bool nextPageTouched = (touchPos.x >= nextPageLeftEdge && touchPos.x < nextPageRightEdge && touchPos.y > footerY) &&
|
||
(initialTouchPos.x >= nextPageLeftEdge && initialTouchPos.x < nextPageRightEdge && initialTouchPos.y > footerY);
|
||
|
||
const bool menuTouched = (touchPos.x > ult::layerEdge+7U && touchPos.x <= menuRightEdge && touchPos.y > 10U && touchPos.y <= 83U) &&
|
||
(initialTouchPos.x > ult::layerEdge+7U && initialTouchPos.x <= menuRightEdge && initialTouchPos.y > 10U && initialTouchPos.y <= 83U);
|
||
|
||
ult::touchingBack.store(backTouched, std::memory_order_release);
|
||
ult::touchingSelect.store(selectTouched, std::memory_order_release);
|
||
ult::touchingNextPage.store(nextPageTouched, std::memory_order_release);
|
||
ult::touchingMenu.store(menuTouched, std::memory_order_release);
|
||
|
||
if (touchDetected) {
|
||
// Update lastSimulatedTouch with current touch states
|
||
lastSimulatedTouch = {
|
||
backTouched,
|
||
selectTouched,
|
||
nextPageTouched,
|
||
menuTouched
|
||
};
|
||
|
||
ult::interruptedTouch.store(((keysHeld & ALL_KEYS_MASK) != 0), std::memory_order_release);
|
||
|
||
const u32 xDistance = std::abs(static_cast<s32>(initialTouchPos.x) - static_cast<s32>(touchPos.x));
|
||
const u32 yDistance = std::abs(static_cast<s32>(initialTouchPos.y) - static_cast<s32>(touchPos.y));
|
||
|
||
const bool isScroll = (xDistance * xDistance + yDistance * yDistance) > 1000;
|
||
if (isScroll) {
|
||
elm::Element::setInputMode(InputMode::TouchScroll);
|
||
touchEvent = elm::TouchEvent::Scroll;
|
||
} else {
|
||
if (touchEvent != elm::TouchEvent::Scroll) {
|
||
touchEvent = elm::TouchEvent::Hold;
|
||
}
|
||
}
|
||
|
||
if (!oldTouchDetected) {
|
||
initialTouchPos = touchPos;
|
||
elm::Element::setInputMode(InputMode::Touch);
|
||
if (!interpreterIsRunning) {
|
||
ult::touchInBounds = (initialTouchPos.y <= footerY && initialTouchPos.y > 73U &&
|
||
initialTouchPos.x <= ult::layerEdge + cfg::FramebufferWidth - 30U &&
|
||
initialTouchPos.x > 40U + ult::layerEdge);
|
||
if (ult::touchInBounds) currentGui->removeFocus();
|
||
}
|
||
touchEvent = elm::TouchEvent::Touch;
|
||
}
|
||
|
||
if (currentGui && topElement && !interpreterIsRunning) {
|
||
topElement->onTouch(touchEvent, touchPos.x, touchPos.y, oldTouchPos.x, oldTouchPos.y, initialTouchPos.x, initialTouchPos.y);
|
||
if (touchPos.x > 40U + ult::layerEdge && touchPos.x <= cfg::FramebufferWidth - 30U + ult::layerEdge &&
|
||
touchPos.y > 73U && touchPos.y <= footerY) {
|
||
currentGui->removeFocus();
|
||
}
|
||
}
|
||
|
||
oldTouchPos = touchPos;
|
||
if ((touchPos.x < ult::layerEdge || touchPos.x > cfg::FramebufferWidth + ult::layerEdge) && tsl::elm::Element::getInputMode() == tsl::InputMode::Touch) {
|
||
oldTouchPos = { 0 };
|
||
initialTouchPos = { 0 };
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (FullMode && !deactivateOriginalFooter) {
|
||
this->hide();
|
||
}
|
||
#else
|
||
this->hide();
|
||
#endif
|
||
}
|
||
ult::stillTouching.store(true, std::memory_order_release);
|
||
} else {
|
||
// Process touch release using stored touch states - no need to recalculate boundaries
|
||
for (int i = 0; i < 4; ++i) {
|
||
if (lastSimulatedTouch[i]) {
|
||
if (!ult::interruptedTouch.load(std::memory_order_acquire) && !interpreterIsRunning) {
|
||
switch (i) {
|
||
case 0: // Back button
|
||
ult::simulatedBack.store(true, std::memory_order_release);
|
||
break;
|
||
case 1: // Select button
|
||
ult::simulatedSelect.store(true, std::memory_order_release);
|
||
break;
|
||
case 2: // Next page button
|
||
ult::simulatedNextPage.store(true, std::memory_order_release);
|
||
break;
|
||
case 3: // Menu button
|
||
ult::simulatedMenu.store(true, std::memory_order_release);
|
||
break;
|
||
}
|
||
} else if (interpreterIsRunning) {
|
||
switch (i) {
|
||
case 0: // Back button when interpreter is running
|
||
this->hide();
|
||
break;
|
||
case 1: // Select button when interpreter is running
|
||
ult::externalAbortCommands.store(true, std::memory_order_release);
|
||
break;
|
||
// cases 2 and 3 don't have interpreter running logic in original code
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
// Update lastSimulatedTouch with current touch states
|
||
lastSimulatedTouch = {
|
||
false,
|
||
false,
|
||
false,
|
||
false
|
||
};
|
||
|
||
elm::Element::setInputMode(InputMode::Controller);
|
||
|
||
oldTouchPos = { 0 };
|
||
initialTouchPos = { 0 };
|
||
touchEvent = elm::TouchEvent::None;
|
||
ult::stillTouching.store(false, std::memory_order_release);
|
||
ult::interruptedTouch.store(false, std::memory_order_release);
|
||
}
|
||
|
||
oldTouchDetected = touchDetected;
|
||
oldTouchEvent = touchEvent;
|
||
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Clears the screen
|
||
*
|
||
*/
|
||
void clearScreen() {
|
||
auto& renderer = gfx::Renderer::get();
|
||
|
||
renderer.startFrame();
|
||
renderer.clearScreen();
|
||
renderer.endFrame();
|
||
}
|
||
|
||
/**
|
||
* @brief Reset hide and close flags that were previously set by \ref Overlay::close() or \ref Overlay::hide()
|
||
*
|
||
*/
|
||
void resetFlags() {
|
||
this->m_shouldHide = false;
|
||
this->m_shouldClose = false;
|
||
this->m_shouldCloseAfter = false;
|
||
}
|
||
|
||
/**
|
||
* @brief Disables the next animation that would play
|
||
*
|
||
*/
|
||
void disableNextAnimation() {
|
||
this->m_disableNextAnimation = true;
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Changes to a different Gui
|
||
*
|
||
* @param gui Gui to change to
|
||
* @return Reference to the Gui
|
||
*/
|
||
std::unique_ptr<tsl::Gui>& changeTo(std::unique_ptr<tsl::Gui>&& gui, bool clearGlyphCache = false) {
|
||
if (this->m_guiStack.top() != nullptr && this->m_guiStack.top()->m_focusedElement != nullptr)
|
||
this->m_guiStack.top()->m_focusedElement->resetClickAnimation();
|
||
|
||
isNavigatingBackwards.store(false, std::memory_order_release);
|
||
|
||
// cache frame for forward rendering using external list method (to be implemented)
|
||
|
||
// Create the top element of the new Gui
|
||
gui->m_topElement = gui->createUI();
|
||
|
||
|
||
// Push the new Gui onto the stack
|
||
this->m_guiStack.push(std::move(gui));
|
||
//if (clearGlyphCache)
|
||
// tsl::gfx::FontManager::clearCache();
|
||
return this->m_guiStack.top();
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Creates a new Gui and changes to it
|
||
*
|
||
* @tparam G Gui to create
|
||
* @tparam Args Arguments to pass to the Gui
|
||
* @param args Arguments to pass to the Gui
|
||
* @return Reference to the newly created Gui
|
||
*/
|
||
// Template version without clearGlyphCache (for backward compatibility)
|
||
template<typename G, typename ...Args>
|
||
std::unique_ptr<tsl::Gui>& changeTo(Args&&... args) {
|
||
return this->changeTo(std::make_unique<G>(std::forward<Args>(args)...), false);
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Swaps to a different Gui
|
||
*
|
||
* @param gui Gui to change to
|
||
* @return Reference to the Gui
|
||
*/
|
||
std::unique_ptr<tsl::Gui>& swapTo(std::unique_ptr<tsl::Gui>&& gui, u32 count = 1) {
|
||
//isNavigatingBackwards = true;
|
||
|
||
isNavigatingBackwards.store(true, std::memory_order_release);
|
||
|
||
// Clamp count to available stack size to prevent underflow
|
||
const u32 actualCount = std::min(count, static_cast<u32>(this->m_guiStack.size()));
|
||
|
||
if (actualCount > 1) {
|
||
tsl::elm::skipDeconstruction.store(true, std::memory_order_release);
|
||
// Pop the specified number of GUIs
|
||
for (u32 i = 0; i < actualCount; ++i) {
|
||
this->m_guiStack.pop();
|
||
}
|
||
tsl::elm::skipDeconstruction.store(false, std::memory_order_release);
|
||
} else {
|
||
this->m_guiStack.pop();
|
||
}
|
||
|
||
|
||
|
||
if (this->m_guiStack.top() != nullptr && this->m_guiStack.top()->m_focusedElement != nullptr)
|
||
this->m_guiStack.top()->m_focusedElement->resetClickAnimation();
|
||
|
||
isNavigatingBackwards.store(false, std::memory_order_release);
|
||
|
||
// cache frame for forward rendering using external list method (to be implemented)
|
||
|
||
// Create the top element of the new Gui
|
||
gui->m_topElement = gui->createUI();
|
||
|
||
|
||
// Push the new Gui onto the stack
|
||
this->m_guiStack.push(std::move(gui));
|
||
//if (clearGlyphCache)
|
||
// tsl::gfx::FontManager::clearCache();
|
||
return this->m_guiStack.top();
|
||
}
|
||
|
||
/**
|
||
* @brief Creates a new Gui and changes to it
|
||
*
|
||
* @tparam G Gui to create
|
||
* @tparam Args Arguments to pass to the Gui
|
||
* @param args Arguments to pass to the Gui
|
||
* @return Reference to the newly created Gui
|
||
*/
|
||
// Template version without clearGlyphCache (for backward compatibility)
|
||
template<typename G, typename ...Args>
|
||
std::unique_ptr<tsl::Gui>& swapTo(SwapDepth depth, Args&&... args) {
|
||
return this->swapTo(std::make_unique<G>(std::forward<Args>(args)...), depth.value);
|
||
}
|
||
|
||
template<typename G, typename ...Args>
|
||
std::unique_ptr<tsl::Gui>& swapTo(Args&&... args) {
|
||
return this->swapTo(std::make_unique<G>(std::forward<Args>(args)...), 1);
|
||
}
|
||
|
||
/**
|
||
* @brief Pops the top Gui(s) from the stack and goes back count number of times
|
||
* @param count Number of Guis to pop from the stack (default: 1)
|
||
* @note The Overlay gets closed once there are no more Guis on the stack
|
||
*/
|
||
void goBack(u32 count = 1) {
|
||
// If there is exactly one GUI and an active notification, handle that first
|
||
if (this->m_guiStack.size() == 1 && notification && notification->isActive()) {
|
||
this->close();
|
||
return;
|
||
}
|
||
|
||
isNavigatingBackwards.store(true, std::memory_order_release);
|
||
|
||
// Clamp count to available stack size to prevent underflow
|
||
const u32 actualCount = std::min(count, static_cast<u32>(this->m_guiStack.size()));
|
||
|
||
// Special case: if we don't close on exit and popping everything would leave us with 0 or 1 GUI
|
||
if (!this->m_closeOnExit && this->m_guiStack.size() <= actualCount) {
|
||
this->hide();
|
||
return;
|
||
}
|
||
|
||
if (actualCount > 1)
|
||
tsl::elm::skipDeconstruction.store(true, std::memory_order_release);
|
||
|
||
// Pop the specified number of GUIs
|
||
for (u32 i = 0; i < actualCount && !this->m_guiStack.empty(); ++i) {
|
||
this->m_guiStack.pop();
|
||
}
|
||
|
||
if (tsl::elm::skipDeconstruction.load(std::memory_order_acquire))
|
||
tsl::elm::skipDeconstruction.store(false, std::memory_order_release);
|
||
|
||
// Close overlay if stack is empty
|
||
if (this->m_guiStack.empty()) {
|
||
this->close();
|
||
}
|
||
}
|
||
|
||
void pop(u32 count = 1) {
|
||
isNavigatingBackwards.store(true, std::memory_order_release);
|
||
|
||
// Clamp count to available stack size to prevent underflow
|
||
const u32 actualCount = std::min(count, static_cast<u32>(this->m_guiStack.size()));
|
||
|
||
if (actualCount > 1) {
|
||
tsl::elm::skipDeconstruction.store(true, std::memory_order_release);
|
||
// Pop the specified number of GUIs
|
||
for (u32 i = 0; i < actualCount; ++i) {
|
||
this->m_guiStack.pop();
|
||
}
|
||
tsl::elm::skipDeconstruction.store(false, std::memory_order_release);
|
||
} else {
|
||
this->m_guiStack.pop();
|
||
}
|
||
}
|
||
|
||
|
||
|
||
template<typename G, typename ...Args>
|
||
friend std::unique_ptr<tsl::Gui>& changeTo(Args&&... args);
|
||
template<typename G, typename ...Args>
|
||
friend std::unique_ptr<tsl::Gui>& swapTo(Args&&... args);
|
||
|
||
template<typename G, typename ...Args>
|
||
friend std::unique_ptr<tsl::Gui>& swapTo(SwapDepth depth, Args&&... args);
|
||
|
||
friend void goBack(u32 count);
|
||
friend void pop(u32 count);
|
||
|
||
template<typename, tsl::impl::LaunchFlags>
|
||
friend int loop(int argc, char** argv);
|
||
|
||
friend class tsl::Gui;
|
||
};
|
||
|
||
|
||
namespace impl {
|
||
static constexpr const char* TESLA_CONFIG_FILE = "/config/tesla/config.ini";
|
||
static constexpr const char* ULTRAHAND_CONFIG_FILE = "/config/ultrahand/config.ini";
|
||
|
||
/**
|
||
* @brief Data shared between the different ult::renderThreads
|
||
*
|
||
*/
|
||
struct SharedThreadData {
|
||
std::atomic<bool> running = false;
|
||
|
||
Event comboEvent = { 0 };
|
||
|
||
std::atomic<bool> overlayOpen = false;
|
||
|
||
std::mutex dataMutex;
|
||
u64 keysDown = 0;
|
||
u64 keysDownPending = 0;
|
||
u64 keysHeld = 0;
|
||
HidTouchScreenState touchState = { 0 };
|
||
HidAnalogStickState joyStickPosLeft = { 0 }, joyStickPosRight = { 0 };
|
||
};
|
||
|
||
|
||
/**
|
||
* @brief Extract values from Tesla settings file
|
||
*
|
||
*/
|
||
static void parseOverlaySettings() {
|
||
hlp::ini::IniData parsedConfig = hlp::ini::readOverlaySettings(ULTRAHAND_CONFIG_FILE);
|
||
|
||
u64 decodedKeys = hlp::comboStringToKeys(parsedConfig[ult::ULTRAHAND_PROJECT_NAME][ult::KEY_COMBO_STR]); // CUSTOM MODIFICATION
|
||
if (decodedKeys)
|
||
tsl::cfg::launchCombo = decodedKeys;
|
||
else {
|
||
parsedConfig = hlp::ini::readOverlaySettings(TESLA_CONFIG_FILE);
|
||
decodedKeys = hlp::comboStringToKeys(parsedConfig["tesla"][ult::KEY_COMBO_STR]);
|
||
if (decodedKeys)
|
||
tsl::cfg::launchCombo = decodedKeys;
|
||
}
|
||
|
||
//#if USING_WIDGET_DIRECTIVE
|
||
ult::datetimeFormat = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["datetime_format"]; // read datetime_format
|
||
ult::removeQuotes(ult::datetimeFormat);
|
||
if (ult::datetimeFormat.empty()) {
|
||
ult::datetimeFormat = ult::DEFAULT_DT_FORMAT;
|
||
ult::removeQuotes(ult::datetimeFormat);
|
||
}
|
||
|
||
|
||
std::string tempStr;
|
||
|
||
tempStr = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["hide_clock"];
|
||
ult::removeQuotes(tempStr);
|
||
ult::hideClock = tempStr != ult::FALSE_STR;
|
||
|
||
tempStr = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["hide_battery"];
|
||
ult::removeQuotes(tempStr);
|
||
ult::hideBattery = tempStr != ult::FALSE_STR;
|
||
|
||
tempStr = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["hide_pcb_temp"];
|
||
ult::removeQuotes(tempStr);
|
||
ult::hidePCBTemp = tempStr != ult::FALSE_STR;
|
||
|
||
tempStr = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["hide_soc_temp"];
|
||
ult::removeQuotes(tempStr);
|
||
ult::hideSOCTemp = tempStr != ult::FALSE_STR;
|
||
|
||
tempStr = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["dynamic_widget_colors"];
|
||
ult::removeQuotes(tempStr);
|
||
ult::dynamicWidgetColors = tempStr != ult::FALSE_STR;
|
||
|
||
tempStr = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["hide_widget_backdrop"];
|
||
ult::removeQuotes(tempStr);
|
||
ult::hideWidgetBackdrop = tempStr != ult::FALSE_STR;
|
||
|
||
tempStr = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["center_widget_alignment"];
|
||
ult::removeQuotes(tempStr);
|
||
ult::centerWidgetAlignment = tempStr != ult::FALSE_STR;
|
||
|
||
tempStr = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["extended_widget_backdrop"];
|
||
ult::removeQuotes(tempStr);
|
||
ult::extendedWidgetBackdrop = tempStr != ult::FALSE_STR;
|
||
|
||
tempStr = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["dynamic_logo"];
|
||
ult::removeQuotes(tempStr);
|
||
ult::useDynamicLogo = tempStr != ult::FALSE_STR;
|
||
|
||
tempStr = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["selection_bg"];
|
||
ult::removeQuotes(tempStr);
|
||
ult::useSelectionBG = tempStr != ult::FALSE_STR;
|
||
|
||
tempStr = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["selection_text"];
|
||
ult::removeQuotes(tempStr);
|
||
ult::useSelectionText = tempStr != ult::FALSE_STR;
|
||
|
||
tempStr = parsedConfig[ult::ULTRAHAND_PROJECT_NAME]["selection_value"];
|
||
ult::removeQuotes(tempStr);
|
||
ult::useSelectionValue = tempStr != ult::FALSE_STR;
|
||
|
||
//#endif
|
||
|
||
}
|
||
|
||
/**
|
||
* @brief Update and save launch combo keys
|
||
*
|
||
* @param keys the new combo keys
|
||
*/
|
||
[[maybe_unused]] static void updateCombo(u64 keys) {
|
||
tsl::cfg::launchCombo = keys;
|
||
hlp::ini::updateOverlaySettings({
|
||
{ ult::TESLA_STR, { // CUSTOM MODIFICATION
|
||
{ ult::KEY_COMBO_STR , tsl::hlp::keysToComboString(keys) }
|
||
}}
|
||
}, TESLA_CONFIG_FILE);
|
||
hlp::ini::updateOverlaySettings({
|
||
{ ult::ULTRAHAND_PROJECT_NAME, { // CUSTOM MODIFICATION
|
||
{ ult::KEY_COMBO_STR , tsl::hlp::keysToComboString(keys) }
|
||
}}
|
||
}, ULTRAHAND_CONFIG_FILE);
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Background event polling loop thread
|
||
*
|
||
* @param args Used to pass in a pointer to a \ref SharedThreadData struct
|
||
*/
|
||
static void backgroundEventPoller(void *args) {
|
||
|
||
tsl::hlp::loadEntryKeyCombos();
|
||
ult::launchingOverlay.store(false, std::memory_order_release);
|
||
|
||
SharedThreadData *shData = static_cast<SharedThreadData*>(args);
|
||
|
||
// To prevent focus glitchout, close the overlay immediately when the home button gets pressed
|
||
Event homeButtonPressEvent = {};
|
||
hidsysAcquireHomeButtonEventHandle(&homeButtonPressEvent, false);
|
||
eventClear(&homeButtonPressEvent);
|
||
tsl::hlp::ScopeGuard homeButtonEventGuard([&] { eventClose(&homeButtonPressEvent); });
|
||
|
||
// To prevent focus glitchout, close the overlay immediately when the power button gets pressed
|
||
Event powerButtonPressEvent = {};
|
||
hidsysAcquireSleepButtonEventHandle(&powerButtonPressEvent, false);
|
||
eventClear(&powerButtonPressEvent);
|
||
tsl::hlp::ScopeGuard powerButtonEventGuard([&] { eventClose(&powerButtonPressEvent); });
|
||
|
||
|
||
// For handling screenshots color alpha
|
||
Event captureButtonPressEvent = {};
|
||
hidsysAcquireCaptureButtonEventHandle(&captureButtonPressEvent, false);
|
||
eventClear(&captureButtonPressEvent);
|
||
hidsysAcquireCaptureButtonEventHandle(&captureButtonPressEvent, false);
|
||
eventClear(&captureButtonPressEvent);
|
||
tsl::hlp::ScopeGuard captureButtonEventGuard([&] { eventClose(&captureButtonPressEvent); });
|
||
|
||
// Parse Tesla settings
|
||
impl::parseOverlaySettings();
|
||
|
||
|
||
// Allow only Player 1 and handheld mode
|
||
HidNpadIdType id_list[2] = { HidNpadIdType_No1, HidNpadIdType_Handheld };
|
||
|
||
// Configure HID system to only listen to these IDs
|
||
hidSetSupportedNpadIdType(id_list, 2);
|
||
|
||
// Configure input for up to 2 supported controllers (P1 + Handheld)
|
||
padConfigureInput(2, HidNpadStyleSet_NpadStandard | HidNpadStyleTag_NpadSystemExt);
|
||
|
||
// Initialize separate pad states for both controllers
|
||
PadState pad_p1;
|
||
PadState pad_handheld;
|
||
padInitialize(&pad_p1, HidNpadIdType_No1);
|
||
padInitialize(&pad_handheld, HidNpadIdType_Handheld);
|
||
|
||
// Touch screen init
|
||
hidInitializeTouchScreen();
|
||
|
||
// Clear any stale input from both controllers
|
||
padUpdate(&pad_p1);
|
||
padUpdate(&pad_handheld);
|
||
|
||
enum WaiterObject {
|
||
WaiterObject_HomeButton,
|
||
WaiterObject_PowerButton,
|
||
WaiterObject_CaptureButton,
|
||
WaiterObject_Count
|
||
};
|
||
|
||
// Construct waiter
|
||
Waiter objects[3] = {
|
||
[WaiterObject_HomeButton] = waiterForEvent(&homeButtonPressEvent),
|
||
[WaiterObject_PowerButton] = waiterForEvent(&powerButtonPressEvent),
|
||
[WaiterObject_CaptureButton] = waiterForEvent(&captureButtonPressEvent),
|
||
};
|
||
|
||
u64 currentTouchTick = 0;
|
||
auto lastTouchX = 0;
|
||
auto lastTouchY = 0;
|
||
|
||
// Preset touch boundaries
|
||
static constexpr int SWIPE_RIGHT_BOUND = 16; // 16 + 80
|
||
static constexpr int SWIPE_LEFT_BOUND = (1280 - 16);
|
||
static constexpr u64 TOUCH_THRESHOLD_NS = 150'000'000ULL; // 150ms in nanoseconds
|
||
static constexpr u64 FAST_SWAP_THRESHOLD_NS = 150'000'000ULL;
|
||
|
||
// Global underscan monitoring - run at most once every 300ms
|
||
static auto lastUnderscanPixels = std::make_pair(0, 0);
|
||
static bool firstUnderscanCheck = true;
|
||
static u64 lastUnderscanCheckNs = 0; // store last execution in nanoseconds
|
||
static constexpr u64 UNDERSCAN_INTERVAL_NS = 300'000'000ULL; // 300ms in ns
|
||
|
||
s32 idx;
|
||
Result rc;
|
||
|
||
std::string currentTitleID;
|
||
|
||
u64 lastPollTick = 0;
|
||
u64 resetStartTick = armGetSystemTick();
|
||
const u64 startNs = armTicksToNs(resetStartTick);
|
||
|
||
ult::lastTitleID = ult::getTitleIdAsString();
|
||
|
||
//u64 elapsedTime_ns;
|
||
|
||
while (shData->running.load(std::memory_order_acquire)) {
|
||
|
||
const u64 nowTick = armGetSystemTick();
|
||
const u64 nowNs = armTicksToNs(nowTick);
|
||
|
||
|
||
|
||
// Scan for input changes from both controllers
|
||
padUpdate(&pad_p1);
|
||
padUpdate(&pad_handheld);
|
||
|
||
// Read in HID values
|
||
{
|
||
if (ult::launchingOverlay.load(std::memory_order_acquire))
|
||
break;
|
||
std::scoped_lock lock(shData->dataMutex);
|
||
if (ult::launchingOverlay.load(std::memory_order_acquire))
|
||
break;
|
||
|
||
// Poll Title ID every 1 seconds
|
||
if (!ult::resetForegroundCheck.load(std::memory_order_acquire)) {
|
||
const u64 elapsedNs = armTicksToNs(nowTick - lastPollTick);
|
||
if (elapsedNs >= 1'000'000'000ULL) {
|
||
lastPollTick = nowTick;
|
||
|
||
currentTitleID = ult::getTitleIdAsString();
|
||
if (currentTitleID != ult::lastTitleID) {
|
||
ult::lastTitleID = currentTitleID;
|
||
ult::resetForegroundCheck.store(true, std::memory_order_release);
|
||
resetStartTick = nowTick;
|
||
}
|
||
}
|
||
}
|
||
|
||
// If a reset is scheduled, trigger after 3.5s delay
|
||
if (ult::resetForegroundCheck.load(std::memory_order_acquire)) {
|
||
const u64 resetElapsedNs = armTicksToNs(nowTick - resetStartTick);
|
||
if (resetElapsedNs >= 3'500'000'000ULL) {
|
||
if (shData->overlayOpen && ult::currentForeground.load(std::memory_order_acquire)) {
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (!isMiniOrMicroMode)
|
||
hlp::requestForeground(true, false);
|
||
#else
|
||
hlp::requestForeground(true, false);
|
||
#endif
|
||
}
|
||
ult::resetForegroundCheck.store(false, std::memory_order_release);
|
||
}
|
||
}
|
||
|
||
if (firstUnderscanCheck || (nowNs - lastUnderscanCheckNs) >= UNDERSCAN_INTERVAL_NS) {
|
||
const auto currentUnderscanPixels = tsl::gfx::getUnderscanPixels();
|
||
|
||
if (firstUnderscanCheck || currentUnderscanPixels != lastUnderscanPixels) {
|
||
// Update layer dimensions without destroying state
|
||
tsl::gfx::Renderer::get().updateLayerSize();
|
||
|
||
lastUnderscanPixels = currentUnderscanPixels;
|
||
firstUnderscanCheck = false;
|
||
}
|
||
|
||
lastUnderscanCheckNs = nowNs;
|
||
}
|
||
|
||
//bool expected = true;
|
||
//if (fireNotificationEvent.compare_exchange_strong(expected, false, std::memory_order_acq_rel)) {
|
||
// if (ult::launchingOverlay.load(std::memory_order_acquire))
|
||
// return;
|
||
// eventFire(&shData->notificationEvent); // wake the loop
|
||
//}
|
||
|
||
// Process notification files every 300ms
|
||
{
|
||
std::lock_guard<std::mutex> jsonLock(notificationJsonMutex);
|
||
static u64 lastNotifCheck = 0;
|
||
|
||
if (armTicksToNs(nowTick - lastNotifCheck) >= 300'000'000ULL) {
|
||
lastNotifCheck = nowTick;
|
||
|
||
DIR* dir = opendir(ult::NOTIFICATIONS_PATH.c_str());
|
||
if (dir) {
|
||
|
||
if (ult::useNotifications) {
|
||
std::string filename, fullPath, prioStr;
|
||
|
||
struct dirent* entry;
|
||
|
||
const std::string& notifPath = ult::NOTIFICATIONS_PATH;
|
||
|
||
static std::vector<std::string> shownFiles;
|
||
|
||
// Structure to hold file info with creation time
|
||
struct NotificationFile {
|
||
std::string filename;
|
||
std::string fullPath;
|
||
time_t creationTime;
|
||
int priority;
|
||
};
|
||
|
||
std::vector<NotificationFile> notificationFiles;
|
||
|
||
// --- Prune missing files from shownFiles ---
|
||
for (auto it = shownFiles.begin(); it != shownFiles.end();) {
|
||
const std::string fullPath = notifPath + *it;
|
||
if (access(fullPath.c_str(), F_OK) != 0) {
|
||
it = shownFiles.erase(it);
|
||
} else {
|
||
++it;
|
||
}
|
||
}
|
||
|
||
static std::string text;
|
||
static int fontSize;
|
||
static int priority;
|
||
|
||
// --- Collect notification files with creation time ---
|
||
while ((entry = readdir(dir)) != nullptr) {
|
||
if (entry->d_type != DT_REG) continue;
|
||
|
||
filename = entry->d_name;
|
||
const size_t filenameLen = filename.size();
|
||
|
||
// Must end with ".notify"
|
||
if (filenameLen <= 7 || filename.compare(filenameLen - 7, 7, ".notify") != 0)
|
||
continue;
|
||
|
||
// Skip if already shown
|
||
if (std::find(shownFiles.begin(), shownFiles.end(), filename) != shownFiles.end())
|
||
continue;
|
||
|
||
// --- Build path ---
|
||
fullPath.clear();
|
||
fullPath = notifPath;
|
||
fullPath += filename;
|
||
|
||
// --- Get file creation/modification time ---
|
||
struct stat fileStat;
|
||
time_t creationTime = 0;
|
||
if (stat(fullPath.c_str(), &fileStat) == 0) {
|
||
// Use modification time as creation time (more universally supported)
|
||
creationTime = fileStat.st_mtime;
|
||
}
|
||
|
||
// --- Read priority from JSON ---
|
||
priority = 20; // default
|
||
std::unique_ptr<ult::json_t, ult::JsonDeleter> root(
|
||
ult::readJsonFromFile(fullPath), ult::JsonDeleter());
|
||
if (root) {
|
||
cJSON* croot = reinterpret_cast<cJSON*>(root.get());
|
||
cJSON* priorityObj = cJSON_GetObjectItemCaseSensitive(croot, "priority");
|
||
if (priorityObj && cJSON_IsNumber(priorityObj)) {
|
||
priority = static_cast<int>(priorityObj->valuedouble);
|
||
}
|
||
}
|
||
|
||
// Add to collection
|
||
notificationFiles.push_back({filename, fullPath, creationTime, priority});
|
||
}
|
||
|
||
closedir(dir);
|
||
|
||
// --- Sort files by priority (higher first), then creation time (older first) ---
|
||
std::sort(notificationFiles.begin(), notificationFiles.end(),
|
||
[](const NotificationFile& a, const NotificationFile& b) {
|
||
// Primary sort: Higher priority first
|
||
if (a.priority != b.priority) {
|
||
return a.priority > b.priority;
|
||
}
|
||
// Secondary sort: Older creation time first (FIFO within same priority)
|
||
return a.creationTime < b.creationTime;
|
||
});
|
||
|
||
// --- Process files in chronological order ---
|
||
for (const auto& notifFile : notificationFiles) {
|
||
// --- Load JSON (safe outside notification lock) ---
|
||
text = ult::getStringFromJsonFile(notifFile.fullPath, "text");
|
||
if (!text.empty()) {
|
||
fontSize = 28; // default
|
||
|
||
std::unique_ptr<ult::json_t, ult::JsonDeleter> root(
|
||
ult::readJsonFromFile(notifFile.fullPath), ult::JsonDeleter());
|
||
if (root) {
|
||
cJSON* croot = reinterpret_cast<cJSON*>(root.get());
|
||
cJSON* fontSizeObj = cJSON_GetObjectItemCaseSensitive(croot, "font_size");
|
||
if (fontSizeObj && cJSON_IsNumber(fontSizeObj)) {
|
||
fontSize = std::clamp(static_cast<int>(fontSizeObj->valuedouble), 1, 34);
|
||
}
|
||
}
|
||
|
||
// --- Show notification safely ---
|
||
if (notification) {
|
||
notification->show(text, fontSize, notifFile.priority, notifFile.filename);
|
||
}
|
||
|
||
// Mark file as shown
|
||
shownFiles.push_back(notifFile.filename);
|
||
}
|
||
}
|
||
|
||
} else {
|
||
// --- Notifications disabled: delete all files ---
|
||
struct dirent* entry;
|
||
std::string fullPath;
|
||
|
||
while ((entry = readdir(dir)) != nullptr) {
|
||
if (entry->d_type != DT_REG) continue;
|
||
|
||
const char* fname = entry->d_name;
|
||
const size_t len = strlen(fname);
|
||
|
||
if (len > 7 && strcmp(fname + len - 7, ".notify") == 0) {
|
||
fullPath.clear();
|
||
fullPath = ult::NOTIFICATIONS_PATH;
|
||
fullPath.append(fname, len);
|
||
|
||
remove(fullPath.c_str());
|
||
}
|
||
}
|
||
closedir(dir);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
|
||
|
||
|
||
// Combine inputs from both controllers
|
||
const u64 kDown_p1 = padGetButtonsDown(&pad_p1);
|
||
const u64 kDown_handheld = padGetButtonsDown(&pad_handheld);
|
||
const u64 kHeld_p1 = padGetButtons(&pad_p1);
|
||
const u64 kHeld_handheld = padGetButtons(&pad_handheld);
|
||
|
||
shData->keysDown = kDown_p1 | kDown_handheld;
|
||
shData->keysHeld = kHeld_p1 | kHeld_handheld;
|
||
|
||
// For joysticks, prioritize handheld if available, otherwise use P1
|
||
const HidAnalogStickState leftStick_handheld = padGetStickPos(&pad_handheld, 0);
|
||
const HidAnalogStickState rightStick_handheld = padGetStickPos(&pad_handheld, 1);
|
||
|
||
// Check if handheld has any stick input (not at center position)
|
||
const bool handheldHasInput = (leftStick_handheld.x != 0 || leftStick_handheld.y != 0 ||
|
||
rightStick_handheld.x != 0 || rightStick_handheld.y != 0);
|
||
|
||
if (handheldHasInput) {
|
||
shData->joyStickPosLeft = leftStick_handheld;
|
||
shData->joyStickPosRight = rightStick_handheld;
|
||
} else {
|
||
shData->joyStickPosLeft = padGetStickPos(&pad_p1, 0);
|
||
shData->joyStickPosRight = padGetStickPos(&pad_p1, 1);
|
||
}
|
||
|
||
|
||
// Read in touch positions
|
||
if (hidGetTouchScreenStates(&shData->touchState, 1) > 0) { // Check if any touch event is present
|
||
if (!shData->overlayOpen) {
|
||
//ult::internalTouchReleased = false;
|
||
ult::internalTouchReleased.store(false, std::memory_order_release);
|
||
}
|
||
|
||
const HidTouchState& currentTouch = shData->touchState.touches[0]; // Correct type is HidTouchPoint
|
||
|
||
|
||
const u64 elapsedTime_ns = armTicksToNs(nowTick - currentTouchTick);
|
||
|
||
// Check if the touch is within bounds for left-to-right swipe within the time window
|
||
if (ult::useSwipeToOpen && elapsedTime_ns <= TOUCH_THRESHOLD_NS) {
|
||
if ((lastTouchX != 0 && lastTouchY != 0) && (currentTouch.x != 0 || currentTouch.y != 0)) {
|
||
if (ult::layerEdge == 0 && currentTouch.x > SWIPE_RIGHT_BOUND + 84 && lastTouchX <= SWIPE_RIGHT_BOUND) {
|
||
eventFire(&shData->comboEvent);
|
||
mainComboHasTriggered.store(true, std::memory_order_release);
|
||
}
|
||
// Check if the touch is within bounds for right-to-left swipe within the time window
|
||
else if (ult::layerEdge > 0 && currentTouch.x < SWIPE_LEFT_BOUND - 84 && lastTouchX >= SWIPE_LEFT_BOUND) {
|
||
eventFire(&shData->comboEvent);
|
||
mainComboHasTriggered.store(true, std::memory_order_release);
|
||
}
|
||
}
|
||
}
|
||
|
||
// Handle touch release state
|
||
if (currentTouch.x == 0 && currentTouch.y == 0) {
|
||
ult::internalTouchReleased.store(true, std::memory_order_release);
|
||
//ult::internalTouchReleased = true; // Indicate that the touch has been released
|
||
//ult::internalTouchReleased.store(true, std::memory_order_release);
|
||
lastTouchX = 0;
|
||
lastTouchY = 0;
|
||
}
|
||
|
||
// If this is the first touch of a gesture, store lastTouchX
|
||
else if ((lastTouchX == 0 && lastTouchY == 0) && (currentTouch.x != 0 || currentTouch.y != 0)) {
|
||
currentTouchTick = nowTick;
|
||
lastTouchX = currentTouch.x;
|
||
lastTouchY = currentTouch.y;
|
||
}
|
||
|
||
} else {
|
||
// Reset touch state if no touch is present
|
||
shData->touchState = { 0 };
|
||
//ult::internalTouchReleased = true;
|
||
ult::internalTouchReleased.store(true, std::memory_order_release);
|
||
//ult::internalTouchReleased.store(true, std::memory_order_release);
|
||
|
||
// Reset touch history to invalid state
|
||
lastTouchX = 0;
|
||
lastTouchY = 0;
|
||
|
||
// Reset time tracking
|
||
//currentTouchTick = nowTick;
|
||
}
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (triggerExitNow) {
|
||
//ult::launchingOverlay.store(true, std::memory_order_release);
|
||
ult::setIniFileValue(
|
||
ult::ULTRAHAND_CONFIG_INI_PATH,
|
||
ult::ULTRAHAND_PROJECT_NAME,
|
||
ult::IN_OVERLAY_STR,
|
||
ult::FALSE_STR
|
||
);
|
||
//tsl::setNextOverlay(
|
||
// ult::OVERLAY_PATH + "ovlmenu.ovl"
|
||
//);
|
||
//tsl::Overlay::get()->close();
|
||
tsl::goBack();
|
||
//triggerExitNow = false;
|
||
break;
|
||
}
|
||
#endif
|
||
//if ((shData->keysDown & KEY_ZL && shData->keysHeld & KEY_L) || (shData->keysDown & KEY_L && shData->keysHeld & KEY_ZL)) {
|
||
// notification->show("Hello world! ¯\\_(ツ)_/¯");
|
||
// eventFire(&shData->notificationEvent); // wake the loop
|
||
//}
|
||
|
||
|
||
// Check main launch combo first (highest priority)
|
||
if ((((shData->keysHeld & tsl::cfg::launchCombo) == tsl::cfg::launchCombo) && shData->keysDown & tsl::cfg::launchCombo)) {
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
if (ult::updateMenuCombos) {
|
||
ult::setIniFileValue(ult::ULTRAHAND_CONFIG_INI_PATH, ult::ULTRAHAND_PROJECT_NAME, ult::KEY_COMBO_STR , ult::ULTRAHAND_COMBO_STR);
|
||
ult::setIniFileValue(ult::TESLA_CONFIG_INI_PATH, ult::TESLA_STR, ult::KEY_COMBO_STR , ult::ULTRAHAND_COMBO_STR);
|
||
ult::updateMenuCombos = false;
|
||
}
|
||
#endif
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
isRendering = false;
|
||
leventSignal(&renderingStopEvent);
|
||
#endif
|
||
|
||
if (shData->overlayOpen) {
|
||
tsl::Overlay::get()->hide();
|
||
shData->overlayOpen = false;
|
||
}
|
||
else {
|
||
eventFire(&shData->comboEvent);
|
||
mainComboHasTriggered.store(true, std::memory_order_release);
|
||
}
|
||
}
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
else if (ult::updateMenuCombos && (((shData->keysHeld & tsl::cfg::launchCombo2) == tsl::cfg::launchCombo2) && shData->keysDown & tsl::cfg::launchCombo2)) {
|
||
std::swap(tsl::cfg::launchCombo, tsl::cfg::launchCombo2); // Swap the two launch combos
|
||
ult::setIniFileValue(ult::ULTRAHAND_CONFIG_INI_PATH, ult::ULTRAHAND_PROJECT_NAME, ult::KEY_COMBO_STR , ult::TESLA_COMBO_STR);
|
||
ult::setIniFileValue(ult::TESLA_CONFIG_INI_PATH, ult::TESLA_STR, ult::KEY_COMBO_STR , ult::TESLA_COMBO_STR);
|
||
eventFire(&shData->comboEvent);
|
||
mainComboHasTriggered.store(true, std::memory_order_release);
|
||
ult::updateMenuCombos = false;
|
||
}
|
||
else if (ult::overlayLaunchRequested.load(std::memory_order_acquire) && !ult::runningInterpreter.load(std::memory_order_acquire) && ult::settingsInitialized.load(std::memory_order_acquire) && (nowNs - startNs) >= FAST_SWAP_THRESHOLD_NS) {
|
||
std::string requestedPath, requestedArgs;
|
||
|
||
// Get the request data safely
|
||
{
|
||
std::lock_guard<std::mutex> lock(ult::overlayLaunchMutex);
|
||
requestedPath = ult::requestedOverlayPath;
|
||
requestedArgs = ult::requestedOverlayArgs;
|
||
ult::overlayLaunchRequested.store(false, std::memory_order_release);
|
||
}
|
||
|
||
if (!requestedPath.empty()) {
|
||
|
||
const std::string overlayFileName = ult::getNameFromPath(requestedPath);
|
||
|
||
// Set overlay state for ovlmenu.ovl
|
||
|
||
// OPTIMIZED: Batch INI file writes
|
||
{
|
||
auto iniData = ult::getParsedDataFromIniFile(ult::ULTRAHAND_CONFIG_INI_PATH);
|
||
auto& section = iniData[ult::ULTRAHAND_PROJECT_NAME];
|
||
section[ult::IN_OVERLAY_STR] = ult::TRUE_STR;
|
||
section["to_packages"] = ult::TRUE_STR;
|
||
ult::saveIniFileData(ult::ULTRAHAND_CONFIG_INI_PATH, iniData);
|
||
}
|
||
|
||
// Reset navigation state variables (these control slide navigation)
|
||
ult::allowSlide.store(false, std::memory_order_release);
|
||
ult::unlockedSlide.store(false, std::memory_order_release);
|
||
|
||
eventClose(&homeButtonPressEvent);
|
||
eventClose(&powerButtonPressEvent);
|
||
eventClose(&captureButtonPressEvent);
|
||
|
||
//hidExit();
|
||
// Launch the overlay using the same mechanism as key combos
|
||
//shData->overlayOpen = false;
|
||
ult::launchingOverlay.store(true, std::memory_order_release);
|
||
tsl::setNextOverlay(requestedPath, requestedArgs+" --direct");
|
||
tsl::Overlay::get()->close();
|
||
eventFire(&shData->comboEvent);
|
||
|
||
launchComboHasTriggered.store(true, std::memory_order_release);
|
||
return;
|
||
}
|
||
}
|
||
#endif
|
||
// Check overlay key combos (only when overlay is not open, keys are pressed, and not conflicting with main combos)
|
||
//else if (!shData->overlayOpen && shData->keysDown != 0) {
|
||
else if (shData->keysDown != 0 && ult::useLaunchCombos) {
|
||
if (shData->keysHeld != tsl::cfg::launchCombo) {
|
||
// Lookup both path and optional mode launch args
|
||
const auto comboInfo = tsl::hlp::getEntryForKeyCombo(shData->keysHeld);
|
||
const std::string& overlayPath = comboInfo.path;
|
||
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
if (!overlayPath.empty() && (shData->keysHeld) && !ult::runningInterpreter.load(std::memory_order_acquire) && ult::settingsInitialized.load(std::memory_order_acquire) && (armTicksToNs(nowTick) - startNs) >= FAST_SWAP_THRESHOLD_NS) {
|
||
#else
|
||
if (!overlayPath.empty() && (shData->keysHeld) && (nowNs - startNs) >= FAST_SWAP_THRESHOLD_NS) {
|
||
#endif
|
||
const std::string& modeArg = comboInfo.launchArg;
|
||
const std::string overlayFileName = ult::getNameFromPath(overlayPath);
|
||
|
||
// hideHidden check
|
||
if (hideHidden) {
|
||
const auto hideStatus = ult::parseValueFromIniSection(
|
||
ult::OVERLAYS_INI_FILEPATH, overlayFileName, ult::HIDE_STR);
|
||
if (hideStatus == ult::TRUE_STR) {
|
||
shData->keysDownPending |= shData->keysDown;
|
||
continue;
|
||
}
|
||
}
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
isRendering = false;
|
||
leventSignal(&renderingStopEvent);
|
||
#endif
|
||
|
||
#if !IS_LAUNCHER_DIRECTIVE
|
||
if (lastOverlayFilename == overlayFileName && lastOverlayMode == modeArg) {
|
||
#else
|
||
if (lastOverlayFilename == overlayFileName && lastOverlayMode == modeArg && lastOverlayMode.find("--package") != std::string::npos) {
|
||
#endif
|
||
ult::setIniFileValue(
|
||
ult::ULTRAHAND_CONFIG_INI_PATH,
|
||
ult::ULTRAHAND_PROJECT_NAME,
|
||
ult::IN_OVERLAY_STR,
|
||
ult::TRUE_STR
|
||
);
|
||
|
||
//shData->overlayOpen = false;
|
||
//hidExit();
|
||
|
||
eventClose(&homeButtonPressEvent);
|
||
eventClose(&powerButtonPressEvent);
|
||
eventClose(&captureButtonPressEvent);
|
||
|
||
ult::launchingOverlay.store(true, std::memory_order_release);
|
||
tsl::setNextOverlay(ult::OVERLAY_PATH + "ovlmenu.ovl", "--direct");
|
||
tsl::Overlay::get()->close();
|
||
eventFire(&shData->comboEvent);
|
||
launchComboHasTriggered.store(true, std::memory_order_release);
|
||
return;
|
||
}
|
||
|
||
// Compose launch args
|
||
std::string finalArgs;
|
||
if (!modeArg.empty()) {
|
||
finalArgs = modeArg;
|
||
} else {
|
||
// Only check overlay-specific launch args for non-ovlmenu entries
|
||
if (overlayFileName != "ovlmenu.ovl") {
|
||
// OPTIMIZED: Single INI read for both values
|
||
auto overlaysIniData = ult::getParsedDataFromIniFile(ult::OVERLAYS_INI_FILEPATH);
|
||
std::string useArgs = "";
|
||
std::string launchArgs = "";
|
||
|
||
auto sectionIt = overlaysIniData.find(overlayFileName);
|
||
if (sectionIt != overlaysIniData.end()) {
|
||
auto useArgsIt = sectionIt->second.find(ult::USE_LAUNCH_ARGS_STR);
|
||
if (useArgsIt != sectionIt->second.end()) {
|
||
useArgs = useArgsIt->second;
|
||
}
|
||
|
||
auto argsIt = sectionIt->second.find(ult::LAUNCH_ARGS_STR);
|
||
if (argsIt != sectionIt->second.end()) {
|
||
launchArgs = argsIt->second;
|
||
}
|
||
}
|
||
|
||
if (useArgs == ult::TRUE_STR) {
|
||
finalArgs = launchArgs;
|
||
ult::removeQuotes(finalArgs);
|
||
}
|
||
}
|
||
}
|
||
if (finalArgs.empty()) {
|
||
finalArgs = "--direct";
|
||
} else {
|
||
finalArgs += " --direct";
|
||
}
|
||
|
||
if (overlayFileName == "ovlmenu.ovl") {
|
||
ult::setIniFileValue(
|
||
ult::ULTRAHAND_CONFIG_INI_PATH,
|
||
ult::ULTRAHAND_PROJECT_NAME,
|
||
ult::IN_OVERLAY_STR,
|
||
ult::TRUE_STR
|
||
);
|
||
}
|
||
|
||
//shData->overlayOpen = false;
|
||
//hidExit();
|
||
eventClose(&homeButtonPressEvent);
|
||
eventClose(&powerButtonPressEvent);
|
||
eventClose(&captureButtonPressEvent);
|
||
|
||
ult::launchingOverlay.store(true, std::memory_order_release);
|
||
tsl::setNextOverlay(overlayPath, finalArgs);
|
||
tsl::Overlay::get()->close();
|
||
eventFire(&shData->comboEvent);
|
||
launchComboHasTriggered.store(true, std::memory_order_release);
|
||
return;
|
||
}
|
||
}
|
||
}
|
||
//#endif
|
||
|
||
shData->keysDownPending |= shData->keysDown;
|
||
}
|
||
|
||
//20 ms
|
||
//s32 idx = 0;
|
||
rc = waitObjects(&idx, objects, WaiterObject_Count, 20'000'000ul);
|
||
if (R_SUCCEEDED(rc)) {
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (idx == WaiterObject_HomeButton || idx == WaiterObject_PowerButton) { // Changed condition to exclude capture button
|
||
if (shData->overlayOpen && !isMiniOrMicroMode) {
|
||
tsl::Overlay::get()->hide();
|
||
shData->overlayOpen = false;
|
||
}
|
||
}
|
||
#else
|
||
if (idx == WaiterObject_HomeButton || idx == WaiterObject_PowerButton) { // Changed condition to exclude capture button
|
||
if (shData->overlayOpen) {
|
||
tsl::Overlay::get()->hide();
|
||
shData->overlayOpen = false;
|
||
}
|
||
}
|
||
#endif
|
||
|
||
switch (idx) {
|
||
case WaiterObject_HomeButton:
|
||
eventClear(&homeButtonPressEvent);
|
||
break;
|
||
case WaiterObject_PowerButton:
|
||
eventClear(&powerButtonPressEvent);
|
||
|
||
// Perform any necessary cleanup
|
||
hidExit();
|
||
|
||
// Reinitialize resources
|
||
ASSERT_FATAL(hidInitialize()); // Reinitialize HID to reset states
|
||
|
||
// Reinitialize both controllers
|
||
padInitialize(&pad_p1, HidNpadIdType_No1);
|
||
padInitialize(&pad_handheld, HidNpadIdType_Handheld);
|
||
hidInitializeTouchScreen();
|
||
|
||
// Update both controllers
|
||
padUpdate(&pad_p1);
|
||
padUpdate(&pad_handheld);
|
||
break;
|
||
|
||
|
||
case WaiterObject_CaptureButton:
|
||
if (screenshotsAreDisabled) {
|
||
eventClear(&captureButtonPressEvent);
|
||
break;
|
||
}
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (isMiniOrMicroMode) {
|
||
delayUpdate = true;
|
||
isRendering = false;
|
||
leventSignal(&renderingStopEvent);
|
||
}
|
||
#endif
|
||
|
||
ult::disableTransparency = true;
|
||
eventClear(&captureButtonPressEvent);
|
||
svcSleepThread(1'500'000'000);
|
||
ult::disableTransparency = false;
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (isMiniOrMicroMode) {
|
||
isRendering = true;
|
||
leventClear(&renderingStopEvent);
|
||
delayUpdate = false;
|
||
}
|
||
#endif
|
||
|
||
break;
|
||
}
|
||
} else if (rc != KERNELRESULT(TimedOut)) {
|
||
ASSERT_FATAL(rc);
|
||
}
|
||
}
|
||
//hidExit();
|
||
|
||
eventClose(&homeButtonPressEvent);
|
||
eventClose(&powerButtonPressEvent);
|
||
eventClose(&captureButtonPressEvent);
|
||
|
||
}
|
||
}
|
||
|
||
/**
|
||
* @brief Creates a new Gui and changes to it
|
||
*
|
||
* @tparam G Gui to create
|
||
* @tparam Args Arguments to pass to the Gui
|
||
* @param args Arguments to pass to the Gui
|
||
* @return Reference to the newly created Gui
|
||
*/
|
||
template<typename G, typename ...Args>
|
||
std::unique_ptr<tsl::Gui>& changeTo(Args&&... args) {
|
||
return Overlay::get()->changeTo<G, Args...>(std::forward<Args>(args)...);
|
||
}
|
||
|
||
template<typename G, typename ...Args>
|
||
std::unique_ptr<tsl::Gui>& swapTo(Args&&... args) {
|
||
return Overlay::get()->swapTo<G, Args...>(std::forward<Args>(args)...);
|
||
}
|
||
|
||
template<typename G, typename ...Args>
|
||
std::unique_ptr<tsl::Gui>& swapTo(SwapDepth depth, Args&&... args) {
|
||
return Overlay::get()->swapTo<G, Args...>(depth, std::forward<Args>(args)...);
|
||
}
|
||
|
||
|
||
/**
|
||
* @brief Pops the top Gui from the stack and goes back to the last one
|
||
* @note The Overlay gets closed once there are no more Guis on the stack
|
||
*/
|
||
void goBack(u32 count) {
|
||
Overlay::get()->goBack(count);
|
||
}
|
||
|
||
void pop(u32 count) {
|
||
Overlay::get()->pop(count);
|
||
}
|
||
|
||
|
||
static inline std::mutex setNextOverlayMutex;
|
||
static void setNextOverlay(const std::string& ovlPath, std::string origArgs) {
|
||
std::lock_guard lk(setNextOverlayMutex);
|
||
char buffer[512];
|
||
char* p = buffer;
|
||
char* bufferEnd = buffer + sizeof(buffer) - 1; // Leave room for null terminator
|
||
|
||
// Store filename and copy it
|
||
const std::string filenameStr = ult::getNameFromPath(ovlPath);
|
||
const char* filename = filenameStr.c_str();
|
||
while (*filename && p < bufferEnd) *p++ = *filename++;
|
||
if (p < bufferEnd) *p++ = ' ';
|
||
|
||
// Single-pass argument filtering
|
||
const char* src = origArgs.c_str();
|
||
const char* end = src + origArgs.length();
|
||
bool hasSkipCombo = false;
|
||
|
||
while (src < end && p < bufferEnd) {
|
||
// Skip whitespace
|
||
while (src < end && *src == ' ' && p < bufferEnd) {
|
||
*p++ = *src++;
|
||
}
|
||
|
||
if (src >= end || p >= bufferEnd) break;
|
||
|
||
// Check for flags to filter/detect
|
||
if (src[0] == '-' && src[1] == '-') {
|
||
|
||
// Check what flag this is
|
||
if (strncmp(src, "--skipCombo", 11) == 0 && (src[11] == ' ' || src[11] == '\0')) {
|
||
hasSkipCombo = true;
|
||
// Copy this flag
|
||
while (src < end && *src != ' ' && p < bufferEnd) *p++ = *src++;
|
||
}
|
||
else if (strncmp(src, "--foregroundFix", 15) == 0) {
|
||
// Skip this flag and its value
|
||
src += 15;
|
||
while (src < end && *src == ' ') src++; // Skip spaces
|
||
if (src < end && (*src == '0' || *src == '1')) src++; // Skip value
|
||
}
|
||
else if (strncmp(src, "--lastTitleID", 13) == 0) {
|
||
// Skip this flag and its value
|
||
src += 13;
|
||
while (src < end && *src == ' ') src++; // Skip spaces
|
||
while (src < end && *src != ' ' && *src != '\0') src++; // Skip title ID
|
||
}
|
||
//else if (strncmp(src, "--promptDuration", 16) == 0) {
|
||
// // skip any existing --promptDuration (we’ll re-add it)
|
||
// src += 16;
|
||
// while (src < end && *src == ' ') src++;
|
||
// while (src < end && *src != ' ' && *src != '\0') src++;
|
||
//}
|
||
else {
|
||
// Copy unknown flag
|
||
while (src < end && *src != ' ' && p < bufferEnd) *p++ = *src++;
|
||
}
|
||
}
|
||
else {
|
||
// Copy regular argument
|
||
while (src < end && *src != ' ' && p < bufferEnd) *p++ = *src++;
|
||
}
|
||
}
|
||
|
||
// Add required flags with bounds checking
|
||
if (!hasSkipCombo && (p + 12) < bufferEnd) {
|
||
memcpy(p, " --skipCombo", 12);
|
||
p += 12;
|
||
}
|
||
|
||
// Add foreground flag with bounds checking
|
||
if ((p + 17) < bufferEnd) {
|
||
memcpy(p, " --foregroundFix ", 17);
|
||
p += 17;
|
||
if (p < bufferEnd) {
|
||
*p++ = (ult::resetForegroundCheck.load(std::memory_order_acquire) || ult::lastTitleID != ult::getTitleIdAsString()) ? '1' : '0';
|
||
}
|
||
}
|
||
|
||
// Add last title ID with bounds checking
|
||
if ((p + 15 + ult::lastTitleID.length()) < bufferEnd) {
|
||
memcpy(p, " --lastTitleID ", 15);
|
||
p += 15;
|
||
const char* titleId = ult::lastTitleID.c_str();
|
||
while (*titleId && p < bufferEnd) *p++ = *titleId++;
|
||
}
|
||
|
||
// Add prompt duration
|
||
//{
|
||
// const s64 duration = notification->remainingTime(); // nanoseconds left
|
||
// if (duration > 0) {
|
||
// const std::string promptArg = " --promptDuration " + std::to_string(duration);
|
||
// if ((p + promptArg.size()) < bufferEnd) {
|
||
// memcpy(p, promptArg.c_str(), promptArg.size());
|
||
// p += promptArg.size();
|
||
// }
|
||
// }
|
||
//}
|
||
|
||
// Safety check - if we're at the end, we might have truncated
|
||
if (p >= bufferEnd) {
|
||
p = bufferEnd;
|
||
}
|
||
|
||
*p = '\0';
|
||
|
||
//isLaunchingNextOverlay.store(true, std::memory_order_release);
|
||
envSetNextLoad(ovlPath.c_str(), buffer);
|
||
}
|
||
|
||
|
||
|
||
struct option_entry {
|
||
const char* name;
|
||
u8 len;
|
||
u8 action;
|
||
};
|
||
|
||
static const struct option_entry options[] = {
|
||
{"direct", 6, 1},
|
||
{"skipCombo", 9, 2},
|
||
{"lastTitleID", 11, 3},
|
||
{"foregroundFix", 13, 4},
|
||
{"package", 7, 5}
|
||
};
|
||
|
||
|
||
/**
|
||
* @brief libtesla's main function
|
||
* @note Call it directly from main passing in argc and argv and returning it e.g `return tsl::loop<OverlayTest>(argc, argv);`
|
||
*
|
||
* @tparam TOverlay Your overlay class
|
||
* @tparam launchFlags \ref LaunchFlags
|
||
* @param argc argc
|
||
* @param argv argv
|
||
* @return int result
|
||
*/
|
||
template<typename TOverlay, impl::LaunchFlags launchFlags>
|
||
static inline int loop(int argc, char** argv) {
|
||
static_assert(std::is_base_of_v<tsl::Overlay, TOverlay>, "tsl::loop expects a type derived from tsl::Overlay");
|
||
|
||
|
||
|
||
// Initialize buffer sizes based on expanded memory setting
|
||
if (ult::expandedMemory) {
|
||
ult::COPY_BUFFER_SIZE = 262144;
|
||
ult::HEX_BUFFER_SIZE = 8192;
|
||
ult::UNZIP_READ_BUFFER = 262144;
|
||
ult::UNZIP_WRITE_BUFFER = 131072;
|
||
ult::DOWNLOAD_READ_BUFFER = 262144;
|
||
ult::DOWNLOAD_WRITE_BUFFER = 131072;
|
||
}
|
||
|
||
if (argc > 0) {
|
||
g_overlayFilename = ult::getNameFromPath(argv[0]);
|
||
lastOverlayFilename = g_overlayFilename;
|
||
|
||
lastOverlayMode.clear();
|
||
bool skip;
|
||
for (u8 arg = 1; arg < argc; arg++) {
|
||
const char* s = argv[arg];
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (s[0] == '-') {
|
||
if (s[1] == 'm') {
|
||
if (strcasecmp(s, "-mini") == 0 || strcasecmp(s, "-micro") == 0) {
|
||
isMiniOrMicroMode = true;
|
||
}
|
||
} else if (s[1] == '-' && s[2] == 'm') {
|
||
if (strcasecmp(s, "--miniOverlay") == 0 || strcasecmp(s, "--microOverlay") == 0) {
|
||
isMiniOrMicroMode = true;
|
||
}
|
||
}
|
||
}
|
||
#endif
|
||
skip = false;
|
||
|
||
if (arg > 1) {
|
||
const char* prev = argv[arg - 1];
|
||
if (prev[0] == '-' && prev[1] == '-') {
|
||
if (strcmp(prev, "--lastTitleID") == 0 || strcmp(prev, "--foregroundFix") == 0) {
|
||
skip = true;
|
||
}
|
||
}
|
||
}
|
||
|
||
if (!skip && s[0] == '-' && s[1] == '-') {
|
||
if (strcmp(s, "--direct") == 0 ||
|
||
strcmp(s, "--skipCombo") == 0 ||
|
||
strcmp(s, "--lastTitleID") == 0 ||
|
||
strcmp(s, "--foregroundFix") == 0) {
|
||
skip = true;
|
||
}
|
||
}
|
||
|
||
if (!skip) {
|
||
if (strcmp(s, "--package") == 0) {
|
||
lastOverlayMode = "--package";
|
||
arg++;
|
||
if (arg < argc) {
|
||
lastOverlayMode += " ";
|
||
lastOverlayMode += argv[arg];
|
||
arg++;
|
||
while (arg < argc && argv[arg][0] != '-') {
|
||
lastOverlayMode += " ";
|
||
lastOverlayMode += argv[arg];
|
||
arg++;
|
||
}
|
||
}
|
||
} else {
|
||
lastOverlayMode = s;
|
||
}
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
|
||
bool skipCombo = false;
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
bool directMode = true;
|
||
#else
|
||
bool directMode = false;
|
||
#endif
|
||
bool usingPackageLauncher = false;
|
||
|
||
for (u8 arg = 0; arg < argc; arg++) {
|
||
const char* s = argv[arg];
|
||
if (s[0] != '-' || s[1] != '-') continue;
|
||
const char* opt = s + 2;
|
||
|
||
for (u8 i = 0; i < 5; i++) {
|
||
if (memcmp(opt, options[i].name, options[i].len) == 0 && opt[options[i].len] == '\0') {
|
||
switch (options[i].action) {
|
||
case 1: // direct
|
||
directMode = true;
|
||
g_overlayFilename = "";
|
||
jumpItemName = "";
|
||
jumpItemValue = "";
|
||
jumpItemExactMatch.store(true, std::memory_order_release);
|
||
break;
|
||
case 2: // skipCombo
|
||
skipCombo = true;
|
||
ult::firstBoot = false;
|
||
break;
|
||
case 3: // lastTitleID
|
||
if (++arg < argc) {
|
||
const char* providedID = argv[arg];
|
||
if (ult::getTitleIdAsString() != providedID) {
|
||
ult::resetForegroundCheck.store(true, std::memory_order_release);
|
||
}
|
||
}
|
||
break;
|
||
case 4: // foregroundFix
|
||
if (++arg < argc) {
|
||
ult::resetForegroundCheck.store(ult::resetForegroundCheck.load(std::memory_order_acquire) ||
|
||
(argv[arg][0] == '1'), std::memory_order_release);
|
||
}
|
||
break;
|
||
case 5: // package
|
||
usingPackageLauncher = true;
|
||
break;
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
impl::SharedThreadData shData;
|
||
shData.running.store(true, std::memory_order_release);
|
||
|
||
Thread backgroundThread;
|
||
threadCreate(&backgroundThread, impl::backgroundEventPoller, &shData, nullptr, 0x2000, 0x2c, -2);
|
||
threadStart(&backgroundThread);
|
||
|
||
eventCreate(&shData.comboEvent, false);
|
||
|
||
auto& overlay = tsl::Overlay::s_overlayInstance;
|
||
overlay = new TOverlay();
|
||
overlay->m_closeOnExit = (u8(launchFlags) & u8(impl::LaunchFlags::CloseOnExit)) == u8(impl::LaunchFlags::CloseOnExit);
|
||
|
||
tsl::hlp::doWithSmSession([&overlay]{
|
||
overlay->initServices();
|
||
});
|
||
#if !IS_LAUNCHER_DIRECTIVE
|
||
tsl::initializeUltrahandSettings();
|
||
#endif
|
||
overlay->initScreen();
|
||
overlay->changeTo(overlay->loadInitialGui());
|
||
|
||
#if IS_LAUNCHER_DIRECTIVE
|
||
{
|
||
bool inOverlay;
|
||
bool shouldFireEvent = false;
|
||
auto configData = ult::getParsedDataFromIniFile(ult::ULTRAHAND_CONFIG_INI_PATH);
|
||
bool needsUpdate = false;
|
||
|
||
if (ult::firstBoot) {
|
||
configData[ult::ULTRAHAND_PROJECT_NAME][ult::IN_OVERLAY_STR] = ult::FALSE_STR;
|
||
needsUpdate = true;
|
||
}
|
||
|
||
auto projectIt = configData.find(ult::ULTRAHAND_PROJECT_NAME);
|
||
if (projectIt != configData.end()) {
|
||
auto overlayIt = projectIt->second.find(ult::IN_OVERLAY_STR);
|
||
inOverlay = (overlayIt == projectIt->second.end() || overlayIt->second != ult::FALSE_STR);
|
||
} else {
|
||
inOverlay = true;
|
||
}
|
||
|
||
if (inOverlay && skipCombo) {
|
||
configData[ult::ULTRAHAND_PROJECT_NAME][ult::IN_OVERLAY_STR] = ult::FALSE_STR;
|
||
needsUpdate = true;
|
||
shouldFireEvent = true;
|
||
}
|
||
|
||
if (needsUpdate) {
|
||
ult::saveIniFileData(ult::ULTRAHAND_CONFIG_INI_PATH, configData);
|
||
}
|
||
|
||
if (shouldFireEvent) {
|
||
eventFire(&shData.comboEvent);
|
||
}
|
||
}
|
||
#else
|
||
if (skipCombo) {
|
||
eventFire(&shData.comboEvent);
|
||
}
|
||
#endif
|
||
|
||
overlay->disableNextAnimation();
|
||
|
||
{
|
||
Handle handles[2] = { shData.comboEvent.revent, notificationEvent.revent };
|
||
s32 index = -1;
|
||
|
||
bool exitAfterPrompt = false;
|
||
bool comboBreakout = false;
|
||
bool firstLoop = !ult::firstBoot;
|
||
|
||
while (shData.running.load(std::memory_order_acquire)) {
|
||
// Early exit if launching new overlay
|
||
if (ult::launchingOverlay.load(std::memory_order_acquire)) {
|
||
//std::scoped_lock lock(shData.dataMutex);
|
||
shData.running.store(false, std::memory_order_release);
|
||
shData.overlayOpen.store(false, std::memory_order_release);
|
||
break;
|
||
}
|
||
|
||
// Wait for events only if no active notification
|
||
if (!(notification && notification->isActive())) {
|
||
svcWaitSynchronization(&index, handles, 2, UINT64_MAX);
|
||
//eventClear(¬ificationEvent);
|
||
//eventClear(&shData.comboEvent);
|
||
}
|
||
eventClear(¬ificationEvent);
|
||
eventClear(&shData.comboEvent);
|
||
|
||
if ((notification && notification->isActive() && !firstLoop) || index == 1) {
|
||
comboBreakout = false;
|
||
|
||
while (shData.running.load(std::memory_order_acquire)) {
|
||
{
|
||
//std::scoped_lock lock(shData.dataMutex);
|
||
if (ult::launchingOverlay.load(std::memory_order_acquire)) {
|
||
shData.running.store(false, std::memory_order_release);
|
||
shData.overlayOpen.store(false, std::memory_order_release);
|
||
break;
|
||
}
|
||
overlay->loop(true); // Draw prompts while hidden
|
||
}
|
||
|
||
if (mainComboHasTriggered.load(std::memory_order_acquire)) {
|
||
mainComboHasTriggered.store(false, std::memory_order_acquire);
|
||
comboBreakout = true;
|
||
exitAfterPrompt = false;
|
||
break;
|
||
}
|
||
|
||
if (launchComboHasTriggered.load(std::memory_order_acquire)) {
|
||
exitAfterPrompt = true;
|
||
usingPackageLauncher = false;
|
||
directMode = false;
|
||
break;
|
||
}
|
||
|
||
if (!(notification && notification->isActive())) {
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (!comboBreakout || !shData.running.load(std::memory_order_acquire)) {
|
||
{
|
||
//std::scoped_lock lock(shData.dataMutex);
|
||
if (!ult::launchingOverlay.load(std::memory_order_acquire)) {
|
||
overlay->clearScreen();
|
||
}
|
||
}
|
||
if (exitAfterPrompt) {
|
||
std::scoped_lock lock(shData.dataMutex);
|
||
exitAfterPrompt = false;
|
||
shData.running.store(false, std::memory_order_release);
|
||
shData.overlayOpen.store(false, std::memory_order_release);
|
||
ult::launchingOverlay.store(true, std::memory_order_release);
|
||
|
||
if (usingPackageLauncher || directMode) {
|
||
tsl::setNextOverlay(ult::OVERLAY_PATH + "ovlmenu.ovl");
|
||
}
|
||
|
||
hlp::requestForeground(false);
|
||
break;
|
||
}
|
||
continue;
|
||
}
|
||
}
|
||
|
||
{
|
||
//std::scoped_lock lock(shData.dataMutex);
|
||
if (ult::launchingOverlay.load(std::memory_order_acquire)) {
|
||
shData.running.store(false, std::memory_order_release);
|
||
shData.overlayOpen.store(false, std::memory_order_release);
|
||
break;
|
||
}
|
||
firstLoop = false;
|
||
shData.overlayOpen.store(true, std::memory_order_release);
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (!isMiniOrMicroMode)
|
||
hlp::requestForeground(true);
|
||
#else
|
||
hlp::requestForeground(true);
|
||
#endif
|
||
|
||
overlay->show();
|
||
if (!comboBreakout && !(notification && notification->isActive()))
|
||
overlay->clearScreen();
|
||
}
|
||
|
||
while (shData.running.load(std::memory_order_acquire)) {
|
||
{
|
||
|
||
if (ult::launchingOverlay.load(std::memory_order_acquire)) {
|
||
shData.running.store(false, std::memory_order_release);
|
||
shData.overlayOpen.store(false, std::memory_order_release);
|
||
break;
|
||
}
|
||
overlay->loop();
|
||
{
|
||
std::scoped_lock lock(shData.dataMutex);
|
||
if (!overlay->fadeAnimationPlaying()) {
|
||
overlay->handleInput(shData.keysDownPending, shData.keysHeld, shData.touchState.count, shData.touchState.touches[0], shData.joyStickPosLeft, shData.joyStickPosRight);
|
||
}
|
||
shData.keysDownPending = 0;
|
||
}
|
||
}
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
if (pendingExit && wasRendering) {
|
||
pendingExit = false;
|
||
wasRendering = false;
|
||
isRendering = true;
|
||
leventClear(&renderingStopEvent);
|
||
}
|
||
#endif
|
||
|
||
if (overlay->shouldHide()) {
|
||
if (overlay->shouldCloseAfter()) {
|
||
if (!directMode) {
|
||
//std::scoped_lock lock(shData.dataMutex);
|
||
shData.running.store(false, std::memory_order_release);
|
||
shData.overlayOpen.store(false, std::memory_order_release);
|
||
break;
|
||
} else {
|
||
exitAfterPrompt = true;
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
pendingExit = true;
|
||
#endif
|
||
}
|
||
}
|
||
break;
|
||
}
|
||
|
||
if (overlay->shouldClose()) {
|
||
//std::scoped_lock lock(shData.dataMutex);
|
||
shData.running.store(false, std::memory_order_release);
|
||
shData.overlayOpen.store(false, std::memory_order_release);
|
||
break;
|
||
}
|
||
}
|
||
|
||
if (shData.running.load(std::memory_order_acquire)) {
|
||
//std::scoped_lock lock(shData.dataMutex);
|
||
if (!(notification && notification->isActive()))
|
||
overlay->clearScreen();
|
||
overlay->resetFlags();
|
||
hlp::requestForeground(false);
|
||
shData.overlayOpen.store(false, std::memory_order_release);
|
||
mainComboHasTriggered.store(false, std::memory_order_acquire);
|
||
launchComboHasTriggered.store(false, std::memory_order_acquire);
|
||
eventClear(&shData.comboEvent);
|
||
}
|
||
}
|
||
//if (notification)
|
||
// notification->forceShutdown();
|
||
|
||
//while (notification && notification->isActive()) {
|
||
// notification->update(true, true); // No file ops, allow state transitions
|
||
// svcSleepThread(10'000'000); // 1ms sleep
|
||
//}
|
||
//tsl::notificationGeneration.fetch_add(1, std::memory_order_release);
|
||
|
||
// Ensure background thread is fully stopped before overlay cleanup
|
||
shData.running.store(false, std::memory_order_release);
|
||
threadWaitForExit(&backgroundThread);
|
||
threadClose(&backgroundThread);
|
||
|
||
// Cleanup overlay resources
|
||
tsl::elm::fullDeconstruction.store(true, std::memory_order_release);
|
||
hlp::requestForeground(false);
|
||
overlay->exitScreen();
|
||
overlay->exitServices();
|
||
delete overlay;
|
||
|
||
// Synchronize notification shutdown
|
||
//if (notification) {
|
||
// notification->forceShutdown(); // Immediate disable of new notifications
|
||
// while (notification->isActive()) {
|
||
// notification->update(true, true); // No file ops, allow state transitions
|
||
// svcSleepThread(1'000'000); // 1ms sleep
|
||
// }
|
||
//
|
||
// notificationGeneration.fetch_add(1, std::memory_order_release);
|
||
// delete notification;
|
||
//
|
||
// //lock.unlock();
|
||
//}
|
||
|
||
|
||
eventClose(&shData.comboEvent);
|
||
|
||
// Brief delay to ensure thread quiescence before nx-ovlloader transition
|
||
//svcSleepThread(100'000'000); // 100ms
|
||
|
||
return 0;
|
||
}
|
||
}
|
||
|
||
}
|
||
|
||
|
||
|
||
|
||
#ifdef TESLA_INIT_IMPL
|
||
|
||
namespace tsl::cfg {
|
||
|
||
u16 LayerWidth = 0;
|
||
u16 LayerHeight = 0;
|
||
u16 LayerPosX = 0;
|
||
u16 LayerPosY = 0;
|
||
u16 FramebufferWidth = 0;
|
||
u16 FramebufferHeight = 0;
|
||
u64 launchCombo = KEY_ZL | KEY_ZR | KEY_DDOWN;
|
||
u64 launchCombo2 = KEY_L | KEY_DDOWN | KEY_RSTICK;
|
||
}
|
||
extern "C" void __libnx_init_time(void);
|
||
|
||
extern "C" {
|
||
|
||
u32 __nx_applet_type = AppletType_None;
|
||
u32 __nx_fs_num_sessions = 1;
|
||
u32 __nx_nv_transfermem_size = 0x15000;
|
||
ViLayerFlags __nx_vi_stray_layer_flags = (ViLayerFlags)0;
|
||
|
||
/**
|
||
* @brief libtesla service initializing function to override libnx's
|
||
*
|
||
*/
|
||
void __appInit(void) {
|
||
tsl::hlp::doWithSmSession([]{
|
||
|
||
ASSERT_FATAL(fsInitialize());
|
||
ASSERT_FATAL(hidInitialize()); // Controller inputs and Touch
|
||
if (hosversionAtLeast(16,0,0)) {
|
||
ASSERT_FATAL(plInitialize(PlServiceType_User)); // Font data. Use pl:u for 16.0.0+
|
||
} else {
|
||
ASSERT_FATAL(plInitialize(PlServiceType_System)); // Use pl:s for 15.0.1 and below to prevent qlaunch/overlaydisp session exhaustion
|
||
}
|
||
ASSERT_FATAL(pmdmntInitialize()); // PID querying
|
||
ASSERT_FATAL(hidsysInitialize()); // Focus control
|
||
ASSERT_FATAL(setsysInitialize()); // Settings querying
|
||
|
||
// Time initializations
|
||
ASSERT_FATAL(timeInitialize()); // CUSTOM MODIFICATION
|
||
__libnx_init_time(); // CUSTOM MODIFICATION
|
||
timeExit(); // CUSTOM MODIFICATION
|
||
|
||
#if USING_WIDGET_DIRECTIVE
|
||
ult::powerInit();
|
||
i2cInitialize();
|
||
#endif
|
||
|
||
fsdevMountSdmc();
|
||
splInitialize();
|
||
spsmInitialize();
|
||
//i2cInitialize();
|
||
//ASSERT_FATAL(socketInitializeDefault());
|
||
//ASSERT_FATAL(nifmInitialize(NifmServiceType_User));
|
||
});
|
||
ASSERT_FATAL(smInitialize()); // needed to prevent issues with powering device into sleep
|
||
|
||
#if IS_STATUS_MONITOR_DIRECTIVE
|
||
Service *plSrv = plGetServiceSession();
|
||
Service plClone;
|
||
ASSERT_FATAL(serviceClone(plSrv, &plClone));
|
||
serviceClose(plSrv);
|
||
*plSrv = plClone;
|
||
#endif
|
||
|
||
eventCreate(&tsl::notificationEvent, false);
|
||
tsl::notification = new tsl::NotificationPrompt();
|
||
//tsl::notification = nullptr;
|
||
}
|
||
|
||
/**
|
||
* @brief libtesla service exiting function to override libnx's
|
||
*
|
||
*/
|
||
void __appExit(void) {
|
||
delete tsl::notification;
|
||
eventClose(&tsl::notificationEvent);
|
||
|
||
smExit();
|
||
//socketExit();
|
||
//nifmExit();
|
||
spsmExit();
|
||
splExit();
|
||
fsdevUnmountAll();
|
||
|
||
#if USING_WIDGET_DIRECTIVE
|
||
i2cExit();
|
||
ult::powerExit(); // CUSTOM MODIFICATION
|
||
#endif
|
||
|
||
fsExit();
|
||
hidExit();
|
||
plExit();
|
||
pmdmntExit();
|
||
hidsysExit();
|
||
setsysExit();
|
||
|
||
// Final cleanup
|
||
tsl::gfx::FontManager::cleanup();
|
||
}
|
||
|
||
}
|
||
|
||
#endif |