InfiniTime.git

commit 654da3827e0c8faaaa54cf4b5b02ae8f7df502ca

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;
+      }
+    };
+  }
+}