workaround for time / battery % x position changing every few seconds.

fixes #142
This commit is contained in:
ITotalJustice
2025-05-24 23:13:53 +01:00
parent 22ebfd4a82
commit 1cdea981de
3 changed files with 68 additions and 57 deletions

View File

@@ -6,6 +6,16 @@
namespace sphaira::ui::menu { namespace sphaira::ui::menu {
struct PolledData {
struct tm tm{};
u32 battery_percetange{};
PsmChargerType charger_type{};
NifmInternetConnectionType type{};
NifmInternetConnectionStatus status{};
u32 strength{};
u32 ip{};
};
struct MenuBase : Widget { struct MenuBase : Widget {
MenuBase(std::string title); MenuBase(std::string title);
virtual ~MenuBase(); virtual ~MenuBase();
@@ -26,8 +36,7 @@ struct MenuBase : Widget {
return m_title; return m_title;
} }
private: static auto GetPolledData(bool force_refresh = false) -> PolledData;
void UpdateVars();
private: private:
std::string m_title{}; std::string m_title{};
@@ -36,16 +45,6 @@ private:
ScrollingText m_scroll_title_sub_heading{}; ScrollingText m_scroll_title_sub_heading{};
ScrollingText m_scroll_sub_heading{}; ScrollingText m_scroll_sub_heading{};
protected:
struct tm m_tm{};
TimeStamp m_poll_timestamp{};
u32 m_battery_percetange{};
PsmChargerType m_charger_type{};
NifmInternetConnectionType m_type{};
NifmInternetConnectionStatus m_status{};
u32 m_strength{};
u32 m_ip{};
}; };
} // namespace sphaira::ui::menu } // namespace sphaira::ui::menu

View File

