summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--main/metric.patch261
1 files changed, 261 insertions, 0 deletions
diff --git a/main/metric.patch b/main/metric.patch
index 0ff3dc7..df2993a 100644
--- a/main/metric.patch
+++ b/main/metric.patch
@@ -34,3 +34,264 @@ index 3557110..17fe809 100644
// New includes go above this line.
#endif // MOVEMENT_FACES_H_
+diff --git a/movement/watch_faces/complication/metric_face.c b/movement/watch_faces/complication/metric_face.c
+new file mode 100644
+index 0000000..9eaafbf
+--- /dev/null
++++ b/movement/watch_faces/complication/metric_face.c
+@@ -0,0 +1,198 @@
++#include <stdlib.h>
++#include <string.h>
++#include <math.h>
++#include "watch.h"
++#include "watch_utility.h"
++#include "vsop87a_micro.h" // smaller size, less accurate
++#include "vsop87a_milli.h"
++#include "astrolib.h"
++#include "metric_face.h"
++
++#define RATIO 8
++
++#include "equinox.c"
++
++static const char *days[10] =
++ { "Pr", "Du", "Tr", "Qa", "Qi", "SH", "Sp", "Oc", "No", "De" };
++
++
++static uint32_t
++jd_to_timestamp (double tjd)
++{
++ tjd -= 2440587.5;
++ tjd *= 86400.;
++ return (uint32_t) floor (tjd);
++}
++
++static uint32_t
++timestamp_start_of_year (int y)
++{
++ double julian_equinox = autumnal_equinox (y);
++ julian_equinox += (2.3372305555 / 360); /* Offset of paris meridian from greenwich meridian */
++ return jd_to_timestamp (julian_equinox);
++}
++
++static int
++days_since_equinox (watch_date_time date_time)
++{
++ uint32_t now = watch_utility_date_time_to_unix_time (date_time, 0);
++ uint32_t start_of_year =
++ timestamp_start_of_year (date_time.unit.year + 2020);
++
++ start_of_year/=86400;
++ start_of_year*=86400;
++
++ if (start_of_year > now)
++ start_of_year =
++ timestamp_start_of_year (date_time.unit.year + 2020 - 1);
++
++ now -= start_of_year;
++
++ now /= 86400;
++
++ return (int) now;
++}
++
++static void
++calculate_metric_date (metric_state_t * state, watch_date_time date_time)
++{
++ int dse = days_since_equinox (date_time);
++
++ state->m_day = (dse % 30) + 1;
++ state->m_wday = (dse % 10);
++ state->c_day = date_time.unit.day;
++
++}
++
++
++static uint32_t
++date_time_to_ds (watch_date_time dt)
++{
++ uint32_t ret;
++ ret = dt.unit.hour;
++ ret *= 60;
++ ret += dt.unit.minute;
++ ret *= 60;
++ ret += dt.unit.second;
++ return ret;
++}
++
++static void
++_metric_face_sync (movement_settings_t * settings, metric_state_t * state)
++{
++ watch_date_time date_time = watch_rtc_get_date_time ();
++
++ (void) settings;
++
++ if (date_time.unit.day != state->c_day)
++ calculate_metric_date (state, date_time);
++
++ state->dt = date_time_to_ds (date_time) * RATIO;
++ state->resync_at = state->dt * (3600 * RATIO);
++ if (state->resync_at > (86400 * RATIO))
++ state->resync_at = (86400 * RATIO);
++
++}
++
++static void
++_metric_face_update (movement_event_t event, movement_settings_t * settings,
++ metric_state_t * state, int show_secs)
++{
++ uint64_t v;
++ char buf[14];
++
++ (void) event;
++
++ if (state->dt >= state->resync_at)
++ _metric_face_sync (settings, state);
++
++ v = state->dt;
++ v *= (uint64_t) 100000;
++ v /= (uint64_t) (RATIO * 86400);
++
++
++#if 0
++ {
++ watch_date_time date_time = watch_rtc_get_date_time ();
++ printf ("%d %d\n", state->dt, date_time_to_ds (date_time) * RATIO);
++ }
++#endif
++
++
++ sprintf (buf, "%s%2d %05d", days[state->m_wday], (int) state->m_day,
++ (int) v);
++
++ if (!show_secs)
++ {
++ buf[8] = ' ';
++ buf[9] = ' ';
++ }
++
++ watch_display_string (buf, 0);
++}
++
++void
++metric_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 (metric_state_t));
++ memset (*context_ptr, 0, sizeof (metric_state_t));
++ }
++}
++
++void
++metric_face_activate (movement_settings_t * settings, void *context)
++{
++ metric_state_t *state = (metric_state_t *) context;
++ (void) settings;
++ movement_request_tick_frequency (RATIO);
++ state->resync_at = 0;
++}
++
++bool
++metric_face_loop (movement_event_t event, movement_settings_t * settings,
++ void *context)
++{
++ (void) settings;
++ metric_state_t *state = (metric_state_t *) context;
++
++ switch (event.event_type)
++ {
++ case EVENT_ACTIVATE:
++ watch_set_colon ();
++ /*fall through */
++ case EVENT_TICK:
++ state->dt++;
++ _metric_face_update (event, settings, state, 1);
++ break;
++ case EVENT_LOW_ENERGY_UPDATE:
++ state->dt += RATIO * 60;
++ _metric_face_update (event, settings, state, 0);
++ break;
++ case EVENT_ALARM_BUTTON_UP:
++ break;
++ case EVENT_ALARM_LONG_PRESS:
++ break;
++ case EVENT_TIMEOUT:
++ state->resync_at = 0;
++ break;
++ default:
++ movement_default_loop_handler (event, settings);
++ break;
++ }
++
++ return true;
++}
++
++void
++metric_face_resign (movement_settings_t * settings, void *context)
++{
++ (void) settings;
++ metric_state_t *state = (metric_state_t *) context;
++ state->resync_at = 0;
++ state->c_day = 0;
++}
+diff --git a/movement/watch_faces/complication/metric_face.h b/movement/watch_faces/complication/metric_face.h
+new file mode 100644
+index 0000000..1968e38
+--- /dev/null
++++ b/movement/watch_faces/complication/metric_face.h
+@@ -0,0 +1,51 @@
++/*
++ * MIT License
++ *
++ * Copyright (c) 2022 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.
++ */
++
++#ifndef METRIC_FACE_H_
++#define METRIC_FACE_H_
++
++#include "movement.h"
++
++typedef struct {
++ uint32_t dt;
++ uint32_t resync_at;
++ uint32_t m_day;
++ uint32_t m_wday;
++ uint32_t c_day;
++} metric_state_t;
++
++void metric_face_setup(movement_settings_t *settings, uint8_t watch_face_index, void ** context_ptr);
++void metric_face_activate(movement_settings_t *settings, void *context);
++bool metric_face_loop(movement_event_t event, movement_settings_t *settings, void *context);
++void metric_face_resign(movement_settings_t *settings, void *context);
++
++#define metric_face ((const watch_face_t){ \
++ metric_face_setup, \
++ metric_face_activate, \
++ metric_face_loop, \
++ metric_face_resign, \
++ NULL, \
++})
++
++#endif // METRIC_FACE_H_