2025-07-06 11:26:09 -04:00

353 lines
13 KiB
C

/*
* MIT License
*
* Copyright (c) 2025 Joey Castillo
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
#include <stdlib.h>
#include "settings_face.h"
#include "watch.h"
static void clock_setting_display(uint8_t subsecond) {
watch_display_text_with_fallback(WATCH_POSITION_TOP, "CLOCK", "CL");
if (subsecond % 2) {
if (movement_clock_mode_24h()) watch_display_text(WATCH_POSITION_BOTTOM, "24h");
else watch_display_text(WATCH_POSITION_BOTTOM, "12h");
}
}
static void clock_setting_advance(void) {
movement_set_clock_mode_24h(((movement_clock_mode_24h() + 1) % MOVEMENT_NUM_CLOCK_MODES));
}
static void beep_setting_display(uint8_t subsecond) {
watch_display_text_with_fallback(WATCH_POSITION_TOP_LEFT, "BTN", "BT");
watch_display_text_with_fallback(WATCH_POSITION_BOTTOM, "beep ", " beep ");
if (subsecond % 2) {
if (movement_button_should_sound()) {
if (movement_button_volume() == WATCH_BUZZER_VOLUME_LOUD) {
// H for HIGH
watch_display_text(WATCH_POSITION_TOP_RIGHT, " H");
}
else {
// L for LOW
watch_display_text(WATCH_POSITION_TOP_RIGHT, " L");
}
} else {
// N for NONE
watch_display_text(WATCH_POSITION_TOP_RIGHT, " N");
}
}
}
static void beep_setting_advance(void) {
if (!movement_button_should_sound()) {
// was muted. make it soft.
movement_set_button_should_sound(true);
movement_set_button_volume(WATCH_BUZZER_VOLUME_SOFT);
beep_setting_display(1);
watch_buzzer_play_note_with_volume(BUZZER_NOTE_C7, 50, WATCH_BUZZER_VOLUME_SOFT);
} else if (movement_button_volume() == WATCH_BUZZER_VOLUME_SOFT) {
// was soft. make it loud.
movement_set_button_volume(WATCH_BUZZER_VOLUME_LOUD);
beep_setting_display(1);
watch_buzzer_play_note_with_volume(BUZZER_NOTE_C7, 50, WATCH_BUZZER_VOLUME_LOUD);
} else {
// was loud. make it silent.
movement_set_button_should_sound(false);
beep_setting_display(1);
}
}
static void timeout_setting_display(uint8_t subsecond) {
watch_display_text_with_fallback(WATCH_POSITION_TOP, "TMOUt", "TO");
if (subsecond % 2) {
switch (movement_get_fast_tick_timeout()) {
case 0:
watch_display_text(WATCH_POSITION_BOTTOM, "60 SeC");
break;
case 1:
watch_display_text(WATCH_POSITION_BOTTOM, "2 n&in");
break;
case 2:
watch_display_text(WATCH_POSITION_BOTTOM, "5 n&in");
break;
case 3:
watch_display_text(WATCH_POSITION_BOTTOM, "30n&in");
break;
}
}
}
static void timeout_setting_advance(void) {
movement_set_fast_tick_timeout((movement_get_fast_tick_timeout() + 1));
}
static void low_energy_setting_display(uint8_t subsecond) {
watch_display_text_with_fallback(WATCH_POSITION_TOP, "LoEne", "LE");
if (subsecond % 2) {
switch (movement_get_low_energy_timeout()) {
case 0:
watch_display_text(WATCH_POSITION_BOTTOM, " Never");
break;
case 1:
watch_display_text(WATCH_POSITION_BOTTOM, "10n&in");
break;
case 2:
watch_display_text(WATCH_POSITION_BOTTOM, "1 hour");
break;
case 3:
watch_display_text(WATCH_POSITION_BOTTOM, "2 hour");
break;
case 4:
watch_display_text(WATCH_POSITION_BOTTOM, "6 hour");
break;
case 5:
watch_display_text(WATCH_POSITION_BOTTOM, "12 hr");
break;
case 6:
watch_display_text(WATCH_POSITION_BOTTOM, " 1 day");
break;
case 7:
watch_display_text(WATCH_POSITION_BOTTOM, " 7 day");
break;
}
}
}
static void low_energy_setting_advance(void) {
movement_set_low_energy_timeout((movement_get_low_energy_timeout() + 1));
}
static void led_duration_setting_display(uint8_t subsecond) {
char buf[8];
watch_display_text_with_fallback(WATCH_POSITION_TOP_LEFT, "LED", "LT");
if (subsecond % 2) {
if (movement_get_backlight_dwell() == 0) {
watch_display_text(WATCH_POSITION_BOTTOM, "instnt");
} else if (movement_get_backlight_dwell() == 0b111) {
watch_display_text(WATCH_POSITION_BOTTOM, "no LEd");
} else {
sprintf(buf, " %1d SeC", (movement_get_backlight_dwell() * 2 - 1) % 10);
watch_display_text(WATCH_POSITION_BOTTOM, buf);
}
}
}
static void led_duration_setting_advance(void) {
movement_set_backlight_dwell(movement_get_backlight_dwell() + 1);
if (movement_get_backlight_dwell() > 3) {
// set all bits to disable the LED
movement_set_backlight_dwell(0b111);
}
}
static void red_led_setting_display(uint8_t subsecond) {
char buf[8];
movement_color_t color = movement_backlight_color();
watch_display_text_with_fallback(WATCH_POSITION_TOP_LEFT, "LED", "LT");
watch_display_text(WATCH_POSITION_BOTTOM, " red ");
if (subsecond % 2) {
sprintf(buf, "%2d", color.red);
watch_display_text(WATCH_POSITION_TOP_RIGHT, buf);
}
}
static void red_led_setting_advance(void) {
movement_color_t color = movement_backlight_color();
color.red++;
movement_set_backlight_color(color);
}
static void green_led_setting_display(uint8_t subsecond) {
char buf[8];
movement_color_t color = movement_backlight_color();
watch_display_text_with_fallback(WATCH_POSITION_TOP_LEFT, "LED", "LT");
watch_display_text(WATCH_POSITION_BOTTOM, " green");
if (subsecond % 2) {
sprintf(buf, "%2d", color.green);
watch_display_text(WATCH_POSITION_TOP_RIGHT, buf);
}
}
static void green_led_setting_advance(void) {
movement_color_t color = movement_backlight_color();
color.green++;
movement_set_backlight_color(color);
}
static void blue_led_setting_display(uint8_t subsecond) {
char buf[8];
movement_color_t color = movement_backlight_color();
watch_display_text_with_fallback(WATCH_POSITION_TOP_LEFT, "LED", "LT");
watch_display_text_with_fallback(WATCH_POSITION_BOTTOM, "blue ", " blue ");
if (subsecond % 2) {
sprintf(buf, "%2d", color.blue);
watch_display_text(WATCH_POSITION_TOP_RIGHT, buf);
}
}
static void blue_led_setting_advance(void) {
movement_color_t color = movement_backlight_color();
color.blue++;
movement_set_backlight_color(color);
}
static void git_hash_setting_display(uint8_t subsecond) {
(void) subsecond;
char buf[8];
// BUILD_GIT_HASH will already be truncated to 6 characters in the makefile, but this is to be safe.
sprintf(buf, "%.6s", BUILD_GIT_HASH);
watch_display_text_with_fallback(WATCH_POSITION_TOP, "Bu{d ", "bU");
watch_display_text(WATCH_POSITION_BOTTOM, buf);
}
static void git_hash_setting_advance(void) {
return;
}
void settings_face_setup(uint8_t watch_face_index, void ** context_ptr) {
(void) watch_face_index;
if (*context_ptr == NULL) {
*context_ptr = malloc(sizeof(settings_state_t));
settings_state_t *state = (settings_state_t *)*context_ptr;
int8_t current_setting = 0;
state->num_settings = 5; // baseline, without LED settings
#ifdef BUILD_GIT_HASH
state->num_settings++;
#endif
#ifdef WATCH_RED_TCC_CHANNEL
state->num_settings++;
#endif
#ifdef WATCH_GREEN_TCC_CHANNEL
state->num_settings++;
#endif
#ifdef WATCH_BLUE_TCC_CHANNEL
state->num_settings++;
#endif
state->settings_screens = malloc(state->num_settings * sizeof(settings_screen_t));
state->settings_screens[current_setting].display = clock_setting_display;
state->settings_screens[current_setting].advance = clock_setting_advance;
current_setting++;
state->settings_screens[current_setting].display = beep_setting_display;
state->settings_screens[current_setting].advance = beep_setting_advance;
current_setting++;
state->settings_screens[current_setting].display = timeout_setting_display;
state->settings_screens[current_setting].advance = timeout_setting_advance;
current_setting++;
#ifndef MOVEMENT_LOW_ENERGY_MODE_FORBIDDEN
state->settings_screens[current_setting].display = low_energy_setting_display;
state->settings_screens[current_setting].advance = low_energy_setting_advance;
current_setting++;
#endif
#ifdef BUILD_GIT_HASH
state->settings_screens[current_setting].display = git_hash_setting_display;
state->settings_screens[current_setting].advance = git_hash_setting_advance;
current_setting++;
#endif
state->settings_screens[current_setting].display = led_duration_setting_display;
state->settings_screens[current_setting].advance = led_duration_setting_advance;
current_setting++;
state->led_color_start = current_setting;
#ifdef WATCH_RED_TCC_CHANNEL
state->settings_screens[current_setting].display = red_led_setting_display;
state->settings_screens[current_setting].advance = red_led_setting_advance;
current_setting++;
#else
(void)red_led_setting_display;
(void)red_led_setting_advance;
#endif
#ifdef WATCH_GREEN_TCC_CHANNEL
state->settings_screens[current_setting].display = green_led_setting_display;
state->settings_screens[current_setting].advance = green_led_setting_advance;
current_setting++;
#else
(void)green_led_setting_display;
(void)green_led_setting_advance;
#endif
#ifdef WATCH_BLUE_TCC_CHANNEL
state->settings_screens[current_setting].display = blue_led_setting_display;
state->settings_screens[current_setting].advance = blue_led_setting_advance;
current_setting++;
#else
(void)blue_led_setting_display;
(void)blue_led_setting_advance;
#endif
}
}
void settings_face_activate(void *context) {
settings_state_t *state = (settings_state_t *)context;
state->current_page = 0;
movement_request_tick_frequency(4); // we need to manually blink some pixels
}
bool settings_face_loop(movement_event_t event, void *context) {
settings_state_t *state = (settings_state_t *)context;
switch (event.event_type) {
case EVENT_LIGHT_BUTTON_DOWN:
state->current_page = (state->current_page + 1) % state->num_settings;
// fall through
case EVENT_TICK:
case EVENT_ACTIVATE:
watch_clear_display();
state->settings_screens[state->current_page].display(event.subsecond);
break;
case EVENT_MODE_BUTTON_UP:
movement_force_led_off();
movement_move_to_next_face();
return false;
case EVENT_ALARM_BUTTON_UP:
state->settings_screens[state->current_page].advance();
break;
case EVENT_TIMEOUT:
movement_move_to_face(0);
break;
default:
return movement_default_loop_handler(event);
}
if (state->current_page >= state->led_color_start) {
movement_color_t color = movement_backlight_color();
// this bitwise math turns #000 into #000000, #111 into #111111, etc.
movement_force_led_on(color.red | color.red << 4,
color.green | color.green << 4,
color.blue | color.blue << 4);
return false;
} else {
movement_force_led_off();
return true;
}
}
void settings_face_resign(void *context) {
(void) context;
movement_force_led_off();
movement_store_settings();
}