1
0
Fork 0

Integrate new notification UI with notifications coming from BLE

This commit is contained in:
JF 2020-10-20 20:57:39 +02:00
parent 55427d83b8
commit ef5670c7e0
14 changed files with 261 additions and 363 deletions

View file

@ -289,6 +289,8 @@ set(LVGL_SRC
libs/lvgl/src/lv_objx/lv_slider.c libs/lvgl/src/lv_objx/lv_slider.c
libs/lvgl/src/lv_objx/lv_ddlist.c libs/lvgl/src/lv_objx/lv_ddlist.c
libs/lvgl/src/lv_objx/lv_ddlist.h libs/lvgl/src/lv_objx/lv_ddlist.h
libs/lvgl/src/lv_objx/lv_line.c
libs/lvgl/src/lv_objx/lv_line.h
) )
list(APPEND IMAGE_FILES list(APPEND IMAGE_FILES
@ -334,6 +336,7 @@ list(APPEND SOURCE_FILES
displayapp/screens/Modal.cpp displayapp/screens/Modal.cpp
displayapp/screens/BatteryIcon.cpp displayapp/screens/BatteryIcon.cpp
displayapp/screens/BleIcon.cpp displayapp/screens/BleIcon.cpp
displayapp/screens/NotificationIcon.cpp
displayapp/screens/Brightness.cpp displayapp/screens/Brightness.cpp
displayapp/screens/SystemInfo.cpp displayapp/screens/SystemInfo.cpp
displayapp/screens/Label.cpp displayapp/screens/Label.cpp
@ -342,7 +345,6 @@ list(APPEND SOURCE_FILES
displayapp/screens/FirmwareValidation.cpp displayapp/screens/FirmwareValidation.cpp
displayapp/screens/ApplicationList.cpp displayapp/screens/ApplicationList.cpp
displayapp/screens/Notifications.cpp displayapp/screens/Notifications.cpp
displayapp/screens/Notifications_swscroll.cpp
main.cpp main.cpp
drivers/St7789.cpp drivers/St7789.cpp
drivers/SpiNorFlash.cpp drivers/SpiNorFlash.cpp
@ -412,7 +414,8 @@ set(INCLUDE_FILES
displayapp/screens/DropDownDemo.h displayapp/screens/DropDownDemo.h
displayapp/screens/Modal.h displayapp/screens/Modal.h
displayapp/screens/BatteryIcon.h displayapp/screens/BatteryIcon.h
displayapp/screens/BleIcon.cpp displayapp/screens/BleIcon.h
displayapp/screens/NotificationIcon.h
displayapp/screens/Brightness.h displayapp/screens/Brightness.h
displayapp/screens/SystemInfo.h displayapp/screens/SystemInfo.h
displayapp/screens/ScreenList.h displayapp/screens/ScreenList.h
@ -422,7 +425,6 @@ set(INCLUDE_FILES
displayapp/screens/ApplicationList.h displayapp/screens/ApplicationList.h
displayapp/Apps.h displayapp/Apps.h
displayapp/screens/Notifications.h displayapp/screens/Notifications.h
displayapp/screens/Notifications_swscroll.h.h
drivers/St7789.h drivers/St7789.h
drivers/SpiNorFlash.h drivers/SpiNorFlash.h
drivers/SpiMaster.h drivers/SpiMaster.h

View file

@ -1,4 +1,5 @@
#include <cstring> #include <cstring>
#include <algorithm>
#include "NotificationManager.h" #include "NotificationManager.h"
using namespace Pinetime::Controllers; using namespace Pinetime::Controllers;
@ -11,20 +12,69 @@ void NotificationManager::Push(Pinetime::Controllers::NotificationManager::Categ
std::memcpy(notif.message.data(), message, checkedSize); std::memcpy(notif.message.data(), message, checkedSize);
notif.message[checkedSize] = '\0'; notif.message[checkedSize] = '\0';
notif.category = category; notif.category = category;
notif.id = GetNextId();
notif.valid = true;
writeIndex = (writeIndex + 1 < TotalNbNotifications) ? writeIndex + 1 : 0; writeIndex = (writeIndex + 1 < TotalNbNotifications) ? writeIndex + 1 : 0;
if(!empty && writeIndex == readIndex) if(!empty)
readIndex = writeIndex + 1; readIndex = (readIndex + 1 < TotalNbNotifications) ? readIndex + 1 : 0;
else empty = false;
newNotification = true;
} }
NotificationManager::Notification Pinetime::Controllers::NotificationManager::Pop() { NotificationManager::Notification NotificationManager::GetLastNotification() {
// TODO handle edge cases on read/write index
NotificationManager::Notification notification = notifications[readIndex]; NotificationManager::Notification notification = notifications[readIndex];
if(readIndex != writeIndex) {
readIndex = (readIndex + 1 < TotalNbNotifications) ? readIndex + 1 : 0;
}
// TODO Check move optimization on return
return notification; return notification;
} }
NotificationManager::Notification::Id NotificationManager::GetNextId() {
return nextId++;
}
NotificationManager::Notification NotificationManager::GetNext(NotificationManager::Notification::Id id) {
auto currentIterator = std::find_if(notifications.begin(), notifications.end(), [id](const Notification& n){return n.valid && n.id == id;});
if(currentIterator == notifications.end() || currentIterator->id != id) return Notification{};
auto& lastNotification = notifications[readIndex];
NotificationManager::Notification result;
if(currentIterator == (notifications.end()-1))
result = *(notifications.begin());
else
result = *(currentIterator+1);
if(result.id <= id) return {};
result.index = (lastNotification.id - result.id)+1;
return result;
}
NotificationManager::Notification NotificationManager::GetPrevious(NotificationManager::Notification::Id id) {
auto currentIterator = std::find_if(notifications.begin(), notifications.end(), [id](const Notification& n){return n.valid && n.id == id;});
if(currentIterator == notifications.end() || currentIterator->id != id) return Notification{};
auto& lastNotification = notifications[readIndex];
NotificationManager::Notification result;
if(currentIterator == notifications.begin())
result = *(notifications.end()-1);
else
result = *(currentIterator-1);
if(result.id >= id) return {};
result.index = (lastNotification.id - result.id)+1;
return result;
}
bool NotificationManager::AreNewNotificationsAvailable() {
return newNotification;
}
bool NotificationManager::ClearNewNotificationFlag() {
return newNotification.exchange(false);
}

View file

@ -1,6 +1,7 @@
#pragma once #pragma once
#include <array> #include <array>
#include <atomic>
namespace Pinetime { namespace Pinetime {
namespace Controllers { namespace Controllers {
@ -10,20 +11,32 @@ namespace Pinetime {
static constexpr uint8_t MessageSize{18}; static constexpr uint8_t MessageSize{18};
struct Notification { struct Notification {
using Id = uint8_t;
Id id;
bool valid = false;
uint8_t index;
uint8_t number = TotalNbNotifications;
std::array<char, MessageSize+1> message; std::array<char, MessageSize+1> message;
Categories category = Categories::Unknown; Categories category = Categories::Unknown;
}; };
Notification::Id nextId {0};
void Push(Categories category, const char* message, uint8_t messageSize); void Push(Categories category, const char* message, uint8_t messageSize);
Notification Pop(); Notification GetLastNotification();
Notification GetNext(Notification::Id id);
Notification GetPrevious(Notification::Id id);
bool ClearNewNotificationFlag();
bool AreNewNotificationsAvailable();
private: private:
Notification::Id GetNextId();
static constexpr uint8_t TotalNbNotifications = 5; static constexpr uint8_t TotalNbNotifications = 5;
std::array<Notification, TotalNbNotifications> notifications; std::array<Notification, TotalNbNotifications> notifications;
uint8_t readIndex = 0; uint8_t readIndex = 0;
uint8_t writeIndex = 0; uint8_t writeIndex = 0;
bool empty = true; bool empty = true;
std::atomic<bool> newNotification{false};
}; };
} }
} }

View file

@ -2,6 +2,6 @@
namespace Pinetime { namespace Pinetime {
namespace Applications { namespace Applications {
enum class Apps {None, Launcher, Clock, SysInfo, Meter, Gauge, Brightness, Music, FirmwareValidation, Paint, Notifications, Notifications2}; enum class Apps {None, Launcher, Clock, SysInfo, Meter, Gauge, Brightness, Music, FirmwareValidation, Paint, Notifications};
} }
} }

View file

@ -9,7 +9,6 @@
#include "components/datetime/DateTimeController.h" #include "components/datetime/DateTimeController.h"
#include <drivers/Cst816s.h> #include <drivers/Cst816s.h>
#include "displayapp/screens/Notifications.h" #include "displayapp/screens/Notifications.h"
#include "displayapp/screens/Notifications_swscroll.h"
#include "displayapp/screens/Tile.h" #include "displayapp/screens/Tile.h"
#include "displayapp/screens/Meter.h" #include "displayapp/screens/Meter.h"
#include "displayapp/screens/Gauge.h" #include "displayapp/screens/Gauge.h"
@ -37,7 +36,7 @@ DisplayApp::DisplayApp(Drivers::St7789 &lcd, Components::LittleVgl &lvgl, Driver
dateTimeController{dateTimeController}, dateTimeController{dateTimeController},
watchdog{watchdog}, watchdog{watchdog},
touchPanel{touchPanel}, touchPanel{touchPanel},
currentScreen{new Screens::Clock(this, dateTimeController, batteryController, bleController) }, currentScreen{new Screens::Clock(this, dateTimeController, batteryController, bleController, notificationManager) },
systemTask{systemTask}, systemTask{systemTask},
notificationManager{notificationManager} { notificationManager{notificationManager} {
msgQueue = xQueueCreate(queueSize, itemSize); msgQueue = xQueueCreate(queueSize, itemSize);
@ -116,8 +115,12 @@ void DisplayApp::Refresh() {
// clockScreen.SetBatteryPercentRemaining(batteryController.PercentRemaining()); // clockScreen.SetBatteryPercentRemaining(batteryController.PercentRemaining());
break; break;
case Messages::NewNotification: { case Messages::NewNotification: {
auto notification = notificationManager.Pop(); if(onClockApp) {
modal->Show(notification.message.data()); currentScreen.reset(nullptr);
lvgl.SetFullRefresh(Components::LittleVgl::FullRefreshDirections::Up);
onClockApp = false;
currentScreen.reset(new Screens::Notifications(this, notificationManager, Screens::Notifications::Modes::Preview));
}
} }
break; break;
case Messages::TouchEvent: { case Messages::TouchEvent: {
@ -193,7 +196,7 @@ void DisplayApp::RunningState() {
case Apps::None: case Apps::None:
case Apps::Launcher: currentScreen.reset(new Screens::ApplicationList(this)); break; case Apps::Launcher: currentScreen.reset(new Screens::ApplicationList(this)); break;
case Apps::Clock: case Apps::Clock:
currentScreen.reset(new Screens::Clock(this, dateTimeController, batteryController, bleController)); currentScreen.reset(new Screens::Clock(this, dateTimeController, batteryController, bleController, notificationManager));
onClockApp = true; onClockApp = true;
break; break;
// case Apps::Test: currentScreen.reset(new Screens::Message(this)); break; // case Apps::Test: currentScreen.reset(new Screens::Message(this)); break;
@ -204,8 +207,7 @@ void DisplayApp::RunningState() {
case Apps::Brightness : currentScreen.reset(new Screens::Brightness(this, brightnessController)); break; case Apps::Brightness : currentScreen.reset(new Screens::Brightness(this, brightnessController)); break;
case Apps::Music : currentScreen.reset(new Screens::Music(this, systemTask.nimble().music())); break; case Apps::Music : currentScreen.reset(new Screens::Music(this, systemTask.nimble().music())); break;
case Apps::FirmwareValidation: currentScreen.reset(new Screens::FirmwareValidation(this, validator)); break; case Apps::FirmwareValidation: currentScreen.reset(new Screens::FirmwareValidation(this, validator)); break;
case Apps::Notifications: currentScreen.reset(new Screens::Notifications(this)); break; case Apps::Notifications: currentScreen.reset(new Screens::Notifications(this, notificationManager, Screens::Notifications::Modes::Normal)); break;
case Apps::Notifications2: currentScreen.reset(new Screens::Notifications2(this)); break;
} }
nextApp = Apps::None; nextApp = Apps::None;
} }

View file

@ -58,8 +58,8 @@ std::unique_ptr<Screen> ApplicationList::CreateScreen2() {
{{Symbols::tachometer, Apps::Gauge}, {{Symbols::tachometer, Apps::Gauge},
{Symbols::asterisk, Apps::Meter}, {Symbols::asterisk, Apps::Meter},
{Symbols::paintbrush, Apps::Paint}, {Symbols::paintbrush, Apps::Paint},
{Symbols::shoe, Apps::Notifications}, {Symbols::info, Apps::Notifications},
{Symbols::shoe, Apps::Notifications2}, {Symbols::none, Apps::None},
{Symbols::none, Apps::None} {Symbols::none, Apps::None}
} }
}; };

View file

@ -8,6 +8,9 @@
#include "BatteryIcon.h" #include "BatteryIcon.h"
#include "BleIcon.h" #include "BleIcon.h"
#include "Symbols.h" #include "Symbols.h"
#include "components/ble/NotificationManager.h"
#include "NotificationIcon.h"
using namespace Pinetime::Applications::Screens; using namespace Pinetime::Applications::Screens;
extern lv_font_t jetbrains_mono_extrabold_compressed; extern lv_font_t jetbrains_mono_extrabold_compressed;
extern lv_font_t jetbrains_mono_bold_20; extern lv_font_t jetbrains_mono_bold_20;
@ -21,8 +24,10 @@ static void event_handler(lv_obj_t * obj, lv_event_t event) {
Clock::Clock(DisplayApp* app, Clock::Clock(DisplayApp* app,
Controllers::DateTime& dateTimeController, Controllers::DateTime& dateTimeController,
Controllers::Battery& batteryController, Controllers::Battery& batteryController,
Controllers::Ble& bleController) : Screen(app), currentDateTime{{}}, Controllers::Ble& bleController,
dateTimeController{dateTimeController}, batteryController{batteryController}, bleController{bleController} { Controllers::NotificationManager& notificatioManager) : Screen(app), currentDateTime{{}},
dateTimeController{dateTimeController}, batteryController{batteryController},
bleController{bleController}, notificatioManager{notificatioManager} {
displayedChar[0] = 0; displayedChar[0] = 0;
displayedChar[1] = 0; displayedChar[1] = 0;
displayedChar[2] = 0; displayedChar[2] = 0;
@ -41,9 +46,11 @@ Clock::Clock(DisplayApp* app,
lv_label_set_text(bleIcon, Symbols::bluetooth); lv_label_set_text(bleIcon, Symbols::bluetooth);
lv_obj_align(bleIcon, batteryPlug, LV_ALIGN_OUT_LEFT_MID, -5, 0); lv_obj_align(bleIcon, batteryPlug, LV_ALIGN_OUT_LEFT_MID, -5, 0);
notificationIcon = lv_label_create(lv_scr_act(), NULL);
lv_label_set_text(notificationIcon, NotificationIcon::GetIcon(false));
lv_obj_align(notificationIcon, nullptr, LV_ALIGN_IN_TOP_LEFT, 10, 0);
label_date = lv_label_create(lv_scr_act(), NULL); label_date = lv_label_create(lv_scr_act(), NULL);
lv_obj_align(label_date, lv_scr_act(), LV_ALIGN_IN_LEFT_MID, 0, 60); lv_obj_align(label_date, lv_scr_act(), LV_ALIGN_IN_LEFT_MID, 0, 60);
label_time = lv_label_create(lv_scr_act(), NULL); label_time = lv_label_create(lv_scr_act(), NULL);
@ -106,6 +113,14 @@ bool Clock::Refresh() {
lv_obj_align(batteryPlug, batteryIcon, LV_ALIGN_OUT_LEFT_MID, -5, 0); lv_obj_align(batteryPlug, batteryIcon, LV_ALIGN_OUT_LEFT_MID, -5, 0);
lv_obj_align(bleIcon, batteryPlug, LV_ALIGN_OUT_LEFT_MID, -5, 0); lv_obj_align(bleIcon, batteryPlug, LV_ALIGN_OUT_LEFT_MID, -5, 0);
notificationState = notificatioManager.AreNewNotificationsAvailable();
if(notificationState.IsUpdated()) {
if(notificationState.Get() == true)
lv_label_set_text(notificationIcon, NotificationIcon::GetIcon(true));
else
lv_label_set_text(notificationIcon, NotificationIcon::GetIcon(false));
}
currentDateTime = dateTimeController.CurrentDateTime(); currentDateTime = dateTimeController.CurrentDateTime();
if(currentDateTime.IsUpdated()) { if(currentDateTime.IsUpdated()) {

View file

@ -7,6 +7,7 @@
#include <bits/unique_ptr.h> #include <bits/unique_ptr.h>
#include <libs/lvgl/src/lv_core/lv_style.h> #include <libs/lvgl/src/lv_core/lv_style.h>
#include <libs/lvgl/src/lv_core/lv_obj.h> #include <libs/lvgl/src/lv_core/lv_obj.h>
#include "components/ble/NotificationManager.h"
#include "components/battery/BatteryController.h" #include "components/battery/BatteryController.h"
#include "components/ble/BleController.h" #include "components/ble/BleController.h"
@ -38,7 +39,8 @@ namespace Pinetime {
Clock(DisplayApp* app, Clock(DisplayApp* app,
Controllers::DateTime& dateTimeController, Controllers::DateTime& dateTimeController,
Controllers::Battery& batteryController, Controllers::Battery& batteryController,
Controllers::Ble& bleController); Controllers::Ble& bleController,
Controllers::NotificationManager& notificatioManager);
~Clock() override; ~Clock() override;
bool Refresh() override; bool Refresh() override;
@ -63,23 +65,25 @@ namespace Pinetime {
DirtyValue<std::chrono::time_point<std::chrono::system_clock, std::chrono::nanoseconds>> currentDateTime; DirtyValue<std::chrono::time_point<std::chrono::system_clock, std::chrono::nanoseconds>> currentDateTime;
DirtyValue<uint32_t> stepCount {0}; DirtyValue<uint32_t> stepCount {0};
DirtyValue<uint8_t> heartbeat {0}; DirtyValue<uint8_t> heartbeat {0};
DirtyValue<bool> notificationState {false};
lv_obj_t* label_time; lv_obj_t* label_time;
lv_obj_t* label_date; lv_obj_t* label_date;
lv_obj_t* backgroundLabel; lv_obj_t* backgroundLabel;
lv_obj_t * batteryIcon; lv_obj_t* batteryIcon;
lv_obj_t * bleIcon; lv_obj_t* bleIcon;
lv_obj_t* batteryPlug; lv_obj_t* batteryPlug;
lv_obj_t* heartbeatIcon; lv_obj_t* heartbeatIcon;
lv_obj_t* heartbeatValue; lv_obj_t* heartbeatValue;
lv_obj_t* heartbeatBpm; lv_obj_t* heartbeatBpm;
lv_obj_t* stepIcon; lv_obj_t* stepIcon;
lv_obj_t* stepValue; lv_obj_t* stepValue;
lv_obj_t* notificationIcon;
Controllers::DateTime& dateTimeController; Controllers::DateTime& dateTimeController;
Controllers::Battery& batteryController; Controllers::Battery& batteryController;
Controllers::Ble& bleController; Controllers::Ble& bleController;
Controllers::NotificationManager& notificatioManager;
bool running = true; bool running = true;

View file

@ -0,0 +1,8 @@
#include "NotificationIcon.h"
#include "Symbols.h"
using namespace Pinetime::Applications::Screens;
const char* NotificationIcon::GetIcon(bool newNotificationAvailable) {
if(newNotificationAvailable) return Symbols::info;
else return "";
}

View file

@ -0,0 +1,12 @@
#pragma once
namespace Pinetime {
namespace Applications {
namespace Screens {
class NotificationIcon {
public:
static const char* GetIcon(bool newNotificationAvailable);
};
}
}
}

View file

@ -2,15 +2,35 @@
#include <displayapp/DisplayApp.h> #include <displayapp/DisplayApp.h>
#include <functional> #include <functional>
#include "Notifications.h" #include "Notifications.h"
using namespace Pinetime::Applications::Screens; using namespace Pinetime::Applications::Screens;
Notifications::Notifications(DisplayApp* app) : Screen(app), screens{app, { Notifications::Notifications(DisplayApp *app, Pinetime::Controllers::NotificationManager &notificationManager, Modes mode) :
[this]() -> std::unique_ptr<Screen> { return CreateScreen1(); }, Screen(app), notificationManager{notificationManager}, mode{mode} {
[this]() -> std::unique_ptr<Screen> { return CreateScreen2(); }, notificationManager.ClearNewNotificationFlag();
[this]() -> std::unique_ptr<Screen> { return CreateScreen3(); } auto notification = notificationManager.GetLastNotification();
} if(notification.valid) {
} { currentId = notification.id;
currentItem.reset(new NotificationItem("Notification", notification.message.data(), notification.index, notification.number, mode));
validDisplay = true;
} else {
currentItem.reset(new NotificationItem("Notification", "No notification to display", 0, 0, mode));
}
if(mode == Modes::Preview) {
static lv_style_t style_line;
lv_style_copy(&style_line, &lv_style_plain);
style_line.line.color = LV_COLOR_WHITE;
style_line.line.width = 3;
style_line.line.rounded = 0;
timeoutLine = lv_line_create(lv_scr_act(), nullptr);
lv_line_set_style(timeoutLine, LV_LINE_STYLE_MAIN, &style_line);
lv_line_set_points(timeoutLine, timeoutLinePoints, 2);
timeoutTickCountStart = xTaskGetTickCount();
timeoutTickCountEnd = timeoutTickCountStart + (5*1024);
}
} }
Notifications::~Notifications() { Notifications::~Notifications() {
@ -18,12 +38,61 @@ Notifications::~Notifications() {
} }
bool Notifications::Refresh() { bool Notifications::Refresh() {
screens.Refresh(); if (mode == Modes::Preview) {
auto tick = xTaskGetTickCount();
int32_t pos = 240 - ((tick - timeoutTickCountStart) / ((timeoutTickCountEnd - timeoutTickCountStart) / 240));
if (pos < 0)
running = false;
timeoutLinePoints[1].x = pos;
lv_line_set_points(timeoutLine, timeoutLinePoints, 2);
if (!running) {
// Start clock app when exiting this one
app->StartApp(Apps::Clock);
}
}
return running; return running;
} }
bool Notifications::OnTouchEvent(Pinetime::Applications::TouchEvents event) { bool Notifications::OnTouchEvent(Pinetime::Applications::TouchEvents event) {
return screens.OnTouchEvent(event); switch (event) {
case Pinetime::Applications::TouchEvents::SwipeUp: {
Controllers::NotificationManager::Notification previousNotification;
if(validDisplay)
previousNotification = notificationManager.GetPrevious(currentId);
else
previousNotification = notificationManager.GetLastNotification();
if (!previousNotification.valid) return true;
validDisplay = true;
currentId = previousNotification.id;
currentItem.reset(nullptr);
app->SetFullRefresh(DisplayApp::FullRefreshDirections::Up);
currentItem.reset(new NotificationItem("Notification", previousNotification.message.data(), previousNotification.index, previousNotification.number, mode));
}
return true;
case Pinetime::Applications::TouchEvents::SwipeDown: {
Controllers::NotificationManager::Notification nextNotification;
if(validDisplay)
nextNotification = notificationManager.GetNext(currentId);
else
nextNotification = notificationManager.GetLastNotification();
if (!nextNotification.valid) return true;
validDisplay = true;
currentId = nextNotification.id;
currentItem.reset(nullptr);
app->SetFullRefresh(DisplayApp::FullRefreshDirections::Down);
currentItem.reset(new NotificationItem("Notification", nextNotification.message.data(), nextNotification.index, nextNotification.number, mode));
}
return true;
default:
return false;
}
} }
@ -32,22 +101,11 @@ bool Notifications::OnButtonPushed() {
return true; return true;
} }
std::unique_ptr<Screen> Notifications::CreateScreen1() {
return std::unique_ptr<Screen>(new Notifications::NotificationItem(app, "Message", "Marcel Pickett: Did you bring your ticket?", "Shot notif", "Short text", 1, 3));
}
std::unique_ptr<Screen> Notifications::CreateScreen2() { Notifications::NotificationItem::NotificationItem(const char *title, const char *msg, uint8_t notifNr, uint8_t notifNb, Modes mode)
return std::unique_ptr<Screen>(new Notifications::NotificationItem(app, "Alarm", "Missed: 09:30", 2, 3)); : notifNr{notifNr}, notifNb{notifNb}, mode{mode} {
}
std::unique_ptr<Screen> Notifications::CreateScreen3() { container1 = lv_cont_create(lv_scr_act(), nullptr);
return std::unique_ptr<Screen>(new Notifications::NotificationItem(app, "Spotify", "Now playing: Bame game - Kanye West", 3, 3));
}
Notifications::NotificationItem::NotificationItem(Pinetime::Applications::DisplayApp *app, const char *title, const char* msg, uint8_t notifNr, uint8_t notifNb) :
Screen(app), notifNr{notifNr}, notifNb{notifNb} {
lv_obj_t* container1 = lv_cont_create(lv_scr_act(), NULL);
static lv_style_t contStyle; static lv_style_t contStyle;
lv_style_copy(&contStyle, lv_cont_get_style(container1, LV_CONT_STYLE_MAIN)); lv_style_copy(&contStyle, lv_cont_get_style(container1, LV_CONT_STYLE_MAIN));
contStyle.body.padding.inner = 20; contStyle.body.padding.inner = 20;
@ -58,10 +116,7 @@ Notifications::NotificationItem::NotificationItem(Pinetime::Applications::Displa
lv_cont_set_layout(container1, LV_LAYOUT_OFF); lv_cont_set_layout(container1, LV_LAYOUT_OFF);
lv_cont_set_fit2(container1, LV_FIT_FLOOD, LV_FIT_FLOOD); lv_cont_set_fit2(container1, LV_FIT_FLOOD, LV_FIT_FLOOD);
t1 = lv_label_create(container1, nullptr);
lv_obj_t* t1 = lv_label_create(container1, NULL);
static lv_style_t titleStyle; static lv_style_t titleStyle;
static lv_style_t textStyle; static lv_style_t textStyle;
static lv_style_t bottomStyle; static lv_style_t bottomStyle;
@ -69,144 +124,46 @@ Notifications::NotificationItem::NotificationItem(Pinetime::Applications::Displa
lv_style_copy(&textStyle, lv_label_get_style(t1, LV_LABEL_STYLE_MAIN)); lv_style_copy(&textStyle, lv_label_get_style(t1, LV_LABEL_STYLE_MAIN));
lv_style_copy(&bottomStyle, lv_label_get_style(t1, LV_LABEL_STYLE_MAIN)); lv_style_copy(&bottomStyle, lv_label_get_style(t1, LV_LABEL_STYLE_MAIN));
titleStyle.body.padding.inner = 5; titleStyle.body.padding.inner = 5;
textStyle.body.padding.inner = 5;
titleStyle.body.grad_color = LV_COLOR_GRAY; titleStyle.body.grad_color = LV_COLOR_GRAY;
titleStyle.body.main_color = LV_COLOR_GRAY; titleStyle.body.main_color = LV_COLOR_GRAY;
titleStyle.body.radius = 20;
textStyle.body.border.part = LV_BORDER_NONE; textStyle.body.border.part = LV_BORDER_NONE;
textStyle.body.padding.inner = 5;
//bottomStyle.body.padding.inner = 5;
bottomStyle.body.main_color = LV_COLOR_GREEN; bottomStyle.body.main_color = LV_COLOR_GREEN;
bottomStyle.body.grad_color = LV_COLOR_GREEN; bottomStyle.body.grad_color = LV_COLOR_GREEN;
bottomStyle.body.border.part = LV_BORDER_TOP; bottomStyle.body.border.part = LV_BORDER_TOP;
bottomStyle.body.border.color = LV_COLOR_RED; bottomStyle.body.border.color = LV_COLOR_RED;
lv_label_set_style(t1, LV_LABEL_STYLE_MAIN, &titleStyle); lv_label_set_style(t1, LV_LABEL_STYLE_MAIN, &titleStyle);
lv_label_set_long_mode(t1, LV_LABEL_LONG_BREAK); lv_label_set_long_mode(t1, LV_LABEL_LONG_BREAK);
lv_label_set_body_draw(t1, true); lv_label_set_body_draw(t1, true);
lv_obj_set_width(t1, LV_HOR_RES - (titleStyle.body.padding.left + titleStyle.body.padding.right) ); lv_obj_set_width(t1, LV_HOR_RES - (titleStyle.body.padding.left + titleStyle.body.padding.right));
lv_label_set_text(t1, title); lv_label_set_text(t1, title);
lv_obj_set_pos(t1, titleStyle.body.padding.left, titleStyle.body.padding.top); lv_obj_set_pos(t1, titleStyle.body.padding.left, titleStyle.body.padding.top);
auto titleHeight = lv_obj_get_height(t1); auto titleHeight = lv_obj_get_height(t1);
lv_obj_t* l1 = lv_label_create(container1, NULL); l1 = lv_label_create(container1, nullptr);
lv_label_set_style(l1, LV_LABEL_STYLE_MAIN, &textStyle); lv_label_set_style(l1, LV_LABEL_STYLE_MAIN, &textStyle);
lv_obj_set_pos(l1, textStyle.body.padding.left, titleHeight + titleStyle.body.padding.bottom + textStyle.body.padding.bottom + textStyle.body.padding.top); lv_obj_set_pos(l1, textStyle.body.padding.left,
titleHeight + titleStyle.body.padding.bottom + textStyle.body.padding.bottom +
textStyle.body.padding.top);
lv_label_set_long_mode(l1, LV_LABEL_LONG_BREAK); lv_label_set_long_mode(l1, LV_LABEL_LONG_BREAK);
lv_label_set_body_draw(l1, true); lv_label_set_body_draw(l1, true);
lv_obj_set_width(l1, LV_HOR_RES - (textStyle.body.padding.left + textStyle.body.padding.right) ); lv_obj_set_width(l1, LV_HOR_RES - (textStyle.body.padding.left + textStyle.body.padding.right));
lv_label_set_text(l1, msg); lv_label_set_text(l1, msg);
lv_obj_t* bottomlabel = lv_label_create(container1, NULL); if(mode == Modes::Normal) {
lv_obj_t *bottomlabel = lv_label_create(container1, nullptr);
lv_label_set_style(bottomlabel, LV_LABEL_STYLE_MAIN, &bottomStyle); lv_label_set_style(bottomlabel, LV_LABEL_STYLE_MAIN, &bottomStyle);
lv_obj_set_width(bottomlabel, LV_HOR_RES - (bottomStyle.body.padding.left + bottomStyle.body.padding.right) ); lv_obj_align(bottomlabel, container1, LV_ALIGN_IN_BOTTOM_RIGHT, 0, 0);
snprintf(pageText, 4, "%d/%d", notifNr, notifNb); snprintf(pageText, 4, "%d/%d", notifNr, notifNb);
lv_label_set_text(bottomlabel, pageText); lv_label_set_text(bottomlabel, pageText);
auto bottomHeight = lv_obj_get_height(bottomlabel); }
lv_obj_set_pos(bottomlabel, 0, LV_VER_RES - (bottomHeight*2));
} }
Notifications::NotificationItem::NotificationItem(Pinetime::Applications::DisplayApp *app, const char *title1, const char* msg1, const char *title2, const char* msg2, uint8_t notifNr, uint8_t notifNb) :
Screen(app), notifNr{notifNr}, notifNb{notifNb} {
lv_obj_t* container1 = lv_cont_create(lv_scr_act(), NULL);
static lv_style_t contStyle;
lv_style_copy(&contStyle, lv_cont_get_style(container1, LV_CONT_STYLE_MAIN));
contStyle.body.padding.inner = 20;
lv_cont_set_style(container1, LV_CONT_STYLE_MAIN, &contStyle);
lv_obj_set_width(container1, LV_HOR_RES);
lv_obj_set_height(container1, LV_VER_RES);
lv_obj_set_pos(container1, 0, 0);
lv_cont_set_layout(container1, LV_LAYOUT_OFF);
lv_cont_set_fit2(container1, LV_FIT_FLOOD, LV_FIT_FLOOD);
lv_obj_t* t1 = lv_label_create(container1, NULL);
static lv_style_t titleStyle;
static lv_style_t textStyle;
static lv_style_t bottomStyle;
lv_style_copy(&titleStyle, lv_label_get_style(t1, LV_LABEL_STYLE_MAIN));
lv_style_copy(&textStyle, lv_label_get_style(t1, LV_LABEL_STYLE_MAIN));
lv_style_copy(&bottomStyle, lv_label_get_style(t1, LV_LABEL_STYLE_MAIN));
titleStyle.body.padding.inner = 5;
textStyle.body.padding.inner = 5;
titleStyle.body.grad_color = LV_COLOR_GRAY;
titleStyle.body.main_color = LV_COLOR_GRAY;
textStyle.body.border.part = LV_BORDER_NONE;
//bottomStyle.body.padding.inner = 5;
bottomStyle.body.main_color = LV_COLOR_GREEN;
bottomStyle.body.grad_color = LV_COLOR_GREEN;
bottomStyle.body.border.part = LV_BORDER_TOP;
bottomStyle.body.border.color = LV_COLOR_RED;
lv_label_set_style(t1, LV_LABEL_STYLE_MAIN, &titleStyle);
lv_label_set_long_mode(t1, LV_LABEL_LONG_BREAK);
lv_label_set_body_draw(t1, true);
lv_obj_set_width(t1, LV_HOR_RES - (titleStyle.body.padding.left + titleStyle.body.padding.right) );
lv_label_set_text(t1, title1);
lv_obj_set_pos(t1, titleStyle.body.padding.left, titleStyle.body.padding.top);
auto titleHeight = lv_obj_get_height(t1);
lv_obj_t* l1 = lv_label_create(container1, NULL);
lv_label_set_style(l1, LV_LABEL_STYLE_MAIN, &textStyle);
lv_obj_set_pos(l1, textStyle.body.padding.left, titleHeight + titleStyle.body.padding.bottom + textStyle.body.padding.bottom + textStyle.body.padding.top);
lv_label_set_long_mode(l1, LV_LABEL_LONG_BREAK);
lv_label_set_body_draw(l1, true);
lv_obj_set_width(l1, LV_HOR_RES - (textStyle.body.padding.left + textStyle.body.padding.right) );
lv_label_set_text(l1, msg1);
auto bottomPos = lv_obj_get_y(l1) + lv_obj_get_height(l1) + textStyle.body.padding.bottom;
/*
lv_obj_t* bottomlabel = lv_label_create(container1, NULL);
lv_label_set_style(bottomlabel, LV_LABEL_STYLE_MAIN, &bottomStyle);
lv_obj_set_width(bottomlabel, LV_HOR_RES - (bottomStyle.body.padding.left + bottomStyle.body.padding.right) );
snprintf(pageText, 4, "%d/%d", notifNr, notifNb);
lv_label_set_text(bottomlabel, pageText);
auto bottomHeight = lv_obj_get_height(bottomlabel);
lv_obj_set_pos(bottomlabel, 0, LV_VER_RES - (bottomHeight*2));
*/
//-------------------------------------------------
/*
lv_obj_t* container2 = lv_cont_create(lv_scr_act(), NULL);
lv_cont_set_style(container2, LV_CONT_STYLE_MAIN, &contStyle);
lv_obj_set_width(container2, LV_HOR_RES);
lv_obj_set_height(container2, LV_VER_RES - bottomPos);
lv_obj_set_pos(container2, 0, bottomPos);
lv_cont_set_layout(container2, LV_LAYOUT_OFF);
lv_cont_set_fit2(container2, LV_FIT_FLOOD, LV_FIT_FLOOD);
*/
lv_obj_t* t2 = lv_label_create(container1, NULL);
lv_label_set_style(t2, LV_LABEL_STYLE_MAIN, &titleStyle);
lv_label_set_long_mode(t2, LV_LABEL_LONG_BREAK);
lv_label_set_body_draw(t2, true);
lv_obj_set_width(t2, LV_HOR_RES - (titleStyle.body.padding.left + titleStyle.body.padding.right) );
lv_label_set_text(t2, title2);
lv_obj_set_pos(t2, titleStyle.body.padding.left, bottomPos + titleStyle.body.padding.top);
auto title2Height = lv_obj_get_height(t2);
lv_obj_t* l2 = lv_label_create(container1, NULL);
lv_label_set_style(l2, LV_LABEL_STYLE_MAIN, &textStyle);
lv_obj_set_pos(l2, textStyle.body.padding.left, bottomPos + title2Height + titleStyle.body.padding.bottom + textStyle.body.padding.bottom + textStyle.body.padding.top);
lv_label_set_long_mode(l2, LV_LABEL_LONG_BREAK);
lv_label_set_body_draw(l2, true);
lv_obj_set_width(l2, LV_HOR_RES - (textStyle.body.padding.left + textStyle.body.padding.right) );
lv_label_set_text(l2, msg2);
}
Notifications::NotificationItem::~NotificationItem() { Notifications::NotificationItem::~NotificationItem() {
lv_obj_clean(lv_scr_act()); lv_obj_clean(lv_scr_act());

View file

@ -12,7 +12,8 @@ namespace Pinetime {
namespace Screens { namespace Screens {
class Notifications : public Screen { class Notifications : public Screen {
public: public:
explicit Notifications(DisplayApp* app); enum class Modes {Normal, Preview};
explicit Notifications(DisplayApp* app, Pinetime::Controllers::NotificationManager& notificationManager, Modes mode);
~Notifications() override; ~Notifications() override;
bool Refresh() override; bool Refresh() override;
@ -20,24 +21,39 @@ namespace Pinetime {
bool OnTouchEvent(Pinetime::Applications::TouchEvents event) override; bool OnTouchEvent(Pinetime::Applications::TouchEvents event) override;
private: private:
ScreenList<3> screens;
bool running = true; bool running = true;
std::unique_ptr<Screen> CreateScreen1();
std::unique_ptr<Screen> CreateScreen2();
std::unique_ptr<Screen> CreateScreen3();
class NotificationItem : public Screen { class NotificationItem {
public: public:
NotificationItem(DisplayApp* app, const char* title, const char* msg, uint8_t notifNr, uint8_t notifNb); NotificationItem(const char* title, const char* msg, uint8_t notifNr, uint8_t notifNb, Modes mode);
NotificationItem(DisplayApp* app, const char* title1, const char* msg1, const char* title2, const char* msg2, uint8_t notifNr, uint8_t notifNb); ~NotificationItem();
~NotificationItem() override; bool Refresh() {return false;}
bool Refresh() override {return false;}
private: private:
uint8_t notifNr = 0; uint8_t notifNr = 0;
uint8_t notifNb = 0; uint8_t notifNb = 0;
char pageText[4]; char pageText[4];
lv_obj_t* container1;
lv_obj_t* t1;
lv_obj_t* l1;
Modes mode;
}; };
struct NotificationData {
const char* title;
const char* text;
};
Pinetime::Controllers::NotificationManager& notificationManager;
Modes mode = Modes::Normal;
std::unique_ptr<NotificationItem> currentItem;
Controllers::NotificationManager::Notification::Id currentId;
bool validDisplay = false;
lv_point_t timeoutLinePoints[2] { {0, 237}, {239, 237} };
lv_obj_t* timeoutLine;
uint32_t timeoutTickCountStart;
uint32_t timeoutTickCountEnd;
}; };
} }
} }

View file

@ -1,145 +0,0 @@
#include "Notifications_swscroll.h"
#include "displayapp/DisplayApp.h"
#include <libs/lvgl/lvgl.h>
using namespace Pinetime::Applications::Screens;
Notifications2::Notifications2(DisplayApp* app) : Screen(app) {
app->SetTouchMode(DisplayApp::TouchModes::Polling);
}
Notifications2::~Notifications2() {
lv_obj_clean(lv_scr_act());
}
bool Notifications2::Refresh() {
return running;
}
void Notifications2::OnObjectEvent(lv_obj_t *obj, lv_event_t event, uint32_t buttonId) {
}
bool Notifications2::OnTouchEvent(Pinetime::Applications::TouchEvents event) { return true; }
bool Notifications2::OnButtonPushed() {
app->StartApp(Apps::Clock);
running = false;
return true;
}
Notifications2::ListWidget::ListWidget() {
static lv_point_t valid_pos[] = {{0,0}, {0,1}};
page = lv_tileview_create(lv_scr_act(), NULL);
lv_obj_set_size(page, LV_HOR_RES, LV_VER_RES);
lv_obj_align(page, NULL, LV_ALIGN_CENTER, 0, 0);
lv_tileview_set_valid_positions(page, valid_pos, 2);
static lv_style_t pageStyle;
lv_style_copy(&pageStyle, lv_tileview_get_style(page, LV_TILEVIEW_STYLE_MAIN));
lv_tileview_set_style(page, LV_TILEVIEW_STYLE_MAIN, &pageStyle);
lv_obj_t* container1 = lv_cont_create(page, NULL);
static lv_style_t contStyle;
lv_style_copy(&contStyle, lv_cont_get_style(container1, LV_CONT_STYLE_MAIN));
contStyle.body.padding.inner = 20;
lv_cont_set_style(container1, LV_CONT_STYLE_MAIN, &contStyle);
lv_obj_set_width(container1, LV_HOR_RES);
lv_obj_set_pos(container1, 0, 0);
lv_cont_set_layout(container1, LV_LAYOUT_COL_M);
lv_cont_set_fit2(container1, LV_FIT_FLOOD, LV_FIT_TIGHT);
lv_tileview_add_element(page, container1);
lv_obj_t* t1 = lv_label_create(container1, NULL);
static lv_style_t titleStyle;
static lv_style_t textStyle;
lv_style_copy(&titleStyle, lv_label_get_style(t1, LV_LABEL_STYLE_MAIN));
lv_style_copy(&textStyle, lv_label_get_style(t1, LV_LABEL_STYLE_MAIN));
//titleStyle.body.main_color = LV_COLOR_RED;
//titleStyle.body.grad_color = LV_COLOR_RED;
titleStyle.body.padding.inner = 5;
//textStyle.body.main_color = LV_COLOR_BLUE;
//textStyle.body.grad_color = LV_COLOR_BLUE;
textStyle.body.padding.inner = 5;
lv_label_set_style(t1, LV_LABEL_STYLE_MAIN, &titleStyle);
lv_label_set_long_mode(t1, LV_LABEL_LONG_BREAK);
lv_label_set_body_draw(t1, true);
lv_obj_set_width(t1, LV_HOR_RES - (titleStyle.body.padding.left + titleStyle.body.padding.right) );
lv_label_set_text(t1, "Message");
lv_obj_t* l1 = lv_label_create(container1, NULL);
lv_label_set_style(l1, LV_PAGE_STYLE_BG, &textStyle);
lv_label_set_long_mode(l1, LV_LABEL_LONG_BREAK);
lv_label_set_body_draw(l1, true);
lv_obj_set_width(l1, LV_HOR_RES - (textStyle.body.padding.left + textStyle.body.padding.right) );
lv_label_set_text(l1, "Marcel Pickett: Did you bring your ticket?");
/*---*/
lv_obj_t* container2 = lv_cont_create(page, NULL);
lv_cont_set_style(container2, LV_CONT_STYLE_MAIN, &contStyle);
lv_obj_set_width(container2, LV_HOR_RES);
lv_obj_set_pos(container2, 0, lv_obj_get_y(container1) + lv_obj_get_height(container1)+5);
lv_cont_set_layout(container2, LV_LAYOUT_COL_M);
lv_cont_set_fit2(container2, LV_FIT_FLOOD, LV_FIT_TIGHT);
lv_tileview_add_element(page, container2);
lv_obj_t* t2 = lv_label_create(container2, NULL);
lv_label_set_style(t2, LV_PAGE_STYLE_BG, &titleStyle);
lv_label_set_long_mode(t2, LV_LABEL_LONG_BREAK);
lv_label_set_body_draw(t2, true);
lv_obj_set_width(t2, LV_HOR_RES - (titleStyle.body.padding.left + titleStyle.body.padding.right) );
lv_label_set_text(t2, "Alarm");
lv_obj_t* l2 = lv_label_create(container2, NULL);
lv_label_set_style(l2, LV_PAGE_STYLE_BG, &textStyle);
lv_label_set_long_mode(l2, LV_LABEL_LONG_BREAK);
lv_label_set_body_draw(l2, true);
lv_obj_set_width(l2, LV_HOR_RES - (textStyle.body.padding.left + textStyle.body.padding.right) );
lv_label_set_text(l2, "Missed: 09:30");
/*****/
lv_obj_t* container3 = lv_cont_create(page, NULL);
lv_cont_set_style(container3, LV_CONT_STYLE_MAIN, &contStyle);
lv_obj_set_width(container3, LV_HOR_RES);
lv_obj_set_pos(container3, 0, lv_obj_get_y(container2) + lv_obj_get_height(container2)+5);
lv_cont_set_layout(container3, LV_LAYOUT_COL_M);
lv_cont_set_fit2(container3, LV_FIT_FLOOD, LV_FIT_TIGHT);
lv_tileview_add_element(page, container3);
lv_obj_t* t3 = lv_label_create(container3, NULL);
lv_label_set_style(t3, LV_PAGE_STYLE_BG, &titleStyle);
lv_label_set_long_mode(t3, LV_LABEL_LONG_BREAK);
lv_label_set_body_draw(t3, true);
lv_obj_set_width(t3, LV_HOR_RES - (titleStyle.body.padding.left + titleStyle.body.padding.right) );
lv_label_set_text(t3, "Spotify");
lv_obj_t* l3 = lv_label_create(container3, NULL);
lv_label_set_style(l3, LV_PAGE_STYLE_BG, &textStyle);
lv_label_set_long_mode(l3, LV_LABEL_LONG_BREAK);
lv_label_set_body_draw(l3, true);
lv_obj_set_width(l3, LV_HOR_RES - (textStyle.body.padding.left + textStyle.body.padding.right) );
lv_label_set_text(l3, "Now playing: Bame game - Kanye West");
}

View file

@ -1,36 +0,0 @@
#pragma once
#include <cstdint>
#include "Screen.h"
#include <bits/unique_ptr.h>
#include "Modal.h"
#include <lvgl/src/lv_core/lv_style.h>
#include <displayapp/Apps.h>
namespace Pinetime {
namespace Applications {
namespace Screens {
class Notifications2 : public Screen {
public:
explicit Notifications2(DisplayApp* app);
~Notifications2() override;
bool Refresh() override;
bool OnButtonPushed() override;
void OnObjectEvent(lv_obj_t* obj, lv_event_t event, uint32_t buttonId);
bool OnTouchEvent(Pinetime::Applications::TouchEvents event) override;
private:
class ListWidget {
public:
ListWidget();
private:
lv_obj_t* page = nullptr;
};
bool running = true;
ListWidget list;
};
}
}
}