#include "Tactility/app/AppContext.h" #include "Tactility/lvgl/LvglSync.h" #include "Tactility/lvgl/Style.h" #include "Tactility/lvgl/Toolbar.h" #include "Tactility/service/loader/Loader.h" #include "Tactility/hal/power/PowerDevice.h" #include #include #include #include namespace tt::app::power { #define TAG "power" extern const AppManifest manifest; class PowerApp; /** Returns the app data if the app is active. Note that this could clash if the same app is started twice and a background thread is slow. */ std::shared_ptr _Nullable optApp() { auto appContext = getCurrentAppContext(); if (appContext != nullptr && appContext->getManifest().id == manifest.id) { return std::static_pointer_cast(appContext->getApp()); } else { return nullptr; } } class PowerApp : public App { Timer update_timer = Timer(Timer::Type::Periodic, []() { onTimer(); }); std::shared_ptr power; lv_obj_t* enableLabel = nullptr; lv_obj_t* enableSwitch = nullptr; lv_obj_t* batteryVoltageLabel = nullptr; lv_obj_t* chargeStateLabel = nullptr; lv_obj_t* chargeLevelLabel = nullptr; lv_obj_t* currentLabel = nullptr; static void onTimer() { auto app = optApp(); if (app != nullptr) { app->updateUi(); } } void onPowerEnabledChanged(lv_event_t* event) { lv_event_code_t code = lv_event_get_code(event); auto* enable_switch = static_cast(lv_event_get_target(event)); if (code == LV_EVENT_VALUE_CHANGED) { bool is_on = lv_obj_has_state(enable_switch, LV_STATE_CHECKED); if (power->isAllowedToCharge() != is_on) { power->setAllowedToCharge(is_on); updateUi(); } } } static void onPowerEnabledChangedCallback(lv_event_t* event) { auto* app = (PowerApp*)lv_event_get_user_data(event); app->onPowerEnabledChanged(event); } void updateUi() { const char* charge_state; hal::power::PowerDevice::MetricData metric_data; if (power->getMetric(hal::power::PowerDevice::MetricType::IsCharging, metric_data)) { charge_state = metric_data.valueAsBool ? "yes" : "no"; } else { charge_state = "N/A"; } uint8_t charge_level; bool charge_level_scaled_set = false; if (power->getMetric(hal::power::PowerDevice::MetricType::ChargeLevel, metric_data)) { charge_level = metric_data.valueAsUint8; charge_level_scaled_set = true; } bool charging_enabled_set = power->supportsChargeControl(); bool charging_enabled_and_allowed = power->supportsChargeControl() && power->isAllowedToCharge(); int32_t current; bool current_set = false; if (power->getMetric(hal::power::PowerDevice::MetricType::Current, metric_data)) { current = metric_data.valueAsInt32; current_set = true; } uint32_t battery_voltage; bool battery_voltage_set = false; if (power->getMetric(hal::power::PowerDevice::MetricType::BatteryVoltage, metric_data)) { battery_voltage = metric_data.valueAsUint32; battery_voltage_set = true; } lvgl::lock(kernel::millisToTicks(1000)); if (charging_enabled_set) { lv_obj_set_state(enableSwitch, LV_STATE_CHECKED, charging_enabled_and_allowed); lv_obj_remove_flag(enableSwitch, LV_OBJ_FLAG_HIDDEN); lv_obj_remove_flag(enableLabel, LV_OBJ_FLAG_HIDDEN); } else { lv_obj_add_flag(enableSwitch, LV_OBJ_FLAG_HIDDEN); lv_obj_add_flag(enableLabel, LV_OBJ_FLAG_HIDDEN); } lv_label_set_text_fmt(chargeStateLabel, "Charging: %s", charge_state); if (battery_voltage_set) { lv_label_set_text_fmt(batteryVoltageLabel, "Battery voltage: %lu mV", battery_voltage); } else { lv_label_set_text_fmt(batteryVoltageLabel, "Battery voltage: N/A"); } if (charge_level_scaled_set) { lv_label_set_text_fmt(chargeLevelLabel, "Charge level: %d%%", charge_level); } else { lv_label_set_text_fmt(chargeLevelLabel, "Charge level: N/A"); } if (current_set) { lv_label_set_text_fmt(currentLabel, "Current: %ld mAh", current); } else { lv_label_set_text_fmt(currentLabel, "Current: N/A"); } lvgl::unlock(); } public: void onCreate(AppContext& app) override { power = hal::findFirstDevice(hal::Device::Type::Power); } void onShow(AppContext& app, lv_obj_t* parent) override { lv_obj_set_flex_flow(parent, LV_FLEX_FLOW_COLUMN); lv_obj_set_style_pad_row(parent, 0, LV_STATE_DEFAULT); lvgl::toolbar_create(parent, app); if (power == nullptr) { return; } lv_obj_t* wrapper = lv_obj_create(parent); lv_obj_set_width(wrapper, LV_PCT(100)); lv_obj_set_style_border_width(wrapper, 0, 0); lv_obj_set_flex_grow(wrapper, 1); lv_obj_set_flex_flow(wrapper, LV_FLEX_FLOW_COLUMN); // Top row: enable/disable lv_obj_t* switch_container = lv_obj_create(wrapper); lv_obj_set_width(switch_container, LV_PCT(100)); lv_obj_set_height(switch_container, LV_SIZE_CONTENT); lv_obj_set_style_pad_all(switch_container, 0, 0); lv_obj_set_style_pad_gap(switch_container, 0, 0); lvgl::obj_set_style_bg_invisible(switch_container); enableLabel = lv_label_create(switch_container); lv_label_set_text(enableLabel, "Charging enabled"); lv_obj_set_align(enableLabel, LV_ALIGN_LEFT_MID); lv_obj_t* enable_switch = lv_switch_create(switch_container); lv_obj_add_event_cb(enable_switch, onPowerEnabledChangedCallback, LV_EVENT_VALUE_CHANGED, this); lv_obj_set_align(enable_switch, LV_ALIGN_RIGHT_MID); enableSwitch = enable_switch; chargeStateLabel = lv_label_create(wrapper); chargeLevelLabel = lv_label_create(wrapper); batteryVoltageLabel = lv_label_create(wrapper); currentLabel = lv_label_create(wrapper); updateUi(); update_timer.start(kernel::millisToTicks(1000)); } void onHide(TT_UNUSED AppContext& app) override { update_timer.stop(); } }; extern const AppManifest manifest = { .id = "Power", .name = "Power", .icon = TT_ASSETS_APP_ICON_POWER_SETTINGS, .type = Type::Settings, .createApp = create }; } // namespace