- Refactor the way apps work: Instead of a C interface, they are now C++ classes. The main reasoning is that attaching data to an app was cumbersome. Having different implementations for different kinds of apps was cumbersome too. (3 or 4 layers of manifest nesting for the TactilityC project) - External apps are still written in C, but they get a createData/destroyData in their manifest, so: - External apps now have their own manifest. - All functions in the original AppManifest are removed and replaced by a single `createApp` function - External apps now automatically register (each app individually!) when they run the first time. As a side-effect they become visible in the `AppList` app! - Adapted all apps for the new interface. - Adapted all internal logic for these changes (Gui, ViewPort, Loader, AppContext, AppInstance, etc.) - Rewrote parts of Loader to use std::shared_ptr to make the code much safer. - Added a refcount check for the `AppInstance` and `App` at the end of their lifecycle. Show warning if refcount is too high.
188 lines
6.3 KiB
C++
188 lines
6.3 KiB
C++
#include "app/AppContext.h"
|
|
#include "Assets.h"
|
|
#include "lvgl.h"
|
|
#include "Tactility.h"
|
|
#include "Timer.h"
|
|
#include "lvgl/LvglSync.h"
|
|
#include "lvgl/Style.h"
|
|
#include "lvgl/Toolbar.h"
|
|
#include "service/loader/Loader.h"
|
|
|
|
namespace tt::app::power {
|
|
|
|
#define TAG "power"
|
|
|
|
extern const AppManifest manifest;
|
|
static void onTimer(TT_UNUSED std::shared_ptr<void> context);
|
|
|
|
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<PowerApp> _Nullable optApp() {
|
|
auto appContext = service::loader::getCurrentAppContext();
|
|
if (appContext != nullptr && appContext->getManifest().id == manifest.id) {
|
|
return std::static_pointer_cast<PowerApp>(appContext->getApp());
|
|
} else {
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
class PowerApp : public App {
|
|
|
|
private:
|
|
|
|
Timer update_timer = Timer(Timer::Type::Periodic, &onTimer, nullptr);
|
|
std::shared_ptr<tt::hal::Power> power = getConfiguration()->hardware->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(TT_UNUSED std::shared_ptr<void> context) {
|
|
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_obj_t*>(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::MetricData metric_data;
|
|
if (power->getMetric(hal::Power::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::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::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::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 onShow(AppContext& app, lv_obj_t* parent) override {
|
|
lv_obj_set_flex_flow(parent, LV_FLEX_FLOW_COLUMN);
|
|
|
|
lvgl::toolbar_create(parent, app);
|
|
|
|
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);
|
|
lvgl::obj_set_style_no_padding(switch_container);
|
|
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<PowerApp>
|
|
};
|
|
|
|
} // namespace
|