Skip to content

Commit 89deb19

Browse files
authored
Merge branch 'main' into milli-g
2 parents 99cdb8e + ecf2f56 commit 89deb19

File tree

7 files changed

+120
-22
lines changed

7 files changed

+120
-22
lines changed

CMakeLists.txt

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@ cmake_minimum_required(VERSION 3.10)
22

33
set(CMAKE_BUILD_TYPE Release CACHE STRING "Choose Debug or Release")
44

5-
project(pinetime VERSION 1.13.0 LANGUAGES C CXX ASM)
5+
project(pinetime VERSION 1.14.0 LANGUAGES C CXX ASM)
66

77
set(CMAKE_C_STANDARD 99)
88
set(CMAKE_CXX_STANDARD 20)

doc/code/Apps.md

+44-5
Original file line numberDiff line numberDiff line change
@@ -35,18 +35,20 @@ that will call the method `Refresh()` periodically.
3535

3636
## App types
3737

38-
There are basically 2 types of applications : **system** apps and **user** apps.
38+
There are basically 3 types of applications : **system** apps and **user** apps and **watch faces**.
3939

4040
**System** applications are always built into InfiniTime, and InfiniTime cannot work properly without those apps.
41-
The watchfaces, settings, notifications and the application launcher are examples of such system applications.
41+
Settings, notifications and the application launcher are examples of such system applications.
4242

4343
**User** applications are optionally built into the firmware. They extend the functionalities of the system.
4444

45-
The distinction between **system** and **user** applications allows for more flexibility and customization.
46-
This allows to easily select which user applications must be built into the firmware
45+
**Watch faces** are very similar to the **user** apps, they are optional, but at least one must be built into the firmware.
46+
47+
The distinction between **system** apps, **user** apps and watch faces allows for more flexibility and customization.
48+
This allows to easily select which user applications and watch faces must be built into the firmware
4749
without overflowing the system memory.
4850

49-
## Apps initialization
51+
## Apps and watch faces initialization
5052

5153
Apps are created by `DisplayApp` in `DisplayApp::LoadScreen()`.
5254
This method simply call the creates an instance of the class that corresponds to the app specified in parameters.
@@ -55,6 +57,8 @@ The constructor of **system** apps is called directly. If the application is a *
5557
the corresponding `AppDescription` is first retrieved from `userApps`
5658
and then the function `create` is called to create an instance of the app.
5759

60+
Watch faces are handled in a very similar way as the **user** apps : they are created by `DisplayApp` in the method `DisplayApp::LoadScreen()` when the application type is `Apps::Clock`.
61+
5862
## User application selection at build time
5963

