2022-02-16 21:42:29 +01:00
|
|
|
#include "displayapp/LittleVgl.h"
|
2022-08-15 13:46:42 +02:00
|
|
|
#include "displayapp/InfiniTimeTheme.h"
|
2022-02-16 21:42:29 +01:00
|
|
|
|
2022-06-11 22:24:50 +02:00
|
|
|
#include <FreeRTOS.h>
|
|
|
|
#include <task.h>
|
2022-06-13 20:46:03 +02:00
|
|
|
#include <timers.h>
|
2022-02-16 21:42:29 +01:00
|
|
|
#include "drivers/St7789.h"
|
2023-02-26 23:42:03 +01:00
|
|
|
#include "littlefs/lfs.h"
|
|
|
|
#include "components/fs/FS.h"
|
2022-02-16 21:42:29 +01:00
|
|
|
|
2022-06-13 20:46:03 +02:00
|
|
|
// lv-sim monitor display driver for monitor_flush() function
|
|
|
|
#include "lv_drivers/display/monitor.h"
|
|
|
|
|
2023-02-26 23:42:03 +01:00
|
|
|
#include <SDL2/SDL.h>
|
2022-06-13 20:46:03 +02:00
|
|
|
#include <array>
|
|
|
|
|
2022-02-16 21:42:29 +01:00
|
|
|
using namespace Pinetime::Components;
|
|
|
|
|
2023-02-26 23:11:00 +01:00
|
|
|
namespace {
|
|
|
|
void InitTheme() {
|
|
|
|
lv_theme_t* theme = lv_pinetime_theme_init();
|
|
|
|
lv_theme_set_act(theme);
|
|
|
|
}
|
2023-02-26 23:42:03 +01:00
|
|
|
|
|
|
|
lv_fs_res_t lvglOpen(lv_fs_drv_t* drv, void* file_p, const char* path, lv_fs_mode_t /*mode*/) {
|
|
|
|
lfs_file_t* file = static_cast<lfs_file_t*>(file_p);
|
|
|
|
Pinetime::Controllers::FS* filesys = static_cast<Pinetime::Controllers::FS*>(drv->user_data);
|
|
|
|
int res = filesys->FileOpen(file, path, LFS_O_RDONLY);
|
|
|
|
if (res == 0) {
|
|
|
|
if (file->type == 0) {
|
|
|
|
return LV_FS_RES_FS_ERR;
|
|
|
|
} else {
|
|
|
|
return LV_FS_RES_OK;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return LV_FS_RES_NOT_EX;
|
|
|
|
}
|
|
|
|
|
|
|
|
lv_fs_res_t lvglClose(lv_fs_drv_t* drv, void* file_p) {
|
|
|
|
Pinetime::Controllers::FS* filesys = static_cast<Pinetime::Controllers::FS*>(drv->user_data);
|
|
|
|
lfs_file_t* file = static_cast<lfs_file_t*>(file_p);
|
|
|
|
filesys->FileClose(file);
|
|
|
|
|
|
|
|
return LV_FS_RES_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
lv_fs_res_t lvglRead(lv_fs_drv_t* drv, void* file_p, void* buf, uint32_t btr, uint32_t* br) {
|
|
|
|
Pinetime::Controllers::FS* filesys = static_cast<Pinetime::Controllers::FS*>(drv->user_data);
|
|
|
|
lfs_file_t* file = static_cast<lfs_file_t*>(file_p);
|
|
|
|
filesys->FileRead(file, static_cast<uint8_t*>(buf), btr);
|
|
|
|
*br = btr;
|
|
|
|
return LV_FS_RES_OK;
|
|
|
|
}
|
|
|
|
|
|
|
|
lv_fs_res_t lvglSeek(lv_fs_drv_t* drv, void* file_p, uint32_t pos) {
|
|
|
|
Pinetime::Controllers::FS* filesys = static_cast<Pinetime::Controllers::FS*>(drv->user_data);
|
|
|
|
lfs_file_t* file = static_cast<lfs_file_t*>(file_p);
|
|
|
|
filesys->FileSeek(file, pos);
|
|
|
|
return LV_FS_RES_OK;
|
|
|
|
}
|
2023-02-26 23:11:00 +01:00
|
|
|
}
|
2022-06-11 22:24:50 +02:00
|
|
|
|
|
|
|
static void disp_flush(lv_disp_drv_t* disp_drv, const lv_area_t* area, lv_color_t* color_p) {
|
|
|
|
auto* lvgl = static_cast<LittleVgl*>(disp_drv->user_data);
|
|
|
|
lvgl->FlushDisplay(area, color_p);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void rounder(lv_disp_drv_t* disp_drv, lv_area_t* area) {
|
|
|
|
auto* lvgl = static_cast<LittleVgl*>(disp_drv->user_data);
|
|
|
|
if (lvgl->GetFullRefresh()) {
|
|
|
|
area->x1 = 0;
|
|
|
|
area->x2 = LV_HOR_RES - 1;
|
|
|
|
area->y1 = 0;
|
|
|
|
area->y2 = LV_VER_RES - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-16 21:42:29 +01:00
|
|
|
bool touchpad_read(lv_indev_drv_t* indev_drv, lv_indev_data_t* data) {
|
|
|
|
auto* lvgl = static_cast<LittleVgl*>(indev_drv->user_data);
|
|
|
|
return lvgl->GetTouchPadInfo(data);
|
|
|
|
}
|
|
|
|
|
2023-02-26 23:42:03 +01:00
|
|
|
LittleVgl::LittleVgl(Pinetime::Drivers::St7789& lcd, Pinetime::Controllers::FS& filesystem) : lcd {lcd}, filesystem {filesystem} {
|
2022-02-16 21:42:29 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void LittleVgl::Init() {
|
|
|
|
// lv_init();
|
2022-06-13 20:46:03 +02:00
|
|
|
InitTheme();
|
|
|
|
InitDisplay();
|
2022-02-16 21:42:29 +01:00
|
|
|
InitTouchpad();
|
2023-02-26 23:42:03 +01:00
|
|
|
InitFileSystem();
|
2022-02-16 21:42:29 +01:00
|
|
|
}
|
|
|
|
|
2022-06-11 22:24:50 +02:00
|
|
|
void LittleVgl::InitDisplay() {
|
|
|
|
lv_disp_buf_init(&disp_buf_2, buf2_1, buf2_2, LV_HOR_RES_MAX * 4); /*Initialize the display buffer*/
|
|
|
|
lv_disp_drv_init(&disp_drv); /*Basic initialization*/
|
|
|
|
|
|
|
|
/*Set up the functions to access to your display*/
|
|
|
|
|
|
|
|
/*Set the resolution of the display*/
|
|
|
|
disp_drv.hor_res = 240;
|
|
|
|
disp_drv.ver_res = 240;
|
|
|
|
|
|
|
|
/*Used to copy the buffer's content to the display*/
|
|
|
|
disp_drv.flush_cb = disp_flush;
|
|
|
|
/*Set a display buffer*/
|
|
|
|
disp_drv.buffer = &disp_buf_2;
|
|
|
|
disp_drv.user_data = this;
|
|
|
|
disp_drv.rounder_cb = rounder;
|
|
|
|
|
|
|
|
/*Finally register the driver*/
|
|
|
|
lv_disp_drv_register(&disp_drv);
|
|
|
|
}
|
2022-02-16 21:42:29 +01:00
|
|
|
|
|
|
|
void LittleVgl::InitTouchpad() {
|
|
|
|
lv_indev_drv_t indev_drv;
|
|
|
|
|
|
|
|
lv_indev_drv_init(&indev_drv);
|
|
|
|
indev_drv.type = LV_INDEV_TYPE_POINTER;
|
|
|
|
indev_drv.read_cb = touchpad_read;
|
|
|
|
indev_drv.user_data = this;
|
|
|
|
lv_indev_drv_register(&indev_drv);
|
|
|
|
}
|
|
|
|
|
2023-02-26 23:42:03 +01:00
|
|
|
void LittleVgl::InitFileSystem() {
|
|
|
|
lv_fs_drv_t fs_drv;
|
|
|
|
lv_fs_drv_init(&fs_drv);
|
|
|
|
|
|
|
|
fs_drv.file_size = sizeof(lfs_file_t);
|
|
|
|
fs_drv.letter = 'F';
|
|
|
|
fs_drv.open_cb = lvglOpen;
|
|
|
|
fs_drv.close_cb = lvglClose;
|
|
|
|
fs_drv.read_cb = lvglRead;
|
|
|
|
fs_drv.seek_cb = lvglSeek;
|
|
|
|
|
|
|
|
fs_drv.user_data = &filesystem;
|
|
|
|
|
|
|
|
lv_fs_drv_register(&fs_drv);
|
|
|
|
}
|
|
|
|
|
2022-02-16 21:42:29 +01:00
|
|
|
void LittleVgl::SetFullRefresh(FullRefreshDirections direction) {
|
|
|
|
if (scrollDirection == FullRefreshDirections::None) {
|
|
|
|
scrollDirection = direction;
|
|
|
|
if (scrollDirection == FullRefreshDirections::Down) {
|
2022-06-11 22:24:50 +02:00
|
|
|
lv_disp_set_direction(lv_disp_get_default(), 1);
|
2022-02-16 21:42:29 +01:00
|
|
|
} else if (scrollDirection == FullRefreshDirections::Right) {
|
2022-06-11 22:24:50 +02:00
|
|
|
lv_disp_set_direction(lv_disp_get_default(), 2);
|
2022-02-16 21:42:29 +01:00
|
|
|
} else if (scrollDirection == FullRefreshDirections::Left) {
|
2022-06-11 22:24:50 +02:00
|
|
|
lv_disp_set_direction(lv_disp_get_default(), 3);
|
2022-02-16 21:42:29 +01:00
|
|
|
} else if (scrollDirection == FullRefreshDirections::RightAnim) {
|
2022-06-11 22:24:50 +02:00
|
|
|
lv_disp_set_direction(lv_disp_get_default(), 5);
|
2022-02-16 21:42:29 +01:00
|
|
|
} else if (scrollDirection == FullRefreshDirections::LeftAnim) {
|
2022-06-11 22:24:50 +02:00
|
|
|
lv_disp_set_direction(lv_disp_get_default(), 4);
|
2022-02-16 21:42:29 +01:00
|
|
|
}
|
|
|
|
}
|
2022-06-11 22:24:50 +02:00
|
|
|
fullRefresh = true;
|
2022-02-16 21:42:29 +01:00
|
|
|
}
|
2022-06-11 22:24:50 +02:00
|
|
|
|
2022-06-13 20:46:03 +02:00
|
|
|
// glue the lvgl code to the lv-sim monitor driver
|
|
|
|
void DrawBuffer(lv_disp_drv_t *disp_drv, uint16_t x, uint16_t y, uint16_t width, uint16_t height, uint8_t* data, size_t size) {
|
|
|
|
lv_area_t area;
|
|
|
|
area.x1 = x;
|
|
|
|
area.x2 = x+width-1;
|
|
|
|
area.y1 = y;
|
|
|
|
area.y2 = y+height-1;
|
|
|
|
lv_color_t* color_p = reinterpret_cast<lv_color_t*>(data);
|
|
|
|
monitor_flush(disp_drv, &area, color_p);
|
|
|
|
}
|
|
|
|
|
|
|
|
// copied from lv_drivers/display/monitor.c to get the SDL_Window for the InfiniTime screen
|
|
|
|
extern "C"
|
|
|
|
{
|
|
|
|
typedef struct {
|
|
|
|
SDL_Window * window;
|
|
|
|
SDL_Renderer * renderer;
|
|
|
|
SDL_Texture * texture;
|
|
|
|
volatile bool sdl_refr_qry;
|
|
|
|
#if MONITOR_DOUBLE_BUFFERED
|
|
|
|
uint32_t * tft_fb_act;
|
|
|
|
#else
|
|
|
|
uint32_t tft_fb[LV_HOR_RES_MAX * LV_VER_RES_MAX];
|
|
|
|
#endif
|
|
|
|
}monitor_t;
|
|
|
|
extern monitor_t monitor;
|
|
|
|
}
|
|
|
|
|
|
|
|
// positive height moves screen down (draw y=0 to y=height)
|
|
|
|
// negative height moves screen up (draw y=height to y=0)
|
|
|
|
void MoveScreen(lv_disp_drv_t *disp_drv, int16_t height) {
|
|
|
|
if (height == 0)
|
|
|
|
return; // nothing to do
|
|
|
|
|
|
|
|
const int sdl_width = 240;
|
|
|
|
const int sdl_height = 240;
|
|
|
|
auto renderer = monitor.renderer;
|
|
|
|
|
2023-05-03 22:28:17 +02:00
|
|
|
constexpr size_t zoom = MONITOR_ZOOM;
|
2022-06-13 20:46:03 +02:00
|
|
|
const Uint32 format = SDL_PIXELFORMAT_RGBA8888;
|
2023-05-03 22:28:17 +02:00
|
|
|
SDL_Surface *surface = SDL_CreateRGBSurfaceWithFormat(0, sdl_width*zoom, sdl_height*zoom, 32, format);
|
2022-06-13 20:46:03 +02:00
|
|
|
SDL_RenderReadPixels(renderer, NULL, format, surface->pixels, surface->pitch);
|
|
|
|
uint8_t *pixels = (uint8_t*) surface->pixels;
|
|
|
|
|
|
|
|
std::array<lv_color16_t, 240*240> color_p;
|
|
|
|
for (int hi = 0; hi < sdl_height; hi++) {
|
|
|
|
for (int wi = 0; wi < sdl_width; wi++) {
|
2023-05-03 22:28:17 +02:00
|
|
|
auto red = pixels[hi*surface->pitch*zoom + wi*4*zoom + 3]; // red
|
|
|
|
auto green = pixels[hi*surface->pitch*zoom + wi*4*zoom + 2]; // greeen
|
|
|
|
auto blue = pixels[hi*surface->pitch*zoom + wi*4*zoom + 1]; // blue
|
2022-06-13 20:46:03 +02:00
|
|
|
color_p.at(hi * sdl_width + wi) = LV_COLOR_MAKE(red, green, blue);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
int16_t buffer_height = sdl_height - abs(height);
|
|
|
|
if (height >= 0) {
|
|
|
|
DrawBuffer(disp_drv, 0, height, sdl_width, sdl_height, (uint8_t*)color_p.data(), sdl_width*buffer_height *2);
|
|
|
|
} else {
|
|
|
|
DrawBuffer(disp_drv, 0, 0, sdl_width, sdl_height, (uint8_t*)(&color_p.at(sdl_width*abs(height))), sdl_width*buffer_height *2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-16 21:42:29 +01:00
|
|
|
void LittleVgl::FlushDisplay(const lv_area_t* area, lv_color_t* color_p) {
|
2022-06-13 20:46:03 +02:00
|
|
|
uint16_t y1, y2, width, height = 0;
|
|
|
|
|
|
|
|
//ulTaskNotifyTake(pdTRUE, 200);
|
|
|
|
// Notification is still needed (even if there is a mutex on SPI) because of the DataCommand pin
|
|
|
|
// which cannot be set/clear during a transfer.
|
|
|
|
|
|
|
|
//if ((scrollDirection == LittleVgl::FullRefreshDirections::Down) && (area->y2 == visibleNbLines - 1)) {
|
|
|
|
// writeOffset = ((writeOffset + totalNbLines) - visibleNbLines) % totalNbLines;
|
|
|
|
//} else if ((scrollDirection == FullRefreshDirections::Up) && (area->y1 == 0)) {
|
|
|
|
// writeOffset = (writeOffset + visibleNbLines) % totalNbLines;
|
|
|
|
//}
|
|
|
|
|
|
|
|
y1 = (area->y1 + writeOffset) % totalNbLines;
|
|
|
|
y2 = (area->y2 + writeOffset) % totalNbLines;
|
|
|
|
|
|
|
|
width = (area->x2 - area->x1) + 1;
|
|
|
|
height = (area->y2 - area->y1) + 1;
|
|
|
|
|
|
|
|
if (scrollDirection == LittleVgl::FullRefreshDirections::Down) {
|
|
|
|
|
|
|
|
if (area->y2 < visibleNbLines - 1) {
|
|
|
|
uint16_t toScroll = 0;
|
|
|
|
if (area->y1 == 0) {
|
|
|
|
toScroll = height * 2;
|
|
|
|
scrollDirection = FullRefreshDirections::None;
|
|
|
|
lv_disp_set_direction(lv_disp_get_default(), 0);
|
|
|
|
} else {
|
|
|
|
toScroll = height;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (scrollOffset >= toScroll)
|
|
|
|
scrollOffset -= toScroll;
|
|
|
|
else {
|
|
|
|
toScroll -= scrollOffset;
|
|
|
|
scrollOffset = (totalNbLines) -toScroll;
|
|
|
|
}
|
|
|
|
lcd.VerticalScrollStartAddress(scrollOffset);
|
|
|
|
}
|
|
|
|
// move the whole screen down and draw the new screen at the top of the display
|
|
|
|
MoveScreen(&disp_drv, static_cast<int16_t>(height));
|
|
|
|
y1 = 0;
|
|
|
|
y2 = height;
|
|
|
|
|
|
|
|
} else if (scrollDirection == FullRefreshDirections::Up) {
|
|
|
|
|
|
|
|
if (area->y1 > 0) {
|
|
|
|
if (area->y2 == visibleNbLines - 1) {
|
|
|
|
scrollOffset += (height * 2);
|
|
|
|
scrollDirection = FullRefreshDirections::None;
|
|
|
|
lv_disp_set_direction(lv_disp_get_default(), 0);
|
|
|
|
} else {
|
|
|
|
scrollOffset += height;
|
|
|
|
}
|
|
|
|
scrollOffset = scrollOffset % totalNbLines;
|
|
|
|
lcd.VerticalScrollStartAddress(scrollOffset);
|
|
|
|
}
|
|
|
|
// move the whole screen up and draw the new screen at the bottom the display
|
|
|
|
MoveScreen(&disp_drv, -static_cast<int16_t>(height));
|
|
|
|
y1 = LV_VER_RES - height;
|
|
|
|
y2 = LV_VER_RES;
|
|
|
|
} else if (scrollDirection == FullRefreshDirections::Left or scrollDirection == FullRefreshDirections::LeftAnim) {
|
|
|
|
if (area->x2 == visibleNbLines - 1) {
|
|
|
|
scrollDirection = FullRefreshDirections::None;
|
|
|
|
lv_disp_set_direction(lv_disp_get_default(), 0);
|
|
|
|
}
|
|
|
|
} else if (scrollDirection == FullRefreshDirections::Right or scrollDirection == FullRefreshDirections::RightAnim) {
|
|
|
|
if (area->x1 == 0) {
|
|
|
|
scrollDirection = FullRefreshDirections::None;
|
|
|
|
lv_disp_set_direction(lv_disp_get_default(), 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (y2 < y1) {
|
|
|
|
height = totalNbLines - y1;
|
|
|
|
|
|
|
|
if (height > 0) {
|
|
|
|
//lcd.DrawBuffer(area->x1, y1, width, height, reinterpret_cast<const uint8_t*>(color_p), width * height * 2);
|
|
|
|
DrawBuffer(&disp_drv, area->x1, y1, width, height, reinterpret_cast<uint8_t*>(color_p), width * height * 2);
|
|
|
|
//ulTaskNotifyTake(pdTRUE, 100);
|
|
|
|
}
|
|
|
|
|
|
|
|
uint16_t pixOffset = width * height;
|
|
|
|
height = y2 + 1;
|
|
|
|
//lcd.DrawBuffer(area->x1, 0, width, height, reinterpret_cast<const uint8_t*>(color_p + pixOffset), width * height * 2);
|
|
|
|
DrawBuffer(&disp_drv, area->x1, 0, width, height, reinterpret_cast<uint8_t*>(color_p + pixOffset), width * height * 2);
|
|
|
|
|
|
|
|
} else {
|
|
|
|
//lcd.DrawBuffer(area->x1, y1, width, height, reinterpret_cast<const uint8_t*>(color_p), width * height * 2);
|
|
|
|
DrawBuffer(&disp_drv, area->x1, y1, width, height, reinterpret_cast<uint8_t*>(color_p), width * height * 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
// IMPORTANT!!!
|
|
|
|
// Inform the graphics library that you are ready with the flushing
|
|
|
|
//lv_disp_flush_ready(&disp_drv);
|
2022-02-16 21:42:29 +01:00
|
|
|
|
2022-06-13 20:46:03 +02:00
|
|
|
// call flush with flushing_last set
|
|
|
|
// workaround because lv_disp_flush_ready() doesn't seem to trigger monitor_flush
|
2022-02-16 21:42:29 +01:00
|
|
|
lv_disp_t *disp = lv_disp_get_default();
|
|
|
|
lv_disp_get_buf(disp)->flushing_last = true;
|
2022-06-13 20:46:03 +02:00
|
|
|
lv_area_t area_zero {0,0,0,0};
|
|
|
|
monitor_flush(&disp_drv, &area_zero, color_p);
|
|
|
|
// delay drawing to mimic PineTime display rendering speed
|
|
|
|
vTaskDelay(pdMS_TO_TICKS(3));
|
2022-02-16 21:42:29 +01:00
|
|
|
}
|
|
|
|
|
2023-02-24 17:11:26 +01:00
|
|
|
void LittleVgl::SetNewTouchPoint(int16_t x, int16_t y, bool contact) {
|
|
|
|
if (contact) {
|
|
|
|
if (!isCancelled) {
|
|
|
|
touchPoint = {x, y};
|
|
|
|
tapped = true;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (isCancelled) {
|
|
|
|
touchPoint = {-1, -1};
|
|
|
|
tapped = false;
|
|
|
|
isCancelled = false;
|
|
|
|
} else {
|
|
|
|
touchPoint = {x, y};
|
|
|
|
tapped = false;
|
|
|
|
}
|
|
|
|
}
|
2022-02-16 21:42:29 +01:00
|
|
|
}
|
|
|
|
|
2023-02-24 17:11:26 +01:00
|
|
|
void LittleVgl::CancelTap() {
|
|
|
|
if (tapped) {
|
|
|
|
isCancelled = true;
|
|
|
|
touchPoint = {-1, -1};
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-02-16 21:42:29 +01:00
|
|
|
bool LittleVgl::GetTouchPadInfo(lv_indev_data_t* ptr) {
|
2023-02-24 17:11:26 +01:00
|
|
|
ptr->point.x = touchPoint.x;
|
|
|
|
ptr->point.y = touchPoint.y;
|
2022-02-16 21:42:29 +01:00
|
|
|
if (tapped) {
|
|
|
|
ptr->state = LV_INDEV_STATE_PR;
|
|
|
|
} else {
|
|
|
|
ptr->state = LV_INDEV_STATE_REL;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|