#include "ScreenshotTask.h" #include "lv_screenshot.h" #include "app/AppContext.h" #include "Mutex.h" #include "TactilityCore.h" #include "Thread.h" #include "service/loader/Loader.h" #include "lvgl/LvglSync.h" namespace tt::service::screenshot::task { #define TAG "screenshot_task" #define TASK_WORK_TYPE_DELAY 1 #define TASK_WORK_TYPE_APPS 2 #define SCREENSHOT_PATH_LIMIT 128 typedef struct { int type; uint8_t delay_in_seconds; uint8_t amount; char path[SCREENSHOT_PATH_LIMIT]; } ScreenshotTaskWork; typedef struct { Thread* thread; Mutex* mutex; bool interrupted; ScreenshotTaskWork work; } ScreenshotTaskData; static void task_lock(ScreenshotTaskData* data) { tt_check(tt_mutex_acquire(data->mutex, TtWaitForever) == TtStatusOk); } static void task_unlock(ScreenshotTaskData* data) { tt_check(tt_mutex_release(data->mutex) == TtStatusOk); } ScreenshotTask* alloc() { auto* data = static_cast(malloc(sizeof(ScreenshotTaskData))); *data = (ScreenshotTaskData) { .thread = nullptr, .mutex = tt_mutex_alloc(Mutex::TypeRecursive), .interrupted = false }; return data; } void free(ScreenshotTask* task) { auto* data = static_cast(task); if (data->thread) { stop(data); } } static bool is_interrupted(ScreenshotTaskData* data) { task_lock(data); bool interrupted = data->interrupted; task_unlock(data); return interrupted; } static int32_t screenshot_task(void* context) { auto* data = static_cast(context); bool interrupted = false; uint8_t screenshots_taken = 0; std::string last_app_id; while (!interrupted) { interrupted = is_interrupted(data); if (data->work.type == TASK_WORK_TYPE_DELAY) { // Splitting up the delays makes it easier to stop the service for (int i = 0; i < (data->work.delay_in_seconds * 10) && !is_interrupted(data); ++i){ kernel::delayMillis(100); } if (is_interrupted(data)) { break; } screenshots_taken++; char filename[SCREENSHOT_PATH_LIMIT + 32]; sprintf(filename, "%s/screenshot-%d.png", data->work.path, screenshots_taken); lvgl::lock(TtWaitForever); if (lv_screenshot_create(lv_scr_act(), LV_COLOR_FORMAT_NATIVE, LV_100ASK_SCREENSHOT_SV_PNG, filename)){ TT_LOG_I(TAG, "Screenshot saved to %s", filename); } else { TT_LOG_E(TAG, "Screenshot not saved to %s", filename); } lvgl::unlock(); if (data->work.amount > 0 && screenshots_taken >= data->work.amount) { break; // Interrupted loop } } else if (data->work.type == TASK_WORK_TYPE_APPS) { app::AppContext* _Nullable app = loader::getCurrentApp(); if (app) { const app::AppManifest& manifest = app->getManifest(); if (manifest.id != last_app_id) { kernel::delayMillis(100); last_app_id = manifest.id; char filename[SCREENSHOT_PATH_LIMIT + 32]; sprintf(filename, "%s/screenshot-%s.png", data->work.path, manifest.id.c_str()); lvgl::lock(TtWaitForever); if (lv_screenshot_create(lv_scr_act(), LV_COLOR_FORMAT_NATIVE, LV_100ASK_SCREENSHOT_SV_PNG, filename)){ TT_LOG_I(TAG, "Screenshot saved to %s", filename); } else { TT_LOG_E(TAG, "Screenshot not saved to %s", filename); } lvgl::unlock(); } } kernel::delayMillis(250); } } return 0; } static void task_start(ScreenshotTaskData* data) { task_lock(data); tt_check(data->thread == nullptr); data->thread = new Thread( "screenshot", 8192, &screenshot_task, data ); data->thread->start(); task_unlock(data); } void startApps(ScreenshotTask* task, const char* path) { tt_check(strlen(path) < (SCREENSHOT_PATH_LIMIT - 1)); auto* data = static_cast(task); task_lock(data); if (data->thread == nullptr) { data->interrupted = false; data->work.type = TASK_WORK_TYPE_APPS; strcpy(data->work.path, path); task_start(data); } else { TT_LOG_E(TAG, "Task was already running"); } task_unlock(data); } void startTimed(ScreenshotTask* task, const char* path, uint8_t delay_in_seconds, uint8_t amount) { tt_check(strlen(path) < (SCREENSHOT_PATH_LIMIT - 1)); auto* data = static_cast(task); task_lock(data); if (data->thread == nullptr) { data->interrupted = false; data->work.type = TASK_WORK_TYPE_DELAY; data->work.delay_in_seconds = delay_in_seconds; data->work.amount = amount; strcpy(data->work.path, path); task_start(data); } else { TT_LOG_E(TAG, "Task was already running"); } task_unlock(data); } void stop(ScreenshotTask* task) { auto* data = static_cast(task); if (data->thread != nullptr) { task_lock(data); data->interrupted = true; task_unlock(data); data->thread->join(); task_lock(data); delete data->thread; data->thread = nullptr; task_unlock(data); } } } // namespace