From 95a2ab6cfb350abe782e914dd705a7912984681b Mon Sep 17 00:00:00 2001 From: JustScott Date: Tue, 4 Feb 2025 22:07:48 -0600 Subject: [PATCH 1/3] Add an hour option to the timer application --- src/components/timer/Timer.cpp | 29 +++++++++++++++++++------- src/components/timer/Timer.h | 6 +++++- src/displayapp/DisplayApp.cpp | 23 +++++++++++++-------- src/displayapp/screens/Timer.cpp | 35 ++++++++++++++++++++++---------- src/displayapp/screens/Timer.h | 5 +++-- 5 files changed, 68 insertions(+), 30 deletions(-) diff --git a/src/components/timer/Timer.cpp b/src/components/timer/Timer.cpp index 279178cd33..cb731bd019 100644 --- a/src/components/timer/Timer.cpp +++ b/src/components/timer/Timer.cpp @@ -3,26 +3,41 @@ using namespace Pinetime::Controllers; Timer::Timer(void* const timerData, TimerCallbackFunction_t timerCallbackFunction) { - timer = xTimerCreate("Timer", 1, pdFALSE, timerData, timerCallbackFunction); + timerHandle = xTimerCreate("Timer", 1, pdFALSE, timerData, timerCallbackFunction); } void Timer::StartTimer(std::chrono::milliseconds duration) { - xTimerChangePeriod(timer, pdMS_TO_TICKS(duration.count()), 0); - xTimerStart(timer, 0); + timerOverflowIntervals = 0; + + if (duration.count() > maxTimerMS) { + timerOverflowIntervals = duration.count() / maxTimerMS; + uint32_t remainingMS = duration.count() % maxTimerMS; + uint8_t leftoverMinutes = remainingMS / 60 / 1000; + uint8_t leftoverSeconds = (remainingMS % (60 * 1000)) / 1000; + if (leftoverMinutes == 0 && leftoverSeconds == 0) { + leftoverMinutes = 59; + leftoverSeconds = 60; + timerOverflowIntervals--; + } + xTimerChangePeriod(timerHandle, pdMS_TO_TICKS((leftoverMinutes * 60 * 1000) + (leftoverSeconds * 1000)), 0); + } else { + xTimerChangePeriod(timerHandle, pdMS_TO_TICKS(duration.count()), 0); + } + xTimerStart(timerHandle, 0); } std::chrono::milliseconds Timer::GetTimeRemaining() { if (IsRunning()) { - TickType_t remainingTime = xTimerGetExpiryTime(timer) - xTaskGetTickCount(); - return std::chrono::milliseconds(remainingTime * 1000 / configTICK_RATE_HZ); + TickType_t remainingTime = xTimerGetExpiryTime(timerHandle) - xTaskGetTickCount(); + return std::chrono::milliseconds((remainingTime * 1000 / configTICK_RATE_HZ) + (timerOverflowIntervals * maxTimerMS)); } return std::chrono::milliseconds(0); } void Timer::StopTimer() { - xTimerStop(timer, 0); + xTimerStop(timerHandle, 0); } bool Timer::IsRunning() { - return (xTimerIsTimerActive(timer) == pdTRUE); + return (xTimerIsTimerActive(timerHandle) == pdTRUE); } diff --git a/src/components/timer/Timer.h b/src/components/timer/Timer.h index 2469666f4f..e9f27c24cc 100644 --- a/src/components/timer/Timer.h +++ b/src/components/timer/Timer.h @@ -19,8 +19,12 @@ namespace Pinetime { bool IsRunning(); + uint8_t timerOverflowIntervals = 0; + + TimerHandle_t timerHandle; + + const uint32_t maxTimerMS = 3'600'000; // 1 hour private: - TimerHandle_t timer; }; } } diff --git a/src/displayapp/DisplayApp.cpp b/src/displayapp/DisplayApp.cpp index 6671ac9e51..09f7efd9f0 100644 --- a/src/displayapp/DisplayApp.cpp +++ b/src/displayapp/DisplayApp.cpp @@ -362,17 +362,22 @@ void DisplayApp::Refresh() { LoadNewScreen(Apps::NotificationsPreview, DisplayApp::FullRefreshDirections::Down); break; case Messages::TimerDone: - if (state != States::Running) { - PushMessageToSystemTask(System::Messages::GoToRunning); - } - if (currentApp == Apps::Timer) { - lv_disp_trig_activity(nullptr); - auto* timer = static_cast(currentScreen.get()); - timer->Reset(); + if (timer.timerOverflowIntervals > 0) { + timer.StopTimer(); + timer.StartTimer(std::chrono::milliseconds(timer.timerOverflowIntervals * timer.maxTimerMS)); } else { - LoadNewScreen(Apps::Timer, DisplayApp::FullRefreshDirections::Up); + if (state != States::Running) { + PushMessageToSystemTask(System::Messages::GoToRunning); + } + if (currentApp == Apps::Timer) { + lv_disp_trig_activity(nullptr); + auto* timer = static_cast(currentScreen.get()); + timer->Reset(); + } else { + LoadNewScreen(Apps::Timer, DisplayApp::FullRefreshDirections::Up); + } + motorController.RunForDuration(35); } - motorController.RunForDuration(35); break; case Messages::AlarmTriggered: if (currentApp == Apps::Alarm) { diff --git a/src/displayapp/screens/Timer.cpp b/src/displayapp/screens/Timer.cpp index f6d5e73bdc..86f3d5097c 100644 --- a/src/displayapp/screens/Timer.cpp +++ b/src/displayapp/screens/Timer.cpp @@ -19,16 +19,24 @@ static void btnEventHandler(lv_obj_t* obj, lv_event_t event) { Timer::Timer(Controllers::Timer& timerController) : timer {timerController} { - lv_obj_t* colonLabel = lv_label_create(lv_scr_act(), nullptr); - lv_obj_set_style_local_text_font(colonLabel, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, &jetbrains_mono_76); - lv_obj_set_style_local_text_color(colonLabel, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, LV_COLOR_WHITE); - lv_label_set_text_static(colonLabel, ":"); - lv_obj_align(colonLabel, lv_scr_act(), LV_ALIGN_CENTER, 0, -29); - + lv_obj_t* colonMinutesSeconds = lv_label_create(lv_scr_act(), nullptr); + lv_obj_set_style_local_text_font(colonMinutesSeconds, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, &jetbrains_mono_42); + lv_obj_set_style_local_text_color(colonMinutesSeconds, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, LV_COLOR_WHITE); + lv_label_set_text_static(colonMinutesSeconds, ":"); + lv_obj_align(colonMinutesSeconds, lv_scr_act(), LV_ALIGN_IN_TOP_MID, 40, 78); + + lv_obj_t* colonHoursMinutes = lv_label_create(lv_scr_act(), nullptr); + lv_obj_set_style_local_text_font(colonHoursMinutes, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, &jetbrains_mono_42); + lv_obj_set_style_local_text_color(colonHoursMinutes, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, LV_COLOR_WHITE); + lv_label_set_text_static(colonHoursMinutes, ":"); + lv_obj_align(colonHoursMinutes, lv_scr_act(), LV_ALIGN_IN_TOP_MID, -41, 78); + + hourCounter.Create(); minuteCounter.Create(); secondCounter.Create(); - lv_obj_align(minuteCounter.GetObject(), nullptr, LV_ALIGN_IN_TOP_LEFT, 0, 0); - lv_obj_align(secondCounter.GetObject(), nullptr, LV_ALIGN_IN_TOP_RIGHT, 0, 0); + lv_obj_align(hourCounter.GetObject(), nullptr, LV_ALIGN_IN_TOP_LEFT, 10, 26); + lv_obj_align(minuteCounter.GetObject(), nullptr, LV_ALIGN_IN_TOP_MID, 0, 26); + lv_obj_align(secondCounter.GetObject(), nullptr, LV_ALIGN_IN_TOP_RIGHT, -10, 26); highlightObjectMask = lv_objmask_create(lv_scr_act(), nullptr); lv_obj_set_size(highlightObjectMask, 240, 50); @@ -120,18 +128,21 @@ void Timer::Refresh() { void Timer::DisplayTime() { displaySeconds = std::chrono::duration_cast(timer.GetTimeRemaining()); if (displaySeconds.IsUpdated()) { - minuteCounter.SetValue(displaySeconds.Get().count() / 60); + hourCounter.SetValue(displaySeconds.Get().count() / 3600); + minuteCounter.SetValue((displaySeconds.Get().count() % 3600) / 60); secondCounter.SetValue(displaySeconds.Get().count() % 60); } } void Timer::SetTimerRunning() { + hourCounter.HideControls(); minuteCounter.HideControls(); secondCounter.HideControls(); lv_label_set_text_static(txtPlayPause, "Pause"); } void Timer::SetTimerStopped() { + hourCounter.ShowControls(); minuteCounter.ShowControls(); secondCounter.ShowControls(); lv_label_set_text_static(txtPlayPause, "Start"); @@ -142,9 +153,11 @@ void Timer::ToggleRunning() { DisplayTime(); timer.StopTimer(); SetTimerStopped(); - } else if (secondCounter.GetValue() + minuteCounter.GetValue() > 0) { - auto timerDuration = std::chrono::minutes(minuteCounter.GetValue()) + std::chrono::seconds(secondCounter.GetValue()); + } else if (secondCounter.GetValue() + minuteCounter.GetValue() + hourCounter.GetValue() > 0) { + std::chrono::milliseconds timerDuration = std::chrono::hours(hourCounter.GetValue()) + std::chrono::minutes(minuteCounter.GetValue()) + + std::chrono::seconds(secondCounter.GetValue()); timer.StartTimer(timerDuration); + displaySeconds = std::chrono::duration_cast(timer.GetTimeRemaining()); Refresh(); SetTimerRunning(); } diff --git a/src/displayapp/screens/Timer.h b/src/displayapp/screens/Timer.h index a07c729b4d..710d563369 100644 --- a/src/displayapp/screens/Timer.h +++ b/src/displayapp/screens/Timer.h @@ -38,8 +38,9 @@ namespace Pinetime::Applications { lv_objmask_mask_t* highlightMask; lv_task_t* taskRefresh; - Widgets::Counter minuteCounter = Widgets::Counter(0, 59, jetbrains_mono_76); - Widgets::Counter secondCounter = Widgets::Counter(0, 59, jetbrains_mono_76); + Widgets::Counter hourCounter = Widgets::Counter(0, 99, jetbrains_mono_42); + Widgets::Counter minuteCounter = Widgets::Counter(0, 59, jetbrains_mono_42); + Widgets::Counter secondCounter = Widgets::Counter(0, 59, jetbrains_mono_42); bool buttonPressing = false; lv_coord_t maskPosition = 0; From 9809f2faf1772ac21374e7b8498d16a3ab42559a Mon Sep 17 00:00:00 2001 From: JustScott Date: Fri, 7 Feb 2025 03:56:31 -0600 Subject: [PATCH 2/3] Remove unecessary time.GetTimeRemaining call --- src/displayapp/screens/Timer.cpp | 1 - 1 file changed, 1 deletion(-) diff --git a/src/displayapp/screens/Timer.cpp b/src/displayapp/screens/Timer.cpp index 86f3d5097c..25b0fd2a91 100644 --- a/src/displayapp/screens/Timer.cpp +++ b/src/displayapp/screens/Timer.cpp @@ -157,7 +157,6 @@ void Timer::ToggleRunning() { std::chrono::milliseconds timerDuration = std::chrono::hours(hourCounter.GetValue()) + std::chrono::minutes(minuteCounter.GetValue()) + std::chrono::seconds(secondCounter.GetValue()); timer.StartTimer(timerDuration); - displaySeconds = std::chrono::duration_cast(timer.GetTimeRemaining()); Refresh(); SetTimerRunning(); } From 00c463ea1bd162ae8eba32642bea6f9dee49ee56 Mon Sep 17 00:00:00 2001 From: JustScott Date: Sat, 15 Feb 2025 02:35:03 -0600 Subject: [PATCH 3/3] Revert timerHandle variable name back to just timer --- src/components/timer/Timer.cpp | 14 +++++++------- src/components/timer/Timer.h | 2 +- 2 files changed, 8 insertions(+), 8 deletions(-) diff --git a/src/components/timer/Timer.cpp b/src/components/timer/Timer.cpp index cb731bd019..bbc80084be 100644 --- a/src/components/timer/Timer.cpp +++ b/src/components/timer/Timer.cpp @@ -3,7 +3,7 @@ using namespace Pinetime::Controllers; Timer::Timer(void* const timerData, TimerCallbackFunction_t timerCallbackFunction) { - timerHandle = xTimerCreate("Timer", 1, pdFALSE, timerData, timerCallbackFunction); + timer = xTimerCreate("Timer", 1, pdFALSE, timerData, timerCallbackFunction); } void Timer::StartTimer(std::chrono::milliseconds duration) { @@ -19,25 +19,25 @@ void Timer::StartTimer(std::chrono::milliseconds duration) { leftoverSeconds = 60; timerOverflowIntervals--; } - xTimerChangePeriod(timerHandle, pdMS_TO_TICKS((leftoverMinutes * 60 * 1000) + (leftoverSeconds * 1000)), 0); + xTimerChangePeriod(timer, pdMS_TO_TICKS((leftoverMinutes * 60 * 1000) + (leftoverSeconds * 1000)), 0); } else { - xTimerChangePeriod(timerHandle, pdMS_TO_TICKS(duration.count()), 0); + xTimerChangePeriod(timer, pdMS_TO_TICKS(duration.count()), 0); } - xTimerStart(timerHandle, 0); + xTimerStart(timer, 0); } std::chrono::milliseconds Timer::GetTimeRemaining() { if (IsRunning()) { - TickType_t remainingTime = xTimerGetExpiryTime(timerHandle) - xTaskGetTickCount(); + TickType_t remainingTime = xTimerGetExpiryTime(timer) - xTaskGetTickCount(); return std::chrono::milliseconds((remainingTime * 1000 / configTICK_RATE_HZ) + (timerOverflowIntervals * maxTimerMS)); } return std::chrono::milliseconds(0); } void Timer::StopTimer() { - xTimerStop(timerHandle, 0); + xTimerStop(timer, 0); } bool Timer::IsRunning() { - return (xTimerIsTimerActive(timerHandle) == pdTRUE); + return (xTimerIsTimerActive(timer) == pdTRUE); } diff --git a/src/components/timer/Timer.h b/src/components/timer/Timer.h index e9f27c24cc..6099e95c2b 100644 --- a/src/components/timer/Timer.h +++ b/src/components/timer/Timer.h @@ -21,7 +21,7 @@ namespace Pinetime { uint8_t timerOverflowIntervals = 0; - TimerHandle_t timerHandle; + TimerHandle_t timer; const uint32_t maxTimerMS = 3'600'000; // 1 hour private: