## Launcher - Launcher now has optional power button to show - Launcher layout improvements - Removed text from Launcher (translations with larger amounts of text did not fit small device formats) ## T-Lora Pager - Implement power off (created `BQ25896` driver) - Implemented haptics (created `DRV2605` driver project) and buzz on startup - Reversed scroll wheel - Created `TloraEncoder` device and relocated its logic from `TloraKeyboard` - Disabled SPIRAM test to save 0.5 seconds of boot time (current boot time is very slow) - Update `ST7796` esp_lcd driver to v1.3.4 - Fixed keyboard bug: delete queue in destructor - Fixed driver dependencies: Avoiding usage of global static shared_ptr. Properly constructor-inject everywhere, or use `tt::hal::findDevices()` - I2C configuration is now immutable (you cannot disable it anymore from the I2C Settings app, as it would break crucial drivers) - Renamed I2C and UART subsystems to "Internal" ## Drivers - On/off interface added to `PowerDevice` - Created `tt::hal::Configuration.createDevices`, which is intended to replace all custom create calls for display, keyboard, etc. - Created `EncoderDevice` as a `Device` subtype ## Other Improvements - Changed `findDevices(type, function)` into a templatized function. - Improved SD card mounting ## Fixes - Show Screenshot app again - Fixed Statusbar: some updates were allowed to time out and fail silently: When the Statusbar service would do a state update, the LVGL statusbar would never get updated due to this timeout. - Fixed memory leaks in all `createSdCard()` functions (in most board implementations)
245 lines
7.6 KiB
C++
245 lines
7.6 KiB
C++
#include <Tactility/app/display/DisplaySettings.h>
|
|
#include <Tactility/hal/Configuration.h>
|
|
#include <Tactility/hal/encoder/EncoderDevice.h>
|
|
#include <Tactility/hal/display/DisplayDevice.h>
|
|
#include <Tactility/hal/keyboard/KeyboardDevice.h>
|
|
#include <Tactility/hal/touch/TouchDevice.h>
|
|
#include <Tactility/lvgl/Keyboard.h>
|
|
#include <Tactility/lvgl/Lvgl.h>
|
|
#include <Tactility/lvgl/LvglSync.h>
|
|
#include <Tactility/kernel/SystemEvents.h>
|
|
#include <Tactility/service/ServiceRegistration.h>
|
|
#include <Tactility/TactilityHeadless.h>
|
|
|
|
#ifdef ESP_PLATFORM
|
|
#include <Tactility/lvgl/EspLvglPort.h>
|
|
#endif
|
|
|
|
#include <lvgl.h>
|
|
|
|
namespace tt::lvgl {
|
|
|
|
constexpr auto* TAG = "Lvgl";
|
|
|
|
static bool started = false;
|
|
|
|
// TODO: Move to hal init
|
|
static void initDisplays(const hal::Configuration& config) {
|
|
TT_LOG_I(TAG, "Init displays");
|
|
if (config.createDisplay != nullptr) {
|
|
auto display = config.createDisplay();
|
|
if (display != nullptr) {
|
|
hal::registerDevice(display);
|
|
}
|
|
}
|
|
|
|
TT_LOG_I(TAG, "Start displays");
|
|
auto displays = hal::findDevices<hal::display::DisplayDevice>(hal::Device::Type::Display);
|
|
for (auto& display : displays) {
|
|
if (!display->start()) {
|
|
TT_LOG_E(TAG, "Display start failed");
|
|
}
|
|
|
|
if (display->supportsBacklightDuty()) {
|
|
display->setBacklightDuty(0);
|
|
}
|
|
|
|
auto touch = display->getTouchDevice();
|
|
if (touch != nullptr) {
|
|
hal::registerDevice(touch);
|
|
touch->start();
|
|
}
|
|
}
|
|
}
|
|
|
|
void init(const hal::Configuration& config) {
|
|
TT_LOG_I(TAG, "Init started");
|
|
|
|
#ifdef ESP_PLATFORM
|
|
if (config.lvglInit == hal::LvglInit::Default && !initEspLvglPort()) {
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
initDisplays(config);
|
|
|
|
start();
|
|
|
|
TT_LOG_I(TAG, "Init finished");
|
|
}
|
|
|
|
bool isStarted() {
|
|
return started;
|
|
}
|
|
|
|
void start() {
|
|
TT_LOG_I(TAG, "Start LVGL");
|
|
|
|
if (started) {
|
|
TT_LOG_W(TAG, "Can't start LVGL twice");
|
|
return;
|
|
}
|
|
|
|
auto lock = getSyncLock()->asScopedLock();
|
|
lock.lock();
|
|
|
|
// Start displays (their related touch devices start automatically within)
|
|
|
|
TT_LOG_I(TAG, "Start displays");
|
|
auto displays = hal::findDevices<hal::display::DisplayDevice>(hal::Device::Type::Display);
|
|
for (auto display : displays) {
|
|
if (display->supportsLvgl()) {
|
|
if (display->startLvgl()) {
|
|
TT_LOG_I(TAG, "Started %s", display->getName().c_str());
|
|
auto lvgl_display = display->getLvglDisplay();
|
|
assert(lvgl_display != nullptr);
|
|
lv_display_rotation_t rotation = app::display::getRotation();
|
|
if (rotation != lv_display_get_rotation(lvgl_display)) {
|
|
lv_display_set_rotation(lvgl_display, rotation);
|
|
}
|
|
} else {
|
|
TT_LOG_E(TAG, "Start failed for %s", display->getName().c_str());
|
|
}
|
|
}
|
|
}
|
|
|
|
// Start touch
|
|
|
|
TT_LOG_I(TAG, "Start touch devices");
|
|
auto touch_devices = hal::findDevices<hal::touch::TouchDevice>(hal::Device::Type::Touch);
|
|
for (auto touch_device : touch_devices) {
|
|
if (displays.size() > 0) {
|
|
// TODO: Consider implementing support for multiple displays
|
|
auto display = displays[0];
|
|
// Start any touch devices that haven't been started yet
|
|
if (touch_device->supportsLvgl() && touch_device->getLvglIndev() == nullptr) {
|
|
if (touch_device->startLvgl(display->getLvglDisplay())) {
|
|
TT_LOG_I(TAG, "Started %s", touch_device->getName().c_str());
|
|
} else {
|
|
TT_LOG_E(TAG, "Start failed for %s", touch_device->getName().c_str());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Start keyboards
|
|
TT_LOG_I(TAG, "Start keyboards");
|
|
auto keyboards = hal::findDevices<hal::keyboard::KeyboardDevice>(hal::Device::Type::Keyboard);
|
|
for (auto keyboard : keyboards) {
|
|
if (displays.size() > 0) {
|
|
// TODO: Consider implementing support for multiple displays
|
|
auto display = displays[0];
|
|
if (keyboard->isAttached()) {
|
|
if (keyboard->startLvgl(display->getLvglDisplay())) {
|
|
lv_indev_t* keyboard_indev = keyboard->getLvglIndev();
|
|
hardware_keyboard_set_indev(keyboard_indev);
|
|
TT_LOG_I(TAG, "Started %s", keyboard->getName().c_str());
|
|
} else {
|
|
TT_LOG_E(TAG, "Start failed for %s", keyboard->getName().c_str());
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// Start encoders
|
|
TT_LOG_I(TAG, "Start encoders");
|
|
auto encoders = hal::findDevices<hal::encoder::EncoderDevice>(hal::Device::Type::Encoder);
|
|
for (auto encoder : encoders) {
|
|
if (displays.size() > 0) {
|
|
// TODO: Consider implementing support for multiple displays
|
|
auto display = displays[0];
|
|
if (encoder->startLvgl(display->getLvglDisplay())) {
|
|
TT_LOG_I(TAG, "Started %s", encoder->getName().c_str());
|
|
} else {
|
|
TT_LOG_E(TAG, "Start failed for %s", encoder->getName().c_str());
|
|
}
|
|
}
|
|
}
|
|
|
|
// Restart services
|
|
|
|
if (service::getState("Gui") == service::State::Stopped) {
|
|
service::startService("Gui");
|
|
} else {
|
|
TT_LOG_E(TAG, "Gui service is not in Stopped state");
|
|
}
|
|
|
|
if (service::getState("Statusbar") == service::State::Stopped) {
|
|
service::startService("Statusbar");
|
|
} else {
|
|
TT_LOG_E(TAG, "Statusbar service is not in Stopped state");
|
|
}
|
|
|
|
// Finalize
|
|
|
|
kernel::publishSystemEvent(kernel::SystemEvent::LvglStarted);
|
|
|
|
started = true;
|
|
}
|
|
|
|
void stop() {
|
|
TT_LOG_I(TAG, "Stopping LVGL");
|
|
|
|
if (!started) {
|
|
TT_LOG_W(TAG, "Can't stop LVGL: not started");
|
|
return;
|
|
}
|
|
|
|
auto lock = getSyncLock()->asScopedLock();
|
|
lock.lock();
|
|
|
|
// Stop services that highly depend on LVGL
|
|
|
|
service::stopService("Statusbar");
|
|
service::stopService("Gui");
|
|
|
|
// Stop keyboards
|
|
|
|
TT_LOG_I(TAG, "Stopping keyboards");
|
|
auto keyboards = hal::findDevices<hal::keyboard::KeyboardDevice>(hal::Device::Type::Keyboard);
|
|
for (auto keyboard : keyboards) {
|
|
if (keyboard->getLvglIndev() != nullptr) {
|
|
keyboard->stopLvgl();
|
|
}
|
|
}
|
|
|
|
// Stop touch
|
|
|
|
TT_LOG_I(TAG, "Stopping touch");
|
|
// The display generally stops their own touch devices, but we'll clean up anything that didn't
|
|
auto touch_devices = hal::findDevices<hal::touch::TouchDevice>(hal::Device::Type::Touch);
|
|
for (auto touch_device : touch_devices) {
|
|
if (touch_device->getLvglIndev() != nullptr) {
|
|
touch_device->stopLvgl();
|
|
}
|
|
}
|
|
|
|
// Stop encoders
|
|
|
|
TT_LOG_I(TAG, "Stopping encoders");
|
|
// The display generally stops their own touch devices, but we'll clean up anything that didn't
|
|
auto encoder_devices = hal::findDevices<hal::encoder::EncoderDevice>(hal::Device::Type::Encoder);
|
|
for (auto encoder_device : encoder_devices) {
|
|
if (encoder_device->getLvglIndev() != nullptr) {
|
|
encoder_device->stopLvgl();
|
|
}
|
|
}
|
|
// Stop displays (and their touch devices)
|
|
|
|
TT_LOG_I(TAG, "Stopping displays");
|
|
auto displays = hal::findDevices<hal::display::DisplayDevice>(hal::Device::Type::Display);
|
|
for (auto display : displays) {
|
|
if (display->supportsLvgl() && display->getLvglDisplay() != nullptr && !display->stopLvgl()) {
|
|
TT_LOG_E("HelloWorld", "Failed to detach display from LVGL");
|
|
}
|
|
}
|
|
|
|
started = false;
|
|
|
|
kernel::publishSystemEvent(kernel::SystemEvent::LvglStopped);
|
|
|
|
TT_LOG_I(TAG, "Stopped LVGL");
|
|
}
|
|
|
|
} // namespace
|