new preferences face with support for RGB LED
This commit is contained in:
245
watch-faces/settings/preferences_face.c
Normal file
245
watch-faces/settings/preferences_face.c
Normal file
@@ -0,0 +1,245 @@
|
||||
/*
|
||||
* MIT License
|
||||
*
|
||||
* Copyright (c) 2022-2024 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 "preferences_face.h"
|
||||
#include "watch.h"
|
||||
|
||||
const char preferences_face_titles[PREFERENCES_PAGE_NUM_PREFERENCES][11] = {
|
||||
"CL ", // Clock: 12 or 24 hour
|
||||
"BT beep ", // Mode button: how loud should it beep?
|
||||
"TO ", // Timeout: how long before we snap back to the clock face?
|
||||
"LE ", // Low Energy mode: how long before it engages?
|
||||
"LT ", // Light: duration
|
||||
"LT red ", // Light: red component
|
||||
"LT green", // Light: green component
|
||||
"LT blue ", // Light: blue component (for watches with blue LED)
|
||||
};
|
||||
|
||||
void preferences_face_setup(movement_settings_t *settings, uint8_t watch_face_index, void ** context_ptr) {
|
||||
(void) settings;
|
||||
(void) watch_face_index;
|
||||
if (*context_ptr == NULL) {
|
||||
*context_ptr = malloc(sizeof(preferences_state_t));
|
||||
preferences_state_t *state = (preferences_state_t *)*context_ptr;
|
||||
for (int i = 0; i < PREFERENCES_PAGE_NUM_PREFERENCES; i++) {
|
||||
state->setting_enabled[i] = true;
|
||||
}
|
||||
#ifdef CLOCK_FACE_24H_ONLY
|
||||
state->setting_enabled[PREFERENCES_PAGE_CLOCK_MODE] = false;
|
||||
#endif
|
||||
#ifndef WATCH_RED_TCC_CHANNEL
|
||||
state->setting_enabled[PREFERENCES_PAGE_LED_RED] = false;
|
||||
#endif
|
||||
#ifndef WATCH_GREEN_TCC_CHANNEL
|
||||
state->setting_enabled[PREFERENCES_PAGE_LED_GREEN] = false;
|
||||
#endif
|
||||
#ifndef WATCH_BLUE_TCC_CHANNEL
|
||||
state->setting_enabled[PREFERENCES_PAGE_LED_BLUE] = false;
|
||||
#endif
|
||||
|
||||
}
|
||||
}
|
||||
|
||||
void preferences_face_activate(movement_settings_t *settings, void *context) {
|
||||
(void) settings;
|
||||
preferences_state_t *state = (preferences_state_t *)context;
|
||||
state->current_page = 0;
|
||||
movement_request_tick_frequency(4); // we need to manually blink some pixels
|
||||
}
|
||||
|
||||
bool preferences_face_loop(movement_event_t event, movement_settings_t *settings, void *context) {
|
||||
preferences_state_t *state = (preferences_state_t *)context;
|
||||
|
||||
switch (event.event_type) {
|
||||
case EVENT_TICK:
|
||||
case EVENT_ACTIVATE:
|
||||
// Do nothing; handled below.
|
||||
break;
|
||||
case EVENT_MODE_BUTTON_UP:
|
||||
watch_set_led_off();
|
||||
movement_move_to_next_face();
|
||||
return false;
|
||||
case EVENT_LIGHT_BUTTON_DOWN:
|
||||
state->current_page = (state->current_page + 1) % PREFERENCES_PAGE_NUM_PREFERENCES;
|
||||
while (!state->setting_enabled[state->current_page]) {
|
||||
state->current_page = (state->current_page + 1) % PREFERENCES_PAGE_NUM_PREFERENCES;
|
||||
}
|
||||
break;
|
||||
case EVENT_ALARM_BUTTON_UP:
|
||||
switch (state->current_page) {
|
||||
case PREFERENCES_PAGE_CLOCK_MODE:
|
||||
settings->bit.clock_mode_24h = !settings->bit.clock_mode_24h;
|
||||
break;
|
||||
case PREFERENCES_PAGE_BUTTON_SOUND:
|
||||
settings->bit.button_should_sound = !settings->bit.button_should_sound;
|
||||
break;
|
||||
case PREFERENCES_PAGE_TIMEOUT:
|
||||
settings->bit.to_interval = settings->bit.to_interval + 1;
|
||||
break;
|
||||
case PREFERENCES_PAGE_LOW_ENERGY:
|
||||
settings->bit.le_interval = settings->bit.le_interval + 1;
|
||||
break;
|
||||
case PREFERENCES_PAGE_LED_DURATION:
|
||||
settings->bit.led_duration = settings->bit.led_duration + 1;
|
||||
if (settings->bit.led_duration > 3) {
|
||||
// set all bits to disable the LED
|
||||
settings->bit.led_duration = 0b111;
|
||||
}
|
||||
break;
|
||||
case PREFERENCES_PAGE_LED_RED:
|
||||
settings->bit.led_red_color = settings->bit.led_red_color + 1;
|
||||
break;
|
||||
case PREFERENCES_PAGE_LED_GREEN:
|
||||
settings->bit.led_green_color = settings->bit.led_green_color + 1;
|
||||
break;
|
||||
case PREFERENCES_PAGE_LED_BLUE:
|
||||
settings->bit.led_blue_color = settings->bit.led_blue_color + 1;
|
||||
break;
|
||||
case PREFERENCES_PAGE_NUM_PREFERENCES:
|
||||
// nothing to do here, just silencing the warning
|
||||
break;
|
||||
}
|
||||
break;
|
||||
case EVENT_TIMEOUT:
|
||||
movement_move_to_face(0);
|
||||
break;
|
||||
default:
|
||||
return movement_default_loop_handler(event, settings);
|
||||
}
|
||||
|
||||
watch_display_text(WATCH_POSITION_FULL, (char *)preferences_face_titles[state->current_page]);
|
||||
watch_clear_all_indicators();
|
||||
watch_clear_colon();
|
||||
|
||||
// blink active setting on even-numbered quarter-seconds
|
||||
if (event.subsecond % 2) {
|
||||
char buf[8];
|
||||
switch (state->current_page) {
|
||||
case PREFERENCES_PAGE_CLOCK_MODE:
|
||||
if (settings->bit.clock_mode_24h) watch_display_text(WATCH_POSITION_BOTTOM, "24h");
|
||||
else watch_display_text(WATCH_POSITION_BOTTOM, "12h");
|
||||
break;
|
||||
case PREFERENCES_PAGE_BUTTON_SOUND:
|
||||
if (settings->bit.button_should_sound) {
|
||||
watch_display_text(WATCH_POSITION_TOP_RIGHT, " Y");
|
||||
} else {
|
||||
watch_display_text(WATCH_POSITION_TOP_RIGHT, " N");
|
||||
}
|
||||
break;
|
||||
case PREFERENCES_PAGE_TIMEOUT:
|
||||
switch (settings->bit.to_interval) {
|
||||
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;
|
||||
}
|
||||
break;
|
||||
case PREFERENCES_PAGE_LOW_ENERGY:
|
||||
switch (settings->bit.le_interval) {
|
||||
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;
|
||||
}
|
||||
break;
|
||||
case PREFERENCES_PAGE_LED_DURATION:
|
||||
if (settings->bit.led_duration == 0) {
|
||||
watch_display_text(WATCH_POSITION_BOTTOM, "instnt");
|
||||
} else if (settings->bit.led_duration == 0b111) {
|
||||
watch_display_text(WATCH_POSITION_BOTTOM, "no LEd");
|
||||
} else {
|
||||
sprintf(buf, " %1d SeC", settings->bit.led_duration * 2 - 1);
|
||||
watch_display_text(WATCH_POSITION_BOTTOM, buf);
|
||||
}
|
||||
break;
|
||||
case PREFERENCES_PAGE_LED_RED:
|
||||
sprintf(buf, "%2d", settings->bit.led_red_color);
|
||||
watch_display_text(WATCH_POSITION_TOP_RIGHT, buf);
|
||||
break;
|
||||
break;
|
||||
case PREFERENCES_PAGE_LED_GREEN:
|
||||
sprintf(buf, "%2d", settings->bit.led_green_color);
|
||||
watch_display_text(WATCH_POSITION_TOP_RIGHT, buf);
|
||||
break;
|
||||
break;
|
||||
case PREFERENCES_PAGE_LED_BLUE:
|
||||
sprintf(buf, "%2d", settings->bit.led_blue_color);
|
||||
watch_display_text(WATCH_POSITION_TOP_RIGHT, buf);
|
||||
break;
|
||||
case PREFERENCES_PAGE_NUM_PREFERENCES:
|
||||
// nothing to do here, just silencing the warning
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
// on LED color select screns, preview the color.
|
||||
if (state->current_page == PREFERENCES_PAGE_LED_RED ||
|
||||
state->current_page == PREFERENCES_PAGE_LED_GREEN ||
|
||||
state->current_page == PREFERENCES_PAGE_LED_BLUE) {
|
||||
watch_set_led_color_rgb(settings->bit.led_red_color ? (0xF | settings->bit.led_red_color << 4) : 0,
|
||||
settings->bit.led_green_color ? (0xF | settings->bit.led_green_color << 4) : 0,
|
||||
settings->bit.led_blue_color ? (0xF | settings->bit.led_blue_color << 4) : 0);
|
||||
// return false so the watch stays awake (needed for the PWM driver to function).
|
||||
return false;
|
||||
}
|
||||
|
||||
watch_set_led_off();
|
||||
return true;
|
||||
}
|
||||
|
||||
void preferences_face_resign(movement_settings_t *settings, void *context) {
|
||||
(void) settings;
|
||||
(void) context;
|
||||
watch_set_led_off();
|
||||
watch_store_backup_data(settings->reg, 0);
|
||||
}
|
||||
113
watch-faces/settings/preferences_face.h
Normal file
113
watch-faces/settings/preferences_face.h
Normal file
@@ -0,0 +1,113 @@
|
||||
/*
|
||||
* MIT License
|
||||
*
|
||||
* Copyright (c) 2022-2024 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.
|
||||
*/
|
||||
|
||||
#pragma once
|
||||
|
||||
/*
|
||||
* PREFERENCES face
|
||||
*
|
||||
* The Preferences watch face allows you to configure various options on your
|
||||
* Sensor Watch. Like all other screens, you advance the field you’re setting
|
||||
* with the Light button, and advance its value with the Alarm button. The
|
||||
* Preferences watch face labels each setting with a two-letter code on the
|
||||
* top row; the following list describes each setting and their options:
|
||||
*
|
||||
* CL - Clock mode.
|
||||
* This setting allows you to select a 12-or 24-hour clock display. All
|
||||
* watch faces that support displaying the time will respect this setting;
|
||||
* for example, both Simple Clock, World Clock and Sunrise/Sunset will
|
||||
* display the time in 24 hour format if the 24 hour clock is selected here.
|
||||
*
|
||||
* MO - Mode button.
|
||||
* This setting allows you to choose whether the Mode button should emit
|
||||
* a beep when pressed, and if so, how loud it should be. Options are
|
||||
* "-" (no beep), "L" (low volume) and "H" (high volume).
|
||||
*
|
||||
* MO - Mode button.
|
||||
* This setting allows you to choose whether the Mode button should emit
|
||||
* a beep when pressed, and if so, how loud it should be. Options are
|
||||
* "-" (no beep), "L" (low volume) and "H" (high volume).
|
||||
*
|
||||
* TO - Timeout.
|
||||
* Sets the time until screens that time out (like Settings and Time Set)
|
||||
* snap back to the first screen. 60 seconds is a good default for the
|
||||
* stock firmware, but if you choose a custom firmware with faces that
|
||||
* you’d like to keep on screen for longer, you can set that here.
|
||||
*
|
||||
* LE - Low Energy mode.
|
||||
* Sets the time until the watch enters its low energy sleep mode.
|
||||
* Options range from 1 hour to 7 days, or Never. The more often Sensor
|
||||
* Watch goes to sleep, the longer its battery will last — but you will
|
||||
* lose the seconds indicator while it is asleep. This setting allows
|
||||
* you to make a tradeoff between the device’s responsiveness and its
|
||||
* longevity.
|
||||
*
|
||||
* LT - Light.
|
||||
* This setting has three screens.
|
||||
* The first lets you choose how long the LED should stay lit when the
|
||||
* LIGHT button is pressed. Options are 1 second, 3 seconds and 5
|
||||
* seconds, or “No LED” to disable the LED entirely.
|
||||
* The second screen, titled “blu” or “grn”, sets the intensity of the
|
||||
* blue or green LED depending on the target Sensor Board hardware.
|
||||
* Values range from 0 (off) to 15 (full intensity).
|
||||
* The third screen, “red”, sets the intensity of the red LED, again
|
||||
* from 0 to 15.
|
||||
* On the last two screens, the LED remains on so that you can see the
|
||||
* effect of mixing the two LED colors. On the Special Edition boards,
|
||||
* you’ll have red, blue and a variety of shades of pink and purple to
|
||||
* experiment with!
|
||||
*/
|
||||
|
||||
#include "movement.h"
|
||||
|
||||
typedef enum {
|
||||
PREFERENCES_PAGE_CLOCK_MODE = 0,
|
||||
PREFERENCES_PAGE_BUTTON_SOUND,
|
||||
PREFERENCES_PAGE_TIMEOUT,
|
||||
PREFERENCES_PAGE_LOW_ENERGY,
|
||||
PREFERENCES_PAGE_LED_DURATION,
|
||||
PREFERENCES_PAGE_LED_RED,
|
||||
PREFERENCES_PAGE_LED_GREEN,
|
||||
PREFERENCES_PAGE_LED_BLUE,
|
||||
PREFERENCES_PAGE_NUM_PREFERENCES,
|
||||
} preferences_page_t;
|
||||
|
||||
|
||||
typedef struct {
|
||||
bool setting_enabled[PREFERENCES_PAGE_NUM_PREFERENCES];
|
||||
preferences_page_t current_page;
|
||||
} preferences_state_t;
|
||||
|
||||
void preferences_face_setup(movement_settings_t *settings, uint8_t watch_face_index, void ** context_ptr);
|
||||
void preferences_face_activate(movement_settings_t *settings, void *context);
|
||||
bool preferences_face_loop(movement_event_t event, movement_settings_t *settings, void *context);
|
||||
void preferences_face_resign(movement_settings_t *settings, void *context);
|
||||
|
||||
#define preferences_face ((const watch_face_t){ \
|
||||
preferences_face_setup, \
|
||||
preferences_face_activate, \
|
||||
preferences_face_loop, \
|
||||
preferences_face_resign, \
|
||||
NULL, \
|
||||
})
|
||||
Reference in New Issue
Block a user