Author: Adam Evyčędo <git@apiote.xyz>
add face mine
src/CMakeLists.txt | 5 src/displayapp/UserApps.h | 5 src/displayapp/apps/Apps.h.in | 5 src/displayapp/apps/CMakeLists.txt | 5 src/displayapp/screens/WatchFaceMine.cpp | 529 ++++++++++++++++++++++++++ src/displayapp/screens/WatchFaceMine.h | 152 +++++++
diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index 0a97a0158b2f9af46d4ef4a073906e25ea5481e8..d4804ffe19b797d69ee317161f1d37c690226dc0 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -420,12 +420,13 @@ displayapp/screens/settings/SettingShakeThreshold.cpp displayapp/screens/settings/SettingBluetooth.cpp ## Watch faces - displayapp/screens/WatchFaceAnalog.cpp + # displayapp/screens/WatchFaceAnalog.cpp displayapp/screens/WatchFaceDigital.cpp displayapp/screens/WatchFaceInfineat.cpp displayapp/screens/WatchFaceTerminal.cpp - displayapp/screens/WatchFacePineTimeStyle.cpp + # displayapp/screens/WatchFacePineTimeStyle.cpp displayapp/screens/WatchFaceCasioStyleG7710.cpp + displayapp/screens/WatchFaceMine.cpp ## diff --git a/src/displayapp/UserApps.h b/src/displayapp/UserApps.h index 67bbfa7d411b99d2a8553bea7f2d7820e35453d1..30e3478ad77fa6c7e268ff4bf9e05964959d48fb 100644 --- a/src/displayapp/UserApps.h +++ b/src/displayapp/UserApps.h @@ -9,10 +9,11 @@ #include "displayapp/screens/Twos.h" #include "displayapp/screens/Tile.h" #include "displayapp/screens/ApplicationList.h" #include "displayapp/screens/WatchFaceDigital.h" -#include "displayapp/screens/WatchFaceAnalog.h" +// #include "displayapp/screens/WatchFaceAnalog.h" #include "displayapp/screens/WatchFaceCasioStyleG7710.h" #include "displayapp/screens/WatchFaceInfineat.h" -#include "displayapp/screens/WatchFacePineTimeStyle.h" +// #include "displayapp/screens/WatchFacePineTimeStyle.h" +#include "displayapp/screens/WatchFaceMine.h" #include "displayapp/screens/WatchFaceTerminal.h" namespace Pinetime { diff --git a/src/displayapp/apps/Apps.h.in b/src/displayapp/apps/Apps.h.in index 2104a267c0f79d840e6bd316729357b270432026..5329f0c02190a5042a026e466dc11405e926e439 100644 --- a/src/displayapp/apps/Apps.h.in +++ b/src/displayapp/apps/Apps.h.in @@ -47,8 +47,9 @@ }; enum class WatchFace : uint8_t { Digital, - Analog, - PineTimeStyle, + // Analog, + // PineTimeStyle, + Mine, Terminal, Infineat, CasioStyleG7710, diff --git a/src/displayapp/apps/CMakeLists.txt b/src/displayapp/apps/CMakeLists.txt index d78587609e500e553c8c645c07874d1422a79a35..78f989a946f113e5f6d2dc80f88b224a8935293d 100644 --- a/src/displayapp/apps/CMakeLists.txt +++ b/src/displayapp/apps/CMakeLists.txt @@ -22,8 +22,9 @@ if(DEFINED ENABLE_WATCHFACES) set(WATCHFACE_TYPES ${ENABLE_WATCHFACES} CACHE STRING "List of watch faces to build into the firmware") else() set(DEFAULT_WATCHFACE_TYPES "WatchFace::Digital") - set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::Analog") - set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::PineTimeStyle") + # set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::Analog") + # set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::PineTimeStyle") + set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::Mine") set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::Terminal") set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::Infineat") set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::CasioStyleG7710") diff --git a/src/displayapp/screens/WatchFaceMine.cpp b/src/displayapp/screens/WatchFaceMine.cpp new file mode 100644 index 0000000000000000000000000000000000000000..392e05544e1e307bbc1c2c6e59970683b282ce2a --- /dev/null +++ b/src/displayapp/screens/WatchFaceMine.cpp @@ -0,0 +1,529 @@ +#include "displayapp/screens/WatchFaceMine.h" +#include <cmath> +#include <lvgl/lvgl.h> +#include <chrono> +#include "displayapp/screens/BatteryIcon.h" +#include "displayapp/screens/BleIcon.h" +#include "displayapp/screens/Symbols.h" +#include "displayapp/screens/NotificationIcon.h" +#include "components/settings/Settings.h" +#include "displayapp/screens/WeatherSymbols.h" +#include "components/ble/SimpleWeatherService.h" +#include "displayapp/InfiniTimeTheme.h" + +using namespace Pinetime::Applications::Screens; +using namespace std::literals::chrono_literals; + +namespace { + constexpr int16_t HourLength = 70; + constexpr int16_t MinuteLength = 100; + constexpr int16_t SecondLength = 100; + + // sin(90) = 1 so the value of _lv_trigo_sin(90) is the scaling factor + const auto LV_TRIG_SCALE = _lv_trigo_sin(90); + + int16_t Cosine(int16_t angle) { + return _lv_trigo_sin(angle + 90); + } + + int16_t Sine(int16_t angle) { + return _lv_trigo_sin(angle); + } + + int16_t CoordinateXRelocate(int16_t x) { + return (x + LV_HOR_RES / 2); + } + + int16_t CoordinateYRelocate(int16_t y) { + return std::abs(y - LV_HOR_RES / 2); + } + + lv_point_t CoordinateRelocate(int16_t radius, int16_t angle) { + return lv_point_t {.x = CoordinateXRelocate(radius * static_cast<int32_t>(Sine(angle)) / LV_TRIG_SCALE), + .y = CoordinateYRelocate(radius * static_cast<int32_t>(Cosine(angle)) / LV_TRIG_SCALE)}; + } + +} + +WatchFaceMine::WatchFaceMine(Controllers::DateTime& dateTimeController, + const Controllers::Battery& batteryController, + const Controllers::Ble& bleController, + Controllers::NotificationManager& notificationManager, + Controllers::Settings& settingsController, + Controllers::MotionController& motionController, + Controllers::SimpleWeatherService& weatherService) + : currentDateTime {{}}, + dateTimeController {dateTimeController}, + batteryController {batteryController}, + bleController {bleController}, + notificationManager {notificationManager}, + settingsController {settingsController}, + motionController {motionController}, + weatherService {weatherService} { + + sHour = 99; + sMinute = 99; + sSecond = 99; + + batteryIcon.Create(lv_scr_act()); + lv_obj_align(batteryIcon.GetObject(), nullptr, LV_ALIGN_IN_TOP_RIGHT, 0, 0); + + plugIcon = lv_label_create(lv_scr_act(), nullptr); + lv_label_set_text_static(plugIcon, Symbols::plug); + lv_obj_align(plugIcon, nullptr, LV_ALIGN_IN_TOP_RIGHT, 0, 0); + + notificationIcon = lv_label_create(lv_scr_act(), NULL); + lv_obj_set_style_local_text_color(notificationIcon, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, lv_color_hex(0xdb9d3b)); + lv_label_set_text_static(notificationIcon, NotificationIcon::GetIcon(false)); + lv_obj_align(notificationIcon, NULL, LV_ALIGN_IN_TOP_LEFT, 0, 0); + + bluetoothIcon = lv_label_create(lv_scr_act(), NULL); + lv_obj_set_style_local_text_color(bluetoothIcon, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, lv_color_hex(0x0082fc)); + lv_label_set_text_static(bluetoothIcon, BleIcon::GetIcon(false)); + lv_obj_align(bluetoothIcon, NULL, LV_ALIGN_IN_TOP_RIGHT, -30, 0); + + weatherIcon = lv_label_create(lv_scr_act(), nullptr); + lv_obj_set_style_local_text_color(weatherIcon, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, lv_color_hex(0x7b7e82)); + lv_obj_set_style_local_text_font(weatherIcon, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, &fontawesome_weathericons); + lv_label_set_text(weatherIcon, Symbols::ban); + lv_obj_align(weatherIcon, nullptr, LV_ALIGN_IN_TOP_MID, -15, 50); + lv_obj_set_auto_realign(weatherIcon, true); + if (settingsController.GetPTSWeather() == Pinetime::Controllers::Settings::PTSWeather::On) { + lv_obj_set_hidden(weatherIcon, false); + } else { + lv_obj_set_hidden(weatherIcon, true); + } + + temperature = lv_label_create(lv_scr_act(), nullptr); + lv_label_set_text(temperature, "--"); + lv_obj_set_style_local_text_color(temperature, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, lv_color_hex(0x7b7e82)); + lv_obj_align(temperature, nullptr, LV_ALIGN_IN_TOP_MID, 15, 50); + if (settingsController.GetPTSWeather() == Pinetime::Controllers::Settings::PTSWeather::On) { + lv_obj_set_hidden(temperature, false); + } else { + lv_obj_set_hidden(temperature, true); + } + + + lv_style_init(&tick_style); + lv_style_set_line_width(&tick_style, LV_STATE_DEFAULT, 5); + lv_style_set_line_color(&tick_style, LV_STATE_DEFAULT, lv_color_hex(0x7b7e82)); + lv_style_set_line_rounded(&tick_style, LV_STATE_DEFAULT, true); + lv_style_init(&tick_style_zero); + lv_style_set_line_width(&tick_style_zero, LV_STATE_DEFAULT, 5); + lv_style_set_line_color(&tick_style_zero, LV_STATE_DEFAULT, lv_color_hex(0xdb9d3b)); + lv_style_set_line_rounded(&tick_style_zero, LV_STATE_DEFAULT, true); + lv_style_init(&tick_style_six); + lv_style_set_line_width(&tick_style_six, LV_STATE_DEFAULT, 5); + lv_style_set_line_color(&tick_style_six, LV_STATE_DEFAULT, lv_color_hex(0x5d3636)); + lv_style_set_line_rounded(&tick_style_six, LV_STATE_DEFAULT, true); + + for (int i = 0; i < 24; ++i) { + tick[i] = lv_line_create(lv_scr_act(), NULL); + if (i == 0) { + lv_obj_add_style(tick[i], LV_LINE_PART_MAIN, &tick_style_zero); + } else if (i%3 == 0) { + lv_obj_add_style(tick[i], LV_LINE_PART_MAIN, &tick_style_six); + } else { + lv_obj_add_style(tick[i], LV_LINE_PART_MAIN, &tick_style); + } + } + + clockType = settingsController.GetClockType(); + DrawFace(); + + label_date_greg = lv_label_create(lv_scr_act(), NULL); + lv_obj_set_style_local_text_color(label_date_greg, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, lv_color_hex(0x919191)); + lv_label_set_text_fmt(label_date_greg, "%s\n%02i", dateTimeController.DayOfWeekShortToString(), dateTimeController.Day()); + lv_label_set_align(label_date_greg, LV_LABEL_ALIGN_CENTER); + lv_obj_align(label_date_greg, NULL, LV_ALIGN_CENTER, 50, 0); + + label_date_world = lv_label_create(lv_scr_act(), NULL); + lv_obj_set_style_local_text_color(label_date_world, LV_LABEL_PART_MAIN, LV_STATE_DEFAULT, lv_color_hex(0x919191)); + lv_label_set_text_fmt(label_date_world, "SUN\n01 \xCF\x80h", dateTimeController.DayOfWeekShortToString(), dateTimeController.Day()); + lv_label_set_align(label_date_world, LV_LABEL_ALIGN_CENTER); + lv_obj_align(label_date_world, NULL, LV_ALIGN_CENTER, -50, 0); + + + // minute_body_world = lv_line_create(lv_scr_act(), NULL); + minute_body = lv_line_create(lv_scr_act(), NULL); + // hour_body_world = lv_line_create(lv_scr_act(), NULL); + hour_body = lv_line_create(lv_scr_act(), NULL); + second_body = lv_line_create(lv_scr_act(), NULL); + + steps_body = lv_line_create(lv_scr_act(), NULL); + steps_body_trace = lv_line_create(lv_scr_act(), NULL); + + lv_style_init(&second_line_style); + lv_style_set_line_width(&second_line_style, LV_STATE_DEFAULT, 3); + lv_style_set_line_color(&second_line_style, LV_STATE_DEFAULT, lv_color_hex(0xff5c57)); + lv_style_set_line_rounded(&second_line_style, LV_STATE_DEFAULT, true); + lv_obj_add_style(second_body, LV_LINE_PART_MAIN, &second_line_style); + + // lv_style_init(&minute_line_style_world); + // lv_style_set_line_width(&minute_line_style_world, LV_STATE_DEFAULT, 5); + // lv_style_set_line_color(&minute_line_style_world, LV_STATE_DEFAULT, lv_color_hex(0x5d3636)); + // lv_style_set_line_rounded(&minute_line_style_world, LV_STATE_DEFAULT, true); + // lv_obj_add_style(minute_body_world, LV_LINE_PART_MAIN, &minute_line_style_world); + + lv_style_init(&minute_line_style); + lv_style_set_line_width(&minute_line_style, LV_STATE_DEFAULT, 5); + lv_style_set_line_color(&minute_line_style, LV_STATE_DEFAULT, lv_color_hex(0xfafafa)); + lv_style_set_line_rounded(&minute_line_style, LV_STATE_DEFAULT, true); + lv_obj_add_style(minute_body, LV_LINE_PART_MAIN, &minute_line_style); + + // lv_style_init(&hour_line_style_world); + // lv_style_set_line_width(&hour_line_style_world, LV_STATE_DEFAULT, 7); + // lv_style_set_line_color(&hour_line_style_world, LV_STATE_DEFAULT, lv_color_hex(0x5d3636)); + // lv_style_set_line_rounded(&hour_line_style_world, LV_STATE_DEFAULT, true); + // lv_obj_add_style(hour_body_world, LV_LINE_PART_MAIN, &hour_line_style_world); + + lv_style_init(&hour_line_style); + lv_style_set_line_width(&hour_line_style, LV_STATE_DEFAULT, 7); + lv_style_set_line_color(&hour_line_style, LV_STATE_DEFAULT, lv_color_hex(0xfafafa)); + lv_style_set_line_rounded(&hour_line_style, LV_STATE_DEFAULT, true); + lv_obj_add_style(hour_body, LV_LINE_PART_MAIN, &hour_line_style); + + + lv_style_init(&steps_line_style); + lv_style_set_line_width(&steps_line_style, LV_STATE_DEFAULT, 7); + lv_style_set_line_color(&steps_line_style, LV_STATE_DEFAULT, lv_color_hex(0x212121)); + lv_style_set_line_rounded(&steps_line_style, LV_STATE_DEFAULT, true); + lv_obj_add_style(steps_body, LV_LINE_PART_MAIN, &steps_line_style); + + lv_style_init(&steps_line_style_trace); + lv_style_set_line_width(&steps_line_style_trace, LV_STATE_DEFAULT, 3); + lv_style_set_line_color(&steps_line_style_trace, LV_STATE_DEFAULT, lv_color_hex(0xdb9d3b)); + lv_style_set_line_rounded(&steps_line_style_trace, LV_STATE_DEFAULT, false); + + lv_style_init(&steps_line_style_full); + lv_style_set_line_width(&steps_line_style_full, LV_STATE_DEFAULT, 3); + lv_style_set_line_color(&steps_line_style_full, LV_STATE_DEFAULT, lv_color_hex(0x1c9867)); + lv_style_set_line_rounded(&steps_line_style_full, LV_STATE_DEFAULT, false); + + lv_obj_add_style(steps_body_trace, LV_LINE_PART_MAIN, &steps_line_style_trace); + + taskRefresh = lv_task_create(RefreshTaskCallback, LV_DISP_DEF_REFR_PERIOD, LV_TASK_PRIO_MID, this); + + Refresh(); +} + +WatchFaceMine::~WatchFaceMine() { + lv_task_del(taskRefresh); + + lv_style_reset(&hour_line_style); + // lv_style_reset(&hour_line_style_world); + lv_style_reset(&minute_line_style); + // lv_style_reset(&minute_line_style_world); + lv_style_reset(&second_line_style); + + lv_style_reset(&tick_style); + lv_style_reset(&tick_style_zero); + lv_style_reset(&tick_style_six); + + lv_style_reset(&steps_line_style); + lv_style_reset(&steps_line_style_trace); + lv_style_reset(&steps_line_style_full); + + lv_obj_clean(lv_scr_act()); +} + +void WatchFaceMine::DrawFace() { + uint8_t div = 1; + switch (clockType.Get()) { + case Controllers::Settings::ClockType::H24: + div = 24; + break; + case Controllers::Settings::ClockType::H12: + div = 12; + break; + } + for (int i = 0; i < div; ++i) { + auto angle = (i * 30) / (div / 12); + tick_point[i][0] = CoordinateRelocate(95, angle); + tick_point[i][1] = CoordinateRelocate(100, angle); + + lv_line_set_points(tick[i], tick_point[i], 2); + } +} + +// void WatchFaceMine::DrawOffsetHands(uint8_t sHour, uint8_t sMinute, uint8_t hour, uint8_t minute) { // from https://github.com/InfiniTimeOrg/InfiniTime/pull/1454 + // worldClockOffset.Get(); + // using days = std::chrono::duration<int, std::ratio<86400>>; + // auto worldDateTime = dateTimeController.WorldDateTime(0); + // auto worldDp = std::chrono::floor<days>(worldDateTime); + // auto worldTime = worldDateTime - worldDp; + // uint8_t worldHour = std::chrono::floor<std::chrono::hours>(worldTime).count(); + + // auto worldHp = std::chrono::floor<std::chrono::hours>(worldTime); + // worldTime = worldTime - worldHp; + // uint8_t worldMinute = std::chrono::floor<std::chrono::minutes>(worldTime).count(); + + // if (sMinute != minute) { + // auto const angle = worldMinute * 6; + // minute_point_world[0] = CoordinateRelocate(0, angle); + // minute_point_world[1] = CoordinateRelocate(MinuteLength, angle); + + // lv_line_set_points(minute_body_world, minute_point_world, 2); + // } + + // if (sHour != hour || sMinute != minute) { + // auto angle = (worldHour * 30 + worldMinute / 2); + // if (settingsController.GetClockType() == Controllers::Settings::ClockType::H24) { + // angle /= 2; + // } + + // hour_point_world[0] = CoordinateRelocate(0, angle); + // hour_point_world[1] = CoordinateRelocate(HourLength, angle); + + // lv_line_set_points(hour_body_world, hour_point_world, 2); + // } +// } + +void WatchFaceMine::UpdateClock() { + uint8_t hour = dateTimeController.Hours(); + uint8_t minute = dateTimeController.Minutes(); + uint8_t second = dateTimeController.Seconds(); + + clockType = settingsController.GetClockType(); + bool clockTypeUpdated = clockType.IsUpdated(); + if (clockTypeUpdated) { + DrawFace(); + } + + // worldClockOffset = dateTimeController.worldOffset(0); + // if (dateTimeController.isWorldTimeEnabled(0) && (worldClockOffset.IsUpdated() || clockTypeUpdated || sHour != hour || sMinute != minute)) { + // DrawOffsetHands(sHour, sMinute, hour, minute); + // } + + if (sSecond != second) { + sSecond = second; + auto const angle = second * 6; + + second_point[0] = CoordinateRelocate(0, angle); + second_point[1] = CoordinateRelocate(SecondLength, angle); + lv_line_set_points(second_body, second_point, 2); + } + + if (sMinute != minute) { + auto const angle = minute * 6; + minute_point[0] = CoordinateRelocate(0, angle); + minute_point[1] = CoordinateRelocate(MinuteLength, angle); + + lv_line_set_points(minute_body, minute_point, 2); + } + + if (sHour != hour || sMinute != minute) { + sHour = hour; + sMinute = minute; + + auto angle = (hour * 30 + minute / 2); + if (settingsController.GetClockType() == Controllers::Settings::ClockType::H24) { + angle /= 2; + } + + hour_point[0] = CoordinateRelocate(0, angle); + hour_point[1] = CoordinateRelocate(HourLength, angle); + + lv_line_set_points(hour_body, hour_point, 2); + } +} + +void WatchFaceMine::DrawWeather() { + currentWeather = weatherService.Current(); + if (currentWeather.IsUpdated()) { + auto optCurrentWeather = currentWeather.Get(); + if (optCurrentWeather) { + int16_t temp = optCurrentWeather->temperature.Celsius(); + std::string hawaii = decimalToDozenal(1.8 * temp - 36); + lv_label_set_text_fmt(temperature, "%s", hawaii.c_str()); + lv_label_set_text(weatherIcon, Symbols::GetSymbol(optCurrentWeather->iconId)); + } else { + lv_label_set_text(temperature, "--"); + lv_label_set_text(weatherIcon, Symbols::ban); + } + lv_obj_realign(temperature); + lv_obj_realign(weatherIcon); + } +} + +void WatchFaceMine::SetBatteryIcon() { + auto batteryPercent = batteryPercentRemaining.Get(); + batteryIcon.SetBatteryPercentage(batteryPercent); +} + +void WatchFaceMine::Refresh() { + isCharging = batteryController.IsCharging(); + if (isCharging.IsUpdated()) { + if (isCharging.Get()) { + lv_obj_set_hidden(batteryIcon.GetObject(), true); + lv_obj_set_hidden(plugIcon, false); + } else { + lv_obj_set_hidden(batteryIcon.GetObject(), false); + lv_obj_set_hidden(plugIcon, true); + SetBatteryIcon(); + } + } + if (!isCharging.Get()) { + batteryPercentRemaining = batteryController.PercentRemaining(); + if (batteryPercentRemaining.IsUpdated()) { + SetBatteryIcon(); + } + } + + notificationState = notificationManager.AreNewNotificationsAvailable(); + + if (notificationState.IsUpdated()) { + lv_label_set_text_static(notificationIcon, NotificationIcon::GetIcon(notificationState.Get())); + } + + bluetoothState = bleController.IsConnected(); + + if (bluetoothState.IsUpdated()) { + lv_label_set_text_static(bluetoothIcon, BleIcon::GetIcon(bluetoothState.Get())); + } + + currentDateTime = dateTimeController.CurrentDateTime(); + + if (currentDateTime.IsUpdated()) { + Pinetime::Controllers::DateTime::Months month = dateTimeController.Month(); + uint8_t day = dateTimeController.Day(); + Pinetime::Controllers::DateTime::Days dayOfWeek = dateTimeController.DayOfWeek(); + uint16_t year = dateTimeController.Year(); + + UpdateClock(); + + if ((month != currentMonth) || (dayOfWeek != currentDayOfWeek) || (day != currentDay)) { + std::string fixedDate = gregorianToFixed(day, month, year); + + lv_label_set_text_fmt(label_date_greg, "%s\n%02i", dateTimeController.DayOfWeekShortToString(), day); + lv_label_set_text_fmt(label_date_world, "%s", fixedDate.c_str()); + + currentMonth = month; + currentDayOfWeek = dayOfWeek; + currentDay = day; + } + } + + stepCount = motionController.NbSteps(); + stepGoal = settingsController.GetStepsGoal(); + + if (stepCount.IsUpdated() || stepGoal.IsUpdated()) { + steps_point[0] = {.x = 0, .y = 230}; + steps_point[1] = {.x = 240, .y = 230}; + + steps_point_trace[0] = {.x = 0, .y = 230}; + steps_point_trace[1] = {.x = static_cast<int16_t>(fmin(stepCount.Get() * 240 / stepGoal.Get(), 240)), .y = 230}; + + lv_line_set_points(steps_body, steps_point, 2); + lv_line_set_points(steps_body_trace, steps_point_trace, 2); + if (stepCount.Get() >= stepGoal.Get()) { + lv_obj_add_style(steps_body_trace, LV_LINE_PART_MAIN, &steps_line_style_full); + } else { + lv_obj_add_style(steps_body_trace, LV_LINE_PART_MAIN, &steps_line_style_trace); + } + } + + // DrawWeather(); +} + +std::string WatchFaceMine::gregorianToFixed(uint8_t day, Pinetime::Controllers::DateTime::Months month, uint16_t year) { + bool isLeap = isLeapYear(year); + uint16_t ordinal = gregorianToOrdinal(day, month, isLeap); + return ordinalToFixed(ordinal, isLeap); +} + +bool WatchFaceMine::isLeapYear(uint16_t year) { + bool by4 = (year % 4) == 0; + bool by100 = (year % 100) == 0; + bool by400 = (year % 400) == 0; + return by4 && (!by100 || by400); +} + +uint16_t WatchFaceMine::gregorianToOrdinal(uint8_t dayOfMonth, Pinetime::Controllers::DateTime::Months month, bool isLeap) { + if (month == Pinetime::Controllers::DateTime::Months::Unknown) { + return 0; + } + + int februaryDays = isLeap? 29 : 28; + int monthDays[12] = {31, februaryDays, 31, + 30, 31, 30, + 31, 31, 30, + 31, 30, 31}; + int ordinal = 0; + for (auto m = Pinetime::Controllers::DateTime::Months::January; + m < month; + m=static_cast<Pinetime::Controllers::DateTime::Months>(static_cast<uint8_t>(m)+1)) { + ordinal += monthDays[static_cast<uint8_t>(m)-1]; + } + ordinal += dayOfMonth; + return ordinal; +} + +std::string WatchFaceMine::ordinalToFixed(uint16_t ordinal, bool isLeap) { + if (ordinal == 0) { + return "--"; + } + + ordinal -= 1; + if (ordinal == 196 and isLeap) { + return "LD"; + } + if (ordinal >196 and isLeap) { + ordinal -= 1; + } + if (ordinal == 364) { + return "OY"; + } + + uint8_t moon = ordinal / 28; + uint8_t sun = ordinal % 28; + uint8_t week = sun / 7; + sun %= 7; + + char const *moons1[13] = {"\xCF\x80", "\xCE\xB4", "\xCF\x84", + "\xCF\x80", "\xCE\xB4", "\xCF\x84", + "\xCE\xBC", + "\xCF\x80", "\xCE\xB4", "\xCF\x84", + "\xCF\x80", "\xCE\xB4", "\xCF\x84"}; + char const moons2[13] = {'h', 'h', 'h', + 'v', 'v', 'v', + 's', + 'e', 'e', 'e', + 'u', 'u', 'u'}; + + char buf[10]; + sprintf(buf, "\n%s%c %d%d", moons1[moon], moons2[moon], week, sun); + + return std::string(buf); +} + +std::string WatchFaceMine::decimalToDozenal(int16_t decimal) { + std::string result = ""; + char sign = ''; + if (decimal < 0) { + sign = '-'; + decimal = -decimal; + } + uint16_t p12, rem; + while (true) { + p12 = decimal / 12; + rem = decimal % 12; + char ch; + if (rem == 11) { + ch = 'E'; + } else if (rem == 10) { + ch = 'X'; + } else { + ch = '0' + rem; + } + result = ch + result; + + if (p12 == 0){ + break; + } + } + + return sign + result; +} + diff --git a/src/displayapp/screens/WatchFaceMine.h b/src/displayapp/screens/WatchFaceMine.h new file mode 100644 index 0000000000000000000000000000000000000000..34170882eff02e869833556d2df07fc490264e77 --- /dev/null +++ b/src/displayapp/screens/WatchFaceMine.h @@ -0,0 +1,152 @@ +#pragma once + +#include <lvgl/src/lv_core/lv_obj.h> +#include <cstdint> +#include <memory> +#include "displayapp/screens/Screen.h" +#include "components/datetime/DateTimeController.h" +#include "components/battery/BatteryController.h" +#include "components/ble/BleController.h" +#include "components/ble/NotificationManager.h" +#include "components/ble/SimpleWeatherService.h" +#include "displayapp/screens/BatteryIcon.h" +#include "utility/DirtyValue.h" + +namespace Pinetime { + namespace Controllers { + class Settings; + class Battery; + class Ble; + class NotificationManager; + class MotionController; + } + namespace Applications { + namespace Screens { + + class WatchFaceMine : public Screen { + public: + WatchFaceMine(Controllers::DateTime& dateTimeController, + const Controllers::Battery& batteryController, + const Controllers::Ble& bleController, + Controllers::NotificationManager& notificationManager, + Controllers::Settings& settingsController, + Controllers::MotionController& motionController, + Controllers::SimpleWeatherService& weather); + + ~WatchFaceMine() override; + + void Refresh() override; + + private: + uint8_t sHour, sMinute, sSecond; + + Pinetime::Controllers::DateTime::Months currentMonth = Pinetime::Controllers::DateTime::Months::Unknown; + Pinetime::Controllers::DateTime::Days currentDayOfWeek = Pinetime::Controllers::DateTime::Days::Unknown; + uint8_t currentDay = 0; + + Utility::DirtyValue<uint8_t> batteryPercentRemaining {0}; + Utility::DirtyValue<bool> isCharging {}; + Utility::DirtyValue<std::chrono::time_point<std::chrono::system_clock, std::chrono::nanoseconds>> currentDateTime; + Utility::DirtyValue<bool> notificationState {false}; + Utility::DirtyValue<bool> bluetoothState {false}; + + Utility::DirtyValue<Pinetime::Controllers::Settings::ClockType> clockType; + + Utility::DirtyValue<uint32_t> stepCount {}; + Utility::DirtyValue<uint32_t> stepGoal {}; + + Utility::DirtyValue<int8_t> worldClockOffset {}; + + Utility::DirtyValue<std::optional<Pinetime::Controllers::SimpleWeatherService::CurrentWeather>> currentWeather {}; + + lv_obj_t* tick[24]; + lv_point_t tick_point[24][2]; + + lv_obj_t* hour_body; + lv_obj_t* hour_body_world; + lv_obj_t* minute_body; + lv_obj_t* minute_body_world; + lv_obj_t* second_body; + + lv_point_t hour_point[2]; + lv_point_t hour_point_world[2]; + lv_point_t minute_point[2]; + lv_point_t minute_point_world[2]; + lv_point_t second_point[2]; + + lv_style_t tick_style; + lv_style_t tick_style_zero; + lv_style_t tick_style_six; + + lv_style_t hour_line_style; + lv_style_t hour_line_style_world; + lv_style_t minute_line_style; + lv_style_t minute_line_style_world; + lv_style_t second_line_style; + + lv_obj_t* label_date_greg; + lv_obj_t* label_date_world; + lv_obj_t* plugIcon; + lv_obj_t* notificationIcon; + lv_obj_t* bluetoothIcon; + lv_obj_t* weatherIcon; + lv_obj_t* temperature; + + Utility::DirtyValue<int16_t> nowTemp {}; + int16_t clouds = 0; + int16_t precip = 0; + + lv_obj_t* steps_body; + lv_obj_t* steps_body_trace; + lv_point_t steps_point[2]; + lv_point_t steps_point_trace[2]; + lv_style_t steps_line_style; + lv_style_t steps_line_style_trace; + lv_style_t steps_line_style_full; + + BatteryIcon batteryIcon{true}; + + Controllers::DateTime& dateTimeController; + const Controllers::Battery& batteryController; + const Controllers::Ble& bleController; + Controllers::NotificationManager& notificationManager; + Controllers::Settings& settingsController; + Controllers::MotionController& motionController; + Controllers::SimpleWeatherService& weatherService; + + void DrawFace(); + void DrawOffsetHands(uint8_t, uint8_t, uint8_t, uint8_t); + void UpdateClock(); + void SetBatteryIcon(); + void DrawWeather(); + std::string gregorianToFixed(uint8_t, Pinetime::Controllers::DateTime::Months, uint16_t); + bool isLeapYear(uint16_t); + uint16_t gregorianToOrdinal(uint8_t, Pinetime::Controllers::DateTime::Months, bool); + std::string ordinalToFixed(uint16_t, bool); + std::string decimalToDozenal(int16_t); + + lv_task_t* taskRefresh; + }; + } + + template <> + struct WatchFaceTraits<WatchFace::Mine> { + static constexpr WatchFace watchFace = WatchFace::Mine; + static constexpr const char* name = "mine"; + + static Screens::Screen* Create(AppControllers& controllers) { + return new Screens::WatchFaceMine(controllers.dateTimeController, + controllers.batteryController, + controllers.bleController, + controllers.notificationManager, + controllers.settingsController, + controllers.motionController, + *controllers.weatherController); + }; + + static bool IsAvailable(Pinetime::Controllers::FS& /*filesystem*/) { + return true; + } + }; + } +}