- 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.
292 lines
9.4 KiB
C++
292 lines
9.4 KiB
C++
#include "TactilityConfig.h"
|
|
#include <Timer.h>
|
|
#include <kernel/Kernel.h>
|
|
|
|
#if TT_FEATURE_SCREENSHOT_ENABLED
|
|
|
|
#include "TactilityHeadless.h"
|
|
#include "app/App.h"
|
|
#include "app/AppManifest.h"
|
|
#include "lvgl/LvglSync.h"
|
|
#include "lvgl/Toolbar.h"
|
|
#include "service/gui/Gui.h"
|
|
#include "service/loader/Loader.h"
|
|
#include "service/screenshot/Screenshot.h"
|
|
|
|
#define TAG "screenshot"
|
|
|
|
namespace tt::app::screenshot {
|
|
|
|
extern const AppManifest manifest;
|
|
|
|
class ScreenshotApp : public App {
|
|
|
|
lv_obj_t* modeDropdown = nullptr;
|
|
lv_obj_t* pathTextArea = nullptr;
|
|
lv_obj_t* startStopButtonLabel = nullptr;
|
|
lv_obj_t* timerWrapper = nullptr;
|
|
lv_obj_t* delayTextArea = nullptr;
|
|
std::unique_ptr<Timer> updateTimer;
|
|
|
|
void createTimerSettingsWidgets(lv_obj_t* parent);
|
|
void createModeSettingWidgets(lv_obj_t* parent);
|
|
void createFilePathWidgets(lv_obj_t* parent);
|
|
|
|
void updateScreenshotMode();
|
|
|
|
public:
|
|
|
|
ScreenshotApp();
|
|
~ScreenshotApp();
|
|
|
|
void onShow(AppContext& app, lv_obj_t* parent) override;
|
|
void onStartPressed();
|
|
void onModeSet();
|
|
void onTimerTick();
|
|
};
|
|
|
|
|
|
/** 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<ScreenshotApp> _Nullable optApp() {
|
|
auto appContext = service::loader::getCurrentAppContext();
|
|
if (appContext != nullptr && appContext->getManifest().id == manifest.id) {
|
|
return std::static_pointer_cast<ScreenshotApp>(appContext->getApp());
|
|
} else {
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
static void onStartPressedCallback(TT_UNUSED lv_event_t* event) {
|
|
auto app = optApp();
|
|
if (app != nullptr) {
|
|
app->onStartPressed();
|
|
}
|
|
}
|
|
|
|
static void onModeSetCallback(TT_UNUSED lv_event_t* event) {
|
|
auto app = optApp();
|
|
if (app != nullptr) {
|
|
app->onModeSet();
|
|
}
|
|
}
|
|
|
|
static void onTimerCallback(TT_UNUSED std::shared_ptr<void> context) {
|
|
auto app = optApp();
|
|
if (app != nullptr) {
|
|
app->onTimerTick();
|
|
}
|
|
}
|
|
|
|
ScreenshotApp::ScreenshotApp() {
|
|
updateTimer = std::make_unique<Timer>(Timer::Type::Periodic, onTimerCallback, nullptr);
|
|
}
|
|
|
|
ScreenshotApp::~ScreenshotApp() {
|
|
if (updateTimer->isRunning()) {
|
|
updateTimer->stop();
|
|
}
|
|
}
|
|
|
|
void ScreenshotApp::onTimerTick() {
|
|
auto lvgl_lock = lvgl::getLvglSyncLockable()->scoped();
|
|
if (lvgl_lock->lock(50 / portTICK_PERIOD_MS)) {
|
|
updateScreenshotMode();
|
|
}
|
|
}
|
|
|
|
void ScreenshotApp::onModeSet() {
|
|
updateScreenshotMode();
|
|
}
|
|
|
|
void ScreenshotApp::onStartPressed() {
|
|
auto service = service::screenshot::optScreenshotService();
|
|
if (service == nullptr) {
|
|
TT_LOG_E(TAG, "Service not found/running");
|
|
return;
|
|
}
|
|
|
|
if (service->isTaskStarted()) {
|
|
TT_LOG_I(TAG, "Stop screenshot");
|
|
service->stop();
|
|
} else {
|
|
uint32_t selected = lv_dropdown_get_selected(modeDropdown);
|
|
const char* path = lv_textarea_get_text(pathTextArea);
|
|
if (selected == 0) {
|
|
TT_LOG_I(TAG, "Start timed screenshots");
|
|
const char* delay_text = lv_textarea_get_text(delayTextArea);
|
|
int delay = atoi(delay_text);
|
|
if (delay > 0) {
|
|
service->startTimed(path, delay, 1);
|
|
} else {
|
|
TT_LOG_W(TAG, "Ignored screenshot start because delay was 0");
|
|
}
|
|
} else {
|
|
TT_LOG_I(TAG, "Start app screenshots");
|
|
service->startApps(path);
|
|
}
|
|
}
|
|
|
|
updateScreenshotMode();
|
|
}
|
|
|
|
void ScreenshotApp::updateScreenshotMode() {
|
|
auto service = service::screenshot::optScreenshotService();
|
|
if (service == nullptr) {
|
|
TT_LOG_E(TAG, "Service not found/running");
|
|
return;
|
|
}
|
|
|
|
lv_obj_t* label = startStopButtonLabel;
|
|
if (service->isTaskStarted()) {
|
|
lv_label_set_text(label, "Stop");
|
|
} else {
|
|
lv_label_set_text(label, "Start");
|
|
}
|
|
|
|
uint32_t selected = lv_dropdown_get_selected(modeDropdown);
|
|
if (selected == 0) { // Timer
|
|
lv_obj_remove_flag(timerWrapper, LV_OBJ_FLAG_HIDDEN);
|
|
} else {
|
|
lv_obj_add_flag(timerWrapper, LV_OBJ_FLAG_HIDDEN);
|
|
}
|
|
}
|
|
|
|
|
|
void ScreenshotApp::createModeSettingWidgets(lv_obj_t* parent) {
|
|
auto service = service::screenshot::optScreenshotService();
|
|
if (service == nullptr) {
|
|
TT_LOG_E(TAG, "Service not found/running");
|
|
return;
|
|
}
|
|
|
|
auto* mode_wrapper = lv_obj_create(parent);
|
|
lv_obj_set_size(mode_wrapper, LV_PCT(100), LV_SIZE_CONTENT);
|
|
lv_obj_set_style_pad_all(mode_wrapper, 0, 0);
|
|
lv_obj_set_style_border_width(mode_wrapper, 0, 0);
|
|
|
|
auto* mode_label = lv_label_create(mode_wrapper);
|
|
lv_label_set_text(mode_label, "Mode:");
|
|
lv_obj_align(mode_label, LV_ALIGN_LEFT_MID, 0, 0);
|
|
|
|
modeDropdown = lv_dropdown_create(mode_wrapper);
|
|
lv_dropdown_set_options(modeDropdown, "Timer\nApp start");
|
|
lv_obj_align_to(modeDropdown, mode_label, LV_ALIGN_OUT_RIGHT_MID, 8, 0);
|
|
lv_obj_add_event_cb(modeDropdown, onModeSetCallback, LV_EVENT_VALUE_CHANGED, nullptr);
|
|
service::screenshot::Mode mode = service->getMode();
|
|
if (mode == service::screenshot::Mode::Apps) {
|
|
lv_dropdown_set_selected(modeDropdown, 1);
|
|
}
|
|
|
|
auto* button = lv_button_create(mode_wrapper);
|
|
lv_obj_align(button, LV_ALIGN_RIGHT_MID, 0, 0);
|
|
lv_obj_add_event_cb(button, &onStartPressedCallback, LV_EVENT_SHORT_CLICKED, nullptr);
|
|
startStopButtonLabel = lv_label_create(button);
|
|
lv_obj_align(startStopButtonLabel, LV_ALIGN_CENTER, 0, 0);
|
|
}
|
|
|
|
void ScreenshotApp::createFilePathWidgets(lv_obj_t* parent) {
|
|
auto* path_wrapper = lv_obj_create(parent);
|
|
lv_obj_set_size(path_wrapper, LV_PCT(100), LV_SIZE_CONTENT);
|
|
lv_obj_set_style_pad_all(path_wrapper, 0, 0);
|
|
lv_obj_set_style_border_width(path_wrapper, 0, 0);
|
|
lv_obj_set_flex_flow(path_wrapper, LV_FLEX_FLOW_ROW);
|
|
|
|
auto* label_wrapper = lv_obj_create(path_wrapper);
|
|
lv_obj_set_style_border_width(label_wrapper, 0, 0);
|
|
lv_obj_set_style_pad_all(label_wrapper, 0, 0);
|
|
lv_obj_set_size(label_wrapper, 44, 36);
|
|
auto* path_label = lv_label_create(label_wrapper);
|
|
lv_label_set_text(path_label, "Path:");
|
|
lv_obj_align(path_label, LV_ALIGN_LEFT_MID, 0, 0);
|
|
|
|
pathTextArea = lv_textarea_create(path_wrapper);
|
|
lv_textarea_set_one_line(pathTextArea, true);
|
|
lv_obj_set_flex_grow(pathTextArea, 1);
|
|
if (kernel::getPlatform() == kernel::PlatformEsp) {
|
|
auto sdcard = tt::hal::getConfiguration()->sdcard;
|
|
if (sdcard != nullptr && sdcard->getState() == hal::SdCard::State::Mounted) {
|
|
lv_textarea_set_text(pathTextArea, "A:/sdcard");
|
|
} else {
|
|
lv_textarea_set_text(pathTextArea, "Error: no SD card");
|
|
}
|
|
} else { // PC
|
|
lv_textarea_set_text(pathTextArea, "A:");
|
|
}
|
|
}
|
|
|
|
void ScreenshotApp::createTimerSettingsWidgets(lv_obj_t* parent) {
|
|
timerWrapper = lv_obj_create(parent);
|
|
lv_obj_set_size(timerWrapper, LV_PCT(100), LV_SIZE_CONTENT);
|
|
lv_obj_set_style_pad_all(timerWrapper, 0, 0);
|
|
lv_obj_set_style_border_width(timerWrapper, 0, 0);
|
|
|
|
auto* delay_wrapper = lv_obj_create(timerWrapper);
|
|
lv_obj_set_size(delay_wrapper, LV_PCT(100), LV_SIZE_CONTENT);
|
|
lv_obj_set_style_pad_all(delay_wrapper, 0, 0);
|
|
lv_obj_set_style_border_width(delay_wrapper, 0, 0);
|
|
lv_obj_set_flex_flow(delay_wrapper, LV_FLEX_FLOW_ROW);
|
|
|
|
auto* delay_label_wrapper = lv_obj_create(delay_wrapper);
|
|
lv_obj_set_style_border_width(delay_label_wrapper, 0, 0);
|
|
lv_obj_set_style_pad_all(delay_label_wrapper, 0, 0);
|
|
lv_obj_set_size(delay_label_wrapper, 44, 36);
|
|
auto* delay_label = lv_label_create(delay_label_wrapper);
|
|
lv_label_set_text(delay_label, "Delay:");
|
|
lv_obj_align(delay_label, LV_ALIGN_LEFT_MID, 0, 0);
|
|
|
|
delayTextArea = lv_textarea_create(delay_wrapper);
|
|
lv_textarea_set_one_line(delayTextArea, true);
|
|
lv_textarea_set_accepted_chars(delayTextArea, "0123456789");
|
|
lv_textarea_set_text(delayTextArea, "10");
|
|
lv_obj_set_flex_grow(delayTextArea, 1);
|
|
|
|
auto* delay_unit_label_wrapper = lv_obj_create(delay_wrapper);
|
|
lv_obj_set_style_border_width(delay_unit_label_wrapper, 0, 0);
|
|
lv_obj_set_style_pad_all(delay_unit_label_wrapper, 0, 0);
|
|
lv_obj_set_size(delay_unit_label_wrapper, LV_SIZE_CONTENT, 36);
|
|
auto* delay_unit_label = lv_label_create(delay_unit_label_wrapper);
|
|
lv_obj_align(delay_unit_label, LV_ALIGN_LEFT_MID, 0, 0);
|
|
lv_label_set_text(delay_unit_label, "seconds");
|
|
}
|
|
|
|
void ScreenshotApp::onShow(AppContext& appContext, lv_obj_t* parent) {
|
|
if (updateTimer->isRunning()) {
|
|
updateTimer->stop();
|
|
}
|
|
|
|
lv_obj_set_flex_flow(parent, LV_FLEX_FLOW_COLUMN);
|
|
auto* toolbar = lvgl::toolbar_create(parent, appContext);
|
|
lv_obj_align(toolbar, LV_ALIGN_TOP_MID, 0, 0);
|
|
|
|
auto* wrapper = lv_obj_create(parent);
|
|
lv_obj_set_width(wrapper, LV_PCT(100));
|
|
lv_obj_set_flex_grow(wrapper, 1);
|
|
lv_obj_set_style_border_width(wrapper, 0, 0);
|
|
lv_obj_set_flex_flow(wrapper, LV_FLEX_FLOW_COLUMN);
|
|
|
|
createModeSettingWidgets(wrapper);
|
|
createFilePathWidgets(wrapper);
|
|
createTimerSettingsWidgets(wrapper);
|
|
|
|
service::gui::keyboardAddTextArea(delayTextArea);
|
|
service::gui::keyboardAddTextArea(pathTextArea);
|
|
|
|
updateScreenshotMode();
|
|
|
|
if (!updateTimer->isRunning()) {
|
|
updateTimer->start(500 / portTICK_PERIOD_MS);
|
|
}
|
|
}
|
|
|
|
extern const AppManifest manifest = {
|
|
.id = "Screenshot",
|
|
.name = "Screenshot",
|
|
.icon = LV_SYMBOL_IMAGE,
|
|
.type = Type::System,
|
|
.createApp = create<ScreenshotApp>
|
|
};
|
|
|
|
} // namespace
|
|
|
|
#endif
|