@@ -239,9 +239,10 @@ void Menu::Draw(NVGcontext* vg, Theme* theme) {
mutexLock(&m_mutex); mutexLock(&m_mutex);
ON_SCOPE_EXIT(mutexUnlock(&m_mutex)); ON_SCOPE_EXIT(mutexUnlock(&m_mutex));
if (m_ip) { const auto pdata = GetPolledData();
if (m_type == NifmInternetConnectionType_WiFi) { if (pdata.ip) {
SetSubHeading("Connection Type: WiFi | Strength: "_i18n + std::to_string(m_strength)); if (pdata.type == NifmInternetConnectionType_WiFi) {
SetSubHeading("Connection Type: WiFi | Strength: "_i18n + std::to_string(pdata.strength));
} else { } else {
SetSubHeading("Connection Type: Ethernet"_i18n); SetSubHeading("Connection Type: Ethernet"_i18n);
} }
@@ -264,15 +265,15 @@ void Menu::Draw(NVGcontext* vg, Theme* theme) {
gfx::drawTextArgs(vg, bounds[2], start_y, font_size, NVG_ALIGN_LEFT | NVG_ALIGN_MIDDLE, theme->GetColour(ThemeEntryID_TEXT_SELECTED), __VA_ARGS__); \ gfx::drawTextArgs(vg, bounds[2], start_y, font_size, NVG_ALIGN_LEFT | NVG_ALIGN_MIDDLE, theme->GetColour(ThemeEntryID_TEXT_SELECTED), __VA_ARGS__); \
start_y += spacing; start_y += spacing;
if (m_ip) { if (pdata.ip) {
draw("Host:"_i18n, " %u.%u.%u.%u", m_ip&0xFF, (m_ip>>8)&0xFF, (m_ip>>16)&0xFF, (m_ip>>24)&0xFF); draw("Host:"_i18n, " %u.%u.%u.%u", pdata.ip&0xFF, (pdata.ip>>8)&0xFF, (pdata.ip>>16)&0xFF, (pdata.ip>>24)&0xFF);
draw("Port:"_i18n, " %u", m_port); draw("Port:"_i18n, " %u", m_port);
if (!m_anon) { if (!m_anon) {
draw("Username:"_i18n, " %s", m_user); draw("Username:"_i18n, " %s", m_user);
draw("Password:"_i18n, " %s", m_pass); draw("Password:"_i18n, " %s", m_pass);
} }
if (m_type == NifmInternetConnectionType_WiFi) { if (pdata.type == NifmInternetConnectionType_WiFi) {
NifmNetworkProfileData profile{}; NifmNetworkProfileData profile{};
if (R_SUCCEEDED(nifmGetCurrentNetworkProfile(&profile))) { if (R_SUCCEEDED(nifmGetCurrentNetworkProfile(&profile))) {
const auto& settings = profile.wireless_setting_data; const auto& settings = profile.wireless_setting_data;

View File

@@ -6,11 +6,44 @@
namespace sphaira::ui::menu { namespace sphaira::ui::menu {
auto MenuBase::GetPolledData(bool force_refresh) -> PolledData {
static PolledData data{};
static TimeStamp timestamp{};
static bool has_init = false;
if (!has_init) {
has_init = true;
force_refresh = true;
}
// update every second, do this in Draw because Update() isn't called if it
// doesn't have focus.
if (force_refresh || timestamp.GetSeconds() >= 1) {
data.tm = {};
data.battery_percetange = {};
data.charger_type = {};
data.type = {};
data.status = {};
data.strength = {};
data.ip = {};
const auto t = std::time(NULL);
localtime_r(&t, &data.tm);
psmGetBatteryChargePercentage(&data.battery_percetange);
psmGetChargerType(&data.charger_type);
nifmGetInternetConnectionStatus(&data.type, &data.strength, &data.status);
nifmGetCurrentIpAddress(&data.ip);
timestamp.Update();
}
return data;
}
MenuBase::MenuBase(std::string title) : m_title{title} { MenuBase::MenuBase(std::string title) : m_title{title} {
// this->SetParent(this); // this->SetParent(this);
this->SetPos(30, 87, 1220 - 30, 646 - 87); this->SetPos(30, 87, 1220 - 30, 646 - 87);
SetAction(Button::START, Action{App::Exit}); SetAction(Button::START, Action{App::Exit});
UpdateVars();
} }
MenuBase::~MenuBase() { MenuBase::~MenuBase() {
@@ -24,11 +57,7 @@ void MenuBase::Draw(NVGcontext* vg, Theme* theme) {
DrawElement(0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, ThemeEntryID_BACKGROUND); DrawElement(0, 0, SCREEN_WIDTH, SCREEN_HEIGHT, ThemeEntryID_BACKGROUND);
Widget::Draw(vg, theme); Widget::Draw(vg, theme);
// update every second, do this in Draw because Update() isn't called if it const auto pdata = GetPolledData();
// doesn't have focus.
if (m_poll_timestamp.GetSeconds() >= 1) {
UpdateVars();
}
const float start_y = 70; const float start_y = 70;
const float font_size = 22; const float font_size = 22;
@@ -39,28 +68,30 @@ void MenuBase::Draw(NVGcontext* vg, Theme* theme) {
nvgFontSize(vg, font_size); nvgFontSize(vg, font_size);
#define draw(colour, ...) \ #define draw(colour, fixed, ...) \
gfx::textBounds(vg, 0, 0, bounds, __VA_ARGS__); \ gfx::drawTextArgs(vg, start_x, start_y, font_size, NVG_ALIGN_RIGHT | NVG_ALIGN_BOTTOM, theme->GetColour(colour), __VA_ARGS__); \
start_x -= bounds[2] - bounds[0]; \ if (fixed) { \
gfx::drawTextArgs(vg, start_x, start_y, font_size, NVG_ALIGN_LEFT | NVG_ALIGN_BOTTOM, theme->GetColour(colour), __VA_ARGS__); \ start_x -= fixed; \
start_x -= spacing; } else { \
gfx::textBounds(vg, 0, 0, bounds, __VA_ARGS__); \
start_x -= spacing - (bounds[2] - bounds[0]); \
}
// draw("version %s", APP_VERSION); draw(ThemeEntryID_TEXT, 83, "%u\uFE6A", pdata.battery_percetange);
draw(ThemeEntryID_TEXT, "%u\uFE6A", m_battery_percetange);
if (App::Get12HourTimeEnable()) { if (App::Get12HourTimeEnable()) {
draw(ThemeEntryID_TEXT, "%02u:%02u:%02u %s", (m_tm.tm_hour == 0 || m_tm.tm_hour == 12) ? 12 : m_tm.tm_hour % 12, m_tm.tm_min, m_tm.tm_sec, (m_tm.tm_hour < 12) ? "AM" : "PM"); draw(ThemeEntryID_TEXT, 175, "%02u:%02u:%02u %s", (pdata.tm.tm_hour == 0 || pdata.tm.tm_hour == 12) ? 12 : pdata.tm.tm_hour % 12, pdata.tm.tm_min, pdata.tm.tm_sec, (pdata.tm.tm_hour < 12) ? "AM" : "PM");
} else { } else {
draw(ThemeEntryID_TEXT, "%02u:%02u:%02u", m_tm.tm_hour, m_tm.tm_min, m_tm.tm_sec); draw(ThemeEntryID_TEXT, 133, "%02u:%02u:%02u", pdata.tm.tm_hour, pdata.tm.tm_min, pdata.tm.tm_sec);
} }
if (m_ip) { if (pdata.ip) {
draw(ThemeEntryID_TEXT, "%u.%u.%u.%u", m_ip&0xFF, (m_ip>>8)&0xFF, (m_ip>>16)&0xFF, (m_ip>>24)&0xFF); draw(ThemeEntryID_TEXT, 0, "%u.%u.%u.%u", pdata.ip&0xFF, (pdata.ip>>8)&0xFF, (pdata.ip>>16)&0xFF, (pdata.ip>>24)&0xFF);
} else { } else {
draw(ThemeEntryID_TEXT, ("No Internet"_i18n).c_str()); draw(ThemeEntryID_TEXT, 0, ("No Internet"_i18n).c_str());
} }
if (!App::IsApplication()) { if (!App::IsApplication()) {
draw(ThemeEntryID_ERROR, ("[Applet Mode]"_i18n).c_str()); draw(ThemeEntryID_ERROR, 0, ("[Applet Mode]"_i18n).c_str());
} }
#undef draw #undef draw
@@ -91,24 +122,4 @@ void MenuBase::SetSubHeading(std::string sub_heading) {
m_sub_heading = sub_heading; m_sub_heading = sub_heading;
} }
void MenuBase::UpdateVars() {
m_tm = {};
m_poll_timestamp = {};
m_battery_percetange = {};
m_charger_type = {};
m_type = {};
m_status = {};
m_strength = {};
m_ip = {};
const auto t = time(NULL);
localtime_r(&t, &m_tm);
psmGetBatteryChargePercentage(&m_battery_percetange);
psmGetChargerType(&m_charger_type);
nifmGetInternetConnectionStatus(&m_type, &m_strength, &m_status);
nifmGetCurrentIpAddress(&m_ip);
m_poll_timestamp.Update();
}
} // namespace sphaira::ui::menu } // namespace sphaira::ui::menu