Wesley Aptekar-Cassels dcf167a8bf Add solstice_face.
2023-11-06 18:01:44 -05:00

237 lines
8.0 KiB
C

/*
* MIT License
*
* Copyright (c) 2023 Wesley Aptekar-Cassels
*
* 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 <string.h>
#include <math.h>
#include "watch_utility.h"
#include "solstice_face.h"
// Find solstice or equinox time in JDE for a given year, via method from Meeus Ch 27
static double calculate_solstice_equinox(uint16_t year, uint8_t k) {
double Y = ((double)year - 2000) / 1000;
double approx_terms[4][5] = {
{2451623.80984, 365242.37404, 0.05169, -0.00411, -0.00057}, // March equinox
{2451716.56767, 365241.62603, 0.00325, 0.00888, -0.00030}, // June solstice
{2451810.21715, 365242.01767, -0.11575, 0.00337, 0.00078}, // September equinox
{2451900.05952, 365242.74049, -0.06223, -0.00823, 0.00032}, // December solstice
};
double JDE0 = approx_terms[k][0] + Y * (approx_terms[k][1] + Y * (approx_terms[k][2] + Y * (approx_terms[k][3] + Y * approx_terms[k][4])));
double T = (JDE0 - 2451545.0) / 36525;
double W = 35999.373 * T - 2.47;
double dlambda = 1 + (0.0334 * cos(W * M_PI / 180.0)) + (0.0007 * cos(2 * W * M_PI / 180.0));
double correction_terms[25][3] = {
{485,324.96,1934.136},
{203,337.23,32964.467},
{199,342.08,20.186},
{182,27.85,445267.112},
{156,73.14,45036.886},
{136,171.52,22518.443},
{77,222.54,65928.934},
{74,296.72,3034.906},
{70,243.58,9037.513},
{58,119.81,33718.147},
{52,297.17,150.678},
{50,21.02,2281.226},
{45,247.54,29929.562},
{44,325.15,31555.956},
{29,60.93,4443.417},
{18,155.12,67555.328},
{17,288.79,4562.452},
{16,198.04,62894.029},
{14,199.76,31436.921},
{12,95.39,14577.848},
{12,287.11,31931.756},
{12,320.81,34777.259},
{9,227.73,1222.114},
{8,15.45,16859.074},
};
double S = 0;
for (int i = 0; i < 25; i++) {
S += correction_terms[i][0] * cos((correction_terms[i][1] + correction_terms[i][2] * T) * M_PI / 180.0);
}
double JDE = JDE0 + (0.00001 * S) / dlambda;
return JDE;
}
// Convert JDE to Gergorian datetime as per Meeus Ch 7
static watch_date_time jde_to_date_time(double JDE) {
double tmp = JDE + 0.5;
double Z = floor(tmp);
double F = fmod(tmp, 1);
double A;
if (Z < 2299161) {
A = Z;
} else {
double alpha = floor((Z - 1867216.25) / 36524.25);
A = Z + 1 + alpha - floor(alpha / 4);
}
double B = A + 1524;
double C = floor((B - 122.1) / 365.25);
double D = floor(365.25 * C);
double E = floor((B - D) / 30.6001);
double day = B - D - floor(30.6001 * E) + F;
double month;
if (E < 14) {
month = E - 1;
} else {
month = E - 13;
}
double year;
if (month > 2) {
year = C - 4716;
} else {
year = C - 4715;
}
double hours = fmod(day, 1) * 24;
double minutes = fmod(hours, 1) * 60;
double seconds = fmod(minutes, 1) * 60;
watch_date_time result = {.unit = {
floor(seconds),
floor(minutes),
floor(hours),
floor(day),
floor(month),
floor(year - 2020)
}};
return result;
}
static void calculate_datetimes(solstice_state_t *state, movement_settings_t *settings) {
for (int i = 0; i < 4; i++) {
// TODO: handle DST changes
state->datetimes[i] = jde_to_date_time(calculate_solstice_equinox(2020 + state->year, i) + (movement_timezone_offsets[settings->bit.time_zone] / (60.0*24.0)));
}
}
void solstice_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(solstice_state_t));
solstice_state_t *state = (solstice_state_t *)*context_ptr;
watch_date_time now = watch_rtc_get_date_time();
state->year = now.unit.year;
state->index = 0;
calculate_datetimes(state, settings);
uint32_t now_unix = watch_utility_date_time_to_unix_time(now, 0);
for (int i = 0; i < 4; i++) {
if (state->index == 0 && watch_utility_date_time_to_unix_time(state->datetimes[i], 0) > now_unix) {
state->index = i;
}
}
}
}
void solstice_face_activate(movement_settings_t *settings, void *context) {
(void) settings;
(void) context;
}
static void show_main_screen(solstice_state_t *state) {
char buf[11];
watch_date_time date_time = state->datetimes[state->index];
sprintf(buf, " %2d %2d%02d", date_time.unit.year + 20, date_time.unit.month, date_time.unit.day);
watch_display_string(buf, 0);
}
static void show_date_time(movement_settings_t *settings, solstice_state_t *state) {
char buf[11];
watch_date_time date_time = state->datetimes[state->index];
if (!settings->bit.clock_mode_24h) {
if (date_time.unit.hour < 12) {
watch_clear_indicator(WATCH_INDICATOR_PM);
} else {
watch_set_indicator(WATCH_INDICATOR_PM);
}
date_time.unit.hour %= 12;
if (date_time.unit.hour == 0) date_time.unit.hour = 12;
}
sprintf(buf, "%s%2d%2d%02d%02d", watch_utility_get_weekday(date_time), date_time.unit.day, date_time.unit.hour, date_time.unit.minute, date_time.unit.second);
watch_set_colon();
watch_display_string(buf, 0);
}
bool solstice_face_loop(movement_event_t event, movement_settings_t *settings, void *context) {
solstice_state_t *state = (solstice_state_t *)context;
switch (event.event_type) {
case EVENT_ALARM_LONG_PRESS:
show_date_time(settings, state);
break;
case EVENT_LIGHT_BUTTON_UP:
if (state->index == 0) {
if (state->year == 0) {
break;
}
state->year--;
state->index = 3;
calculate_datetimes(state, settings);
} else {
state->index--;
}
show_main_screen(state);
break;
case EVENT_ALARM_BUTTON_UP:
state->index++;
if (state->index > 3) {
if (state->year == 83) {
break;
}
state->year++;
state->index = 0;
calculate_datetimes(state, settings);
}
show_main_screen(state);
break;
case EVENT_ALARM_LONG_UP:
watch_clear_colon();
watch_clear_indicator(WATCH_INDICATOR_PM);
show_main_screen(state);
break;
case EVENT_ACTIVATE:
show_main_screen(state);
break;
case EVENT_TIMEOUT:
movement_move_to_face(0);
break;
default:
return movement_default_loop_handler(event, settings);
}
return true;
}
void solstice_face_resign(movement_settings_t *settings, void *context) {
(void) settings;
(void) context;
}