2021-09-28 13:06:37 +08:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2021-10-04 08:37:15 +08:00
|
|
|
#include <limits.h>
|
2021-09-28 13:06:37 +08:00
|
|
|
#include "watch.h"
|
|
|
|
#include "launcher.h"
|
|
|
|
#include "launcher_config.h"
|
|
|
|
|
|
|
|
LauncherState launcher_state;
|
|
|
|
void * widget_contexts[LAUNCHER_NUM_WIDGETS];
|
2021-10-04 08:37:15 +08:00
|
|
|
const int32_t launcher_screensaver_deadlines[8] = {INT_MAX, 3600, 7200, 21600, 43200, 86400, 172800, 604800};
|
2021-10-04 12:37:58 +08:00
|
|
|
LauncherEvent event;
|
2021-09-28 13:06:37 +08:00
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
void cb_mode_btn_interrupt();
|
|
|
|
void cb_light_btn_interrupt();
|
|
|
|
void cb_alarm_btn_interrupt();
|
|
|
|
void cb_alarm_btn_extwake();
|
|
|
|
void cb_alarm_fired();
|
2021-10-04 02:26:17 +08:00
|
|
|
void cb_tick();
|
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
static inline void _launcher_reset_screensaver_countdown() {
|
|
|
|
// for testing, make the timeout happen 60x faster.
|
|
|
|
launcher_state.screensaver_ticks = launcher_screensaver_deadlines[launcher_state.launcher_settings.bit.screensaver_interval] / 60;
|
|
|
|
}
|
|
|
|
|
2021-10-04 00:31:51 +08:00
|
|
|
void launcher_request_tick_frequency(uint8_t freq) {
|
|
|
|
watch_rtc_disable_all_periodic_callbacks();
|
2021-10-04 06:49:21 +08:00
|
|
|
launcher_state.subsecond = 0;
|
|
|
|
launcher_state.tick_frequency = freq;
|
2021-10-04 00:31:51 +08:00
|
|
|
watch_rtc_register_periodic_callback(cb_tick, freq);
|
|
|
|
}
|
|
|
|
|
|
|
|
void launcher_illuminate_led() {
|
|
|
|
launcher_state.light_ticks = 3;
|
|
|
|
}
|
|
|
|
|
2021-10-04 06:49:21 +08:00
|
|
|
void launcher_move_to_widget(uint8_t widget_index) {
|
2021-10-04 02:26:17 +08:00
|
|
|
launcher_state.widget_changed = true;
|
2021-10-04 21:51:49 +08:00
|
|
|
launcher_state.next_widget = widget_index;
|
2021-10-04 02:26:17 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 06:49:21 +08:00
|
|
|
void launcher_move_to_next_widget() {
|
|
|
|
launcher_move_to_widget((launcher_state.current_widget + 1) % LAUNCHER_NUM_WIDGETS);
|
2021-10-04 02:26:17 +08:00
|
|
|
}
|
2021-10-04 00:31:51 +08:00
|
|
|
|
2021-09-28 13:06:37 +08:00
|
|
|
void app_init() {
|
|
|
|
memset(&launcher_state, 0, sizeof(launcher_state));
|
2021-10-04 08:37:15 +08:00
|
|
|
|
2021-10-04 00:31:51 +08:00
|
|
|
launcher_state.launcher_settings.bit.led_green_color = 0xF;
|
2021-10-04 07:16:14 +08:00
|
|
|
launcher_state.launcher_settings.bit.button_should_sound = true;
|
2021-10-04 08:37:15 +08:00
|
|
|
launcher_state.launcher_settings.bit.screensaver_interval = 1;
|
|
|
|
_launcher_reset_screensaver_countdown();
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void app_wake_from_deep_sleep() {
|
|
|
|
// This app does not support deep sleep mode.
|
|
|
|
}
|
|
|
|
|
|
|
|
void app_setup() {
|
2021-10-06 03:55:34 +08:00
|
|
|
static bool is_first_launch = true;
|
|
|
|
|
|
|
|
if (is_first_launch) {
|
|
|
|
for(uint8_t i = 0; i < LAUNCHER_NUM_WIDGETS; i++) {
|
|
|
|
widget_contexts[i] = NULL;
|
|
|
|
is_first_launch = false;
|
|
|
|
}
|
|
|
|
}
|
2021-10-04 08:37:15 +08:00
|
|
|
if (launcher_state.screensaver_ticks != -1) {
|
|
|
|
watch_disable_extwake_interrupt(BTN_ALARM);
|
2021-10-04 11:39:52 +08:00
|
|
|
watch_rtc_disable_alarm_callback();
|
2021-09-28 13:06:37 +08:00
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
watch_enable_external_interrupts();
|
|
|
|
watch_register_interrupt_callback(BTN_MODE, cb_mode_btn_interrupt, INTERRUPT_TRIGGER_BOTH);
|
|
|
|
watch_register_interrupt_callback(BTN_LIGHT, cb_light_btn_interrupt, INTERRUPT_TRIGGER_BOTH);
|
|
|
|
watch_register_interrupt_callback(BTN_ALARM, cb_alarm_btn_interrupt, INTERRUPT_TRIGGER_BOTH);
|
2021-09-28 13:06:37 +08:00
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
watch_enable_buzzer();
|
|
|
|
watch_enable_leds();
|
|
|
|
watch_enable_display();
|
2021-09-28 13:06:37 +08:00
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
launcher_request_tick_frequency(1);
|
|
|
|
|
|
|
|
for(uint8_t i = 0; i < LAUNCHER_NUM_WIDGETS; i++) {
|
|
|
|
widgets[i].setup(&launcher_state.launcher_settings, &widget_contexts[i]);
|
|
|
|
}
|
2021-10-04 02:26:17 +08:00
|
|
|
|
2021-10-04 23:57:29 +08:00
|
|
|
widgets[0].activate(&launcher_state.launcher_settings, widget_contexts[0]);
|
2021-10-04 12:37:58 +08:00
|
|
|
event.value = 0;
|
|
|
|
event.bit.event_type = EVENT_ACTIVATE;
|
2021-10-04 08:37:15 +08:00
|
|
|
}
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void app_prepare_for_sleep() {
|
|
|
|
}
|
|
|
|
|
|
|
|
void app_wake_from_sleep() {
|
|
|
|
}
|
|
|
|
|
|
|
|
bool app_loop() {
|
|
|
|
if (launcher_state.widget_changed) {
|
2021-10-04 07:16:14 +08:00
|
|
|
if (launcher_state.launcher_settings.bit.button_should_sound) {
|
2021-10-04 21:51:49 +08:00
|
|
|
// low note for nonzero case, high note for return to widget 0
|
2021-10-05 03:27:29 +08:00
|
|
|
watch_buzzer_play_note(launcher_state.next_widget ? BUZZER_NOTE_C7 : BUZZER_NOTE_C8, 50);
|
2021-10-04 07:16:14 +08:00
|
|
|
}
|
2021-10-04 21:51:49 +08:00
|
|
|
widgets[launcher_state.current_widget].resign(&launcher_state.launcher_settings, widget_contexts[launcher_state.current_widget]);
|
|
|
|
launcher_state.current_widget = launcher_state.next_widget;
|
|
|
|
watch_clear_display();
|
|
|
|
widgets[launcher_state.current_widget].activate(&launcher_state.launcher_settings, widget_contexts[launcher_state.current_widget]);
|
|
|
|
event.value = 0;
|
|
|
|
event.bit.event_type = EVENT_ACTIVATE;
|
2021-09-28 13:06:37 +08:00
|
|
|
launcher_state.widget_changed = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the LED is off and should be on, turn it on
|
|
|
|
if (launcher_state.light_ticks > 0 && !launcher_state.led_on) {
|
2021-10-04 02:26:17 +08:00
|
|
|
watch_set_led_color(launcher_state.launcher_settings.bit.led_red_color ? (0xF | launcher_state.launcher_settings.bit.led_red_color << 4) : 0,
|
|
|
|
launcher_state.launcher_settings.bit.led_green_color ? (0xF | launcher_state.launcher_settings.bit.led_green_color << 4) : 0);
|
2021-09-28 13:06:37 +08:00
|
|
|
launcher_state.led_on = true;
|
2021-10-04 02:26:17 +08:00
|
|
|
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// if the LED is on and should be off, turn it off
|
|
|
|
if (launcher_state.led_on && launcher_state.light_ticks == 0) {
|
|
|
|
// unless the user is holding down the LIGHT button, in which case, give them more time.
|
|
|
|
if (watch_get_pin_level(BTN_LIGHT)) {
|
|
|
|
launcher_state.light_ticks = 3;
|
|
|
|
} else {
|
|
|
|
watch_set_led_off();
|
|
|
|
launcher_state.led_on = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
// if we have timed out of our screensaver countdown, enter screensaver mode.
|
|
|
|
if (launcher_state.screensaver_ticks == 0) {
|
|
|
|
launcher_state.screensaver_ticks = -1;
|
|
|
|
watch_date_time alarm_time;
|
|
|
|
alarm_time.reg = 0;
|
|
|
|
alarm_time.unit.second = 59; // after a match, the alarm fires at the next rising edge of CLK_RTC_CNT, so 59 seconds lets us update at :00
|
|
|
|
watch_rtc_register_alarm_callback(cb_alarm_fired, alarm_time, ALARM_MATCH_SS);
|
|
|
|
watch_register_extwake_callback(BTN_ALARM, cb_alarm_btn_extwake, true);
|
2021-10-04 12:37:58 +08:00
|
|
|
event.value = 0;
|
2021-10-04 23:09:11 +08:00
|
|
|
|
|
|
|
// this is a little mini-runloop.
|
|
|
|
// as long as screensaver_ticks is -1 (i.e. screensaver is active), we wake up here, update the screen, and go right back to sleep.
|
|
|
|
while (launcher_state.screensaver_ticks == -1) {
|
|
|
|
event.bit.event_type = EVENT_SCREENSAVER;
|
|
|
|
widgets[launcher_state.current_widget].loop(event, &launcher_state.launcher_settings, widget_contexts[launcher_state.current_widget]);
|
|
|
|
watch_enter_shallow_sleep(true);
|
|
|
|
}
|
|
|
|
// as soon as screensaver_ticks is reset by the extwake handler, we bail out of the loop and reactivate ourselves.
|
|
|
|
event.bit.event_type = EVENT_ACTIVATE;
|
|
|
|
// this is a hack tho: waking from shallow sleep, app_setup does get called, but it happens before we have reset our ticks.
|
|
|
|
// need to figure out if there's a better heuristic for determining how we woke up.
|
|
|
|
app_setup();
|
2021-10-04 08:37:15 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 23:57:57 +08:00
|
|
|
static bool can_sleep = true;
|
2021-10-04 21:51:49 +08:00
|
|
|
|
2021-10-04 12:37:58 +08:00
|
|
|
if (event.bit.event_type) {
|
|
|
|
event.bit.subsecond = launcher_state.subsecond;
|
2021-10-04 21:51:49 +08:00
|
|
|
can_sleep = widgets[launcher_state.current_widget].loop(event, &launcher_state.launcher_settings, widget_contexts[launcher_state.current_widget]);
|
2021-10-04 12:37:58 +08:00
|
|
|
event.value = 0;
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 21:51:49 +08:00
|
|
|
return can_sleep && !launcher_state.led_on;
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 12:37:58 +08:00
|
|
|
LauncherEventType _figure_out_button_event(LauncherEventType button_down_event, uint8_t *down_timestamp) {
|
2021-10-04 02:26:17 +08:00
|
|
|
watch_date_time date_time = watch_rtc_get_date_time();
|
|
|
|
if (*down_timestamp) {
|
|
|
|
uint8_t diff = ((61 + date_time.unit.second) - *down_timestamp) % 60;
|
|
|
|
*down_timestamp = 0;
|
|
|
|
if (diff > 1) return button_down_event + 2;
|
|
|
|
else return button_down_event + 1;
|
|
|
|
} else {
|
|
|
|
*down_timestamp = date_time.unit.second + 1;
|
|
|
|
return button_down_event;
|
|
|
|
}
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
void cb_light_btn_interrupt() {
|
|
|
|
_launcher_reset_screensaver_countdown();
|
2021-10-04 12:37:58 +08:00
|
|
|
event.bit.event_type = _figure_out_button_event(EVENT_LIGHT_BUTTON_DOWN, &launcher_state.light_down_timestamp);
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
void cb_mode_btn_interrupt() {
|
|
|
|
_launcher_reset_screensaver_countdown();
|
2021-10-04 12:37:58 +08:00
|
|
|
event.bit.event_type = _figure_out_button_event(EVENT_MODE_BUTTON_DOWN, &launcher_state.mode_down_timestamp);
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
void cb_alarm_btn_interrupt() {
|
|
|
|
_launcher_reset_screensaver_countdown();
|
2021-10-04 12:37:58 +08:00
|
|
|
event.bit.event_type = _figure_out_button_event(EVENT_ALARM_BUTTON_DOWN, &launcher_state.alarm_down_timestamp);
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|
|
|
|
|
2021-10-04 08:37:15 +08:00
|
|
|
void cb_alarm_btn_extwake() {
|
2021-10-04 23:09:11 +08:00
|
|
|
// wake up!
|
2021-10-04 08:37:15 +08:00
|
|
|
_launcher_reset_screensaver_countdown();
|
|
|
|
}
|
|
|
|
|
|
|
|
void cb_alarm_fired() {
|
2021-10-04 12:37:58 +08:00
|
|
|
event.bit.event_type = EVENT_SCREENSAVER;
|
2021-10-04 08:37:15 +08:00
|
|
|
}
|
|
|
|
|
2021-09-28 13:06:37 +08:00
|
|
|
void cb_tick() {
|
2021-10-04 12:37:58 +08:00
|
|
|
event.bit.event_type = EVENT_TICK;
|
2021-10-04 06:49:21 +08:00
|
|
|
watch_date_time date_time = watch_rtc_get_date_time();
|
|
|
|
if (date_time.unit.second != launcher_state.last_second) {
|
|
|
|
if (launcher_state.light_ticks) launcher_state.light_ticks--;
|
2021-10-04 08:37:15 +08:00
|
|
|
if (launcher_state.launcher_settings.bit.screensaver_interval && launcher_state.screensaver_ticks > 0) launcher_state.screensaver_ticks--;
|
2021-10-04 06:49:21 +08:00
|
|
|
|
|
|
|
launcher_state.last_second = date_time.unit.second;
|
|
|
|
launcher_state.subsecond = 0;
|
|
|
|
} else {
|
|
|
|
launcher_state.subsecond++;
|
|
|
|
}
|
2021-09-28 13:06:37 +08:00
|
|
|
}
|