6064
The list of user applications is generated at build time by the `consteval` function `CreateAppDescriptions()`
@@ -85,6 +89,32 @@ struct AppTraits<Apps::Alarm> {
8589
This array `userApps` is used by `DisplayApp` to create the applications and the `AppLauncher`
8690
to list all available applications.
8791
92+
## Watch face selection at build time
93+
94+
The list of available watch faces is also generated at build time by the `consteval`
95+
function `CreateWatchFaceDescriptions()` in `UserApps.h` in the same way as the **user** apps.
96+
Watch faces must declare a `WatchFaceTraits` so that the corresponding `WatchFaceDescription` can be generated.
97+
Here is an example of `WatchFaceTraits`:
98+
```c++
99+
template <>
100+
struct WatchFaceTraits<WatchFace::Analog> {
101+
static constexpr WatchFace watchFace = WatchFace::Analog;
102+
static constexpr const char* name = "Analog face";
103+
104+
static Screens::Screen* Create(AppControllers& controllers) {
105+
return new Screens::WatchFaceAnalog(controllers.dateTimeController,
106+
controllers.batteryController,
107+
controllers.bleController,
108+
controllers.notificationManager,
109+
controllers.settingsController);
110+
};
111+
112+
static bool IsAvailable(Pinetime::Controllers::FS& /*filesystem*/) {
113+
return true;
114+
}
115+
};
116+
```
117+
88118
## Creating your own app
89119

90120
A minimal user app could look like this:
@@ -168,6 +198,15 @@ Ex : build the firmware with 3 user application : Alarm, Timer and MyApp (the ap
168198
$ cmake ... -DENABLE_USERAPPS="Apps::Alarm, Apps::Timer, Apps::MyApp" ...
169199
```
170200

201+
Similarly, the list of watch faces is also generated by CMake, so you need to add the variable `ENABLE_WATCHFACES` to the command line of CMake.
202+
It must be set with the comma separated list of watch faces that will be built into the firmware.
203+
204+
Ex: build the firmware with 3 watch faces : Analog, PineTimeStyle and Infineat:
205+
206+
```cmake
207+
$ cmake ... -DENABLE_WATCHFACES="WatchFace::Analog,WatchFace::PineTimeStyle,WatchFace::Infineat" ...
208+
```
209+
171210
You should now be able to [build](../buildAndProgram.md) the firmware
172211
and flash it to your PineTime. Yay!
173212

src/displayapp/DisplayApp.cpp

+3-2
Original file line numberDiff line numberDiff line change
@@ -489,10 +489,11 @@ void DisplayApp::LoadScreen(Apps app, DisplayApp::FullRefreshDirections directio
489489
currentScreen = std::make_unique<Screens::Settings>(this, settingsController);
490490
break;
491491
case Apps::SettingWatchFace: {
492-
std::array<Screens::CheckboxList::Item, UserWatchFaceTypes::Count> items;
492+
std::array<Screens::SettingWatchFace::Item, UserWatchFaceTypes::Count> items;
493493
int i = 0;
494494
for (const auto& userWatchFace : userWatchFaces) {
495-
items[i++] = Screens::CheckboxList::Item {userWatchFace.name, userWatchFace.isAvailable(controllers.filesystem)};
495+
items[i++] =
496+
Screens::SettingWatchFace::Item {userWatchFace.name, userWatchFace.watchFace, userWatchFace.isAvailable(controllers.filesystem)};
496497
}
497498
currentScreen = std::make_unique<Screens::SettingWatchFace>(this, std::move(items), settingsController, filesystem);
498499
} break;

src/displayapp/apps/Apps.h.in

+1-6
Original file line numberDiff line numberDiff line change
@@ -71,12 +71,7 @@ namespace Pinetime {
7171
static constexpr size_t Count = sizeof...(Ws);
7272
};
7373

74-
using UserWatchFaceTypes = WatchFaceTypeList<WatchFace::Digital,
75-
WatchFace::Analog,
76-
WatchFace::PineTimeStyle,
77-
WatchFace::Terminal,
78-
WatchFace::Infineat,
79-
WatchFace::CasioStyleG7710>;
74+
using UserWatchFaceTypes = WatchFaceTypeList<@WATCHFACE_TYPES@>;
8075

8176
static_assert(UserWatchFaceTypes::Count >= 1);
8277
}

src/displayapp/apps/CMakeLists.txt

+26-1
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,34 @@
11
if(DEFINED ENABLE_USERAPPS)
22
set(USERAPP_TYPES ${ENABLE_USERAPPS} CACHE STRING "List of user apps to build into the firmware")
33
else ()
4-
set(USERAPP_TYPES "Apps::Navigation, Apps::StopWatch, Apps::Alarm, Apps::Timer, Apps::Steps, Apps::Motion, Apps::HeartRate, Apps::Music, Apps::Paint, Apps::Paddle, Apps::Twos, Apps::Metronome" CACHE STRING "List of user apps to build into the firmware")
4+
set(DEFAULT_USER_APP_TYPES "Apps::StopWatch")
5+
set(DEFAULT_USER_APP_TYPES "${DEFAULT_USER_APP_TYPES}, Apps::Alarm")
6+
set(DEFAULT_USER_APP_TYPES "${DEFAULT_USER_APP_TYPES}, Apps::Timer")
7+
set(DEFAULT_USER_APP_TYPES "${DEFAULT_USER_APP_TYPES}, Apps::Steps")
8+
set(DEFAULT_USER_APP_TYPES "${DEFAULT_USER_APP_TYPES}, Apps::HeartRate")
9+
set(DEFAULT_USER_APP_TYPES "${DEFAULT_USER_APP_TYPES}, Apps::Music")
10+
set(DEFAULT_USER_APP_TYPES "${DEFAULT_USER_APP_TYPES}, Apps::Paint")
11+
set(DEFAULT_USER_APP_TYPES "${DEFAULT_USER_APP_TYPES}, Apps::Paddle")
12+
set(DEFAULT_USER_APP_TYPES "${DEFAULT_USER_APP_TYPES}, Apps::Twos")
13+
set(DEFAULT_USER_APP_TYPES "${DEFAULT_USER_APP_TYPES}, Apps::Metronome")
14+
set(DEFAULT_USER_APP_TYPES "${DEFAULT_USER_APP_TYPES}, Apps::Navigation")
15+
#set(DEFAULT_USER_APP_TYPES "${DEFAULT_USER_APP_TYPES}, Apps::Weather")
16+
#set(DEFAULT_USER_APP_TYPES "${DEFAULT_USER_APP_TYPES}, Apps::Motion")
17+
set(USERAPP_TYPES "${DEFAULT_USER_APP_TYPES}" CACHE STRING "List of user apps to build into the firmware")
518
endif ()
619

20+
if(DEFINED ENABLE_WATCHFACES)
21+
set(WATCHFACE_TYPES ${ENABLE_WATCHFACES} CACHE STRING "List of watch faces to build into the firmware")
22+
else()
23+
set(DEFAULT_WATCHFACE_TYPES "WatchFace::Digital")
24+
set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::Analog")
25+
set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::PineTimeStyle")
26+
set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::Terminal")
27+
set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::Infineat")
28+
set(DEFAULT_WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}, WatchFace::CasioStyleG7710")
29+
set(WATCHFACE_TYPES "${DEFAULT_WATCHFACE_TYPES}" CACHE STRING "List of watch faces to build into the firmware")
30+
endif()
31+
732
add_library(infinitime_apps INTERFACE)
833
target_sources(infinitime_apps INTERFACE "${CMAKE_CURRENT_BINARY_DIR}/Apps.h")
934
target_include_directories(infinitime_apps INTERFACE "${CMAKE_CURRENT_BINARY_DIR}/")

src/displayapp/screens/settings/SettingWatchFace.cpp

+37-5
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,37 @@ using namespace Pinetime::Applications::Screens;
99
constexpr const char* SettingWatchFace::title;
1010
constexpr const char* SettingWatchFace::symbol;
1111

12+
namespace {
13+
uint32_t IndexOf(const std::array<Pinetime::Applications::Screens::SettingWatchFace::Item,
14+
Pinetime::Applications::UserWatchFaceTypes::Count>& watchfaces,
15+
Pinetime::Applications::WatchFace watchface) {
16+
size_t index = 0;
17+
auto found = std::find_if(watchfaces.begin(),
18+
watchfaces.end(),
19+
[&index, &watchface](const Pinetime::Applications::Screens::SettingWatchFace::Item& item) {
20+
const bool result = item.watchface == watchface;
21+
if (!result) {
22+
index++;
23+
}
24+
return result;
25+
});
26+
if (found == watchfaces.end()) {
27+
index = 0;
28+
}
29+
30+
return index;
31+
}
32+
33+
Pinetime::Applications::WatchFace IndexToWatchFace(const std::array<Pinetime::Applications::Screens::SettingWatchFace::Item,
34+
Pinetime::Applications::UserWatchFaceTypes::Count>& watchfaces,
35+
size_t index) {
36+
if (index >= watchfaces.size()) {
37+
return watchfaces[0].watchface;
38+
}
39+
return watchfaces[index].watchface;
40+
}
41+
}
42+
1243
auto SettingWatchFace::CreateScreenList() const {
1344
std::array<std::function<std::unique_ptr<Screen>()>, nScreens> screens;
1445
for (size_t i = 0; i < screens.size(); i++) {
@@ -20,7 +51,7 @@ auto SettingWatchFace::CreateScreenList() const {
2051
}
2152

2253
SettingWatchFace::SettingWatchFace(Pinetime::Applications::DisplayApp* app,
23-
std::array<Screens::CheckboxList::Item, UserWatchFaceTypes::Count>&& watchfaceItems,
54+
std::array<Screens::SettingWatchFace::Item, UserWatchFaceTypes::Count>&& watchfaceItems,
2455
Pinetime::Controllers::Settings& settingsController,
2556
Pinetime::Controllers::FS& filesystem)
2657
: app {app},
@@ -44,7 +75,8 @@ std::unique_ptr<Screen> SettingWatchFace::CreateScreen(unsigned int screenNum) c
4475
if (i + (screenNum * settingsPerScreen) >= watchfaceItems.size()) {
4576
watchfacesOnThisScreen[i] = {"", false};
4677
} else {
47-
watchfacesOnThisScreen[i] = watchfaceItems[i + (screenNum * settingsPerScreen)];
78+
auto& item = watchfaceItems[i + (screenNum * settingsPerScreen)];
79+
watchfacesOnThisScreen[i] = Screens::CheckboxList::Item {item.name, item.enabled};
4880
}
4981
}
5082

@@ -53,9 +85,9 @@ std::unique_ptr<Screen> SettingWatchFace::CreateScreen(unsigned int screenNum) c
5385
nScreens,
5486
title,
5587
symbol,
56-
static_cast<uint32_t>(settingsController.GetWatchFace()),
57-
[&settings = settingsController](uint32_t index) {
58-
settings.SetWatchFace(static_cast<WatchFace>(index));
88+
static_cast<uint32_t>(IndexOf(watchfaceItems, settingsController.GetWatchFace())),
89+
[this, &settings = settingsController](uint32_t index) {
90+
settings.SetWatchFace(IndexToWatchFace(watchfaceItems, index));
5991
settings.SaveSettings();
6092
},
6193
watchfacesOnThisScreen);

src/displayapp/screens/settings/SettingWatchFace.h

+8-2
Original file line numberDiff line numberDiff line change
@@ -19,8 +19,14 @@ namespace Pinetime {
1919

2020
class SettingWatchFace : public Screen {
2121
public:
22+
struct Item {
23+
const char* name;
24+
WatchFace watchface;
25+
bool enabled;
26+
};
27+
2228
SettingWatchFace(DisplayApp* app,
23-
std::array<Screens::CheckboxList::Item, UserWatchFaceTypes::Count>&& watchfaceItems,
29+
std::array<Item, UserWatchFaceTypes::Count>&& watchfaceItems,
2430
Pinetime::Controllers::Settings& settingsController,
2531
Pinetime::Controllers::FS& filesystem);
2632
~SettingWatchFace() override;
@@ -33,7 +39,7 @@ namespace Pinetime {
3339
std::unique_ptr<Screen> CreateScreen(unsigned int screenNum) const;
3440

3541
static constexpr int settingsPerScreen = 4;
36-
std::array<Screens::CheckboxList::Item, UserWatchFaceTypes::Count> watchfaceItems;
42+
std::array<Item, UserWatchFaceTypes::Count> watchfaceItems;
3743
static constexpr int nScreens = UserWatchFaceTypes::Count > 0 ? (UserWatchFaceTypes ::Count - 1) / settingsPerScreen + 1 : 1;
3844

3945
Controllers::Settings& settingsController;

0 commit comments

Comments
 (0)