summaryrefslogtreecommitdiffstats
path: root/movement/watch_faces/thermistor/thermistor_logging_face.c
blob: 0d456785e49314539b57c69b256b1c33c1af3296 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
#include <stdlib.h>
#include <string.h>
#include "thermistor_logging_face.h"
#include "thermistor_driver.h"
#include "watch.h"

static void _thermistor_logging_face_log_data(thermistor_logger_state_t *logger_state) {
    thermistor_driver_enable();
    watch_date_time date_time = watch_rtc_get_date_time();
    size_t pos = logger_state->data_points % THERMISTOR_LOGGING_NUM_DATA_POINTS;

    logger_state->data[pos].timestamp.reg = date_time.reg;
    logger_state->data[pos].temperature_c = thermistor_driver_get_temperature();
    logger_state->data_points++;

    thermistor_driver_disable();
}

static void _thermistor_logging_face_update_display(thermistor_logger_state_t *logger_state, bool in_fahrenheit, bool clock_mode_24h) {
    int8_t pos = (logger_state->data_points - 1 - logger_state->display_index) % THERMISTOR_LOGGING_NUM_DATA_POINTS;
    char buf[14];

    watch_clear_indicator(WATCH_INDICATOR_24H);
    watch_clear_indicator(WATCH_INDICATOR_PM);
    watch_clear_colon();

    if (pos < 0) {
        sprintf(buf, "TL%2dno dat", logger_state->display_index);
    } else if (logger_state->ts_ticks) {
        watch_date_time date_time = logger_state->data[pos].timestamp;
        watch_set_colon();
        if (clock_mode_24h) {
            watch_set_indicator(WATCH_INDICATOR_24H);
        } else {
            if (date_time.unit.hour > 11) watch_set_indicator(WATCH_INDICATOR_PM);
            date_time.unit.hour %= 12;
            if (date_time.unit.hour == 0) date_time.unit.hour = 12;
        }
        sprintf(buf, "AT%2d%2d%02d%02d", date_time.unit.day, date_time.unit.hour, date_time.unit.minute, date_time.unit.second);
    } else {
        if (in_fahrenheit) {
            sprintf(buf, "TL%2d%4.1f#F", logger_state->display_index, logger_state->data[pos].temperature_c * 1.8 + 32.0);
        } else {
            sprintf(buf, "TL%2d%4.1f#C", logger_state->display_index, logger_state->data[pos].temperature_c);
        }
    }

    watch_display_string(buf, 0);
}

void thermistor_logging_face_setup(movement_settings_t *settings, void ** context_ptr) {
    (void) settings;
    if (*context_ptr == NULL) {
        *context_ptr = malloc(sizeof(thermistor_logger_state_t));
        memset(*context_ptr, 0, sizeof(thermistor_logger_state_t));
    }
}

void thermistor_logging_face_activate(movement_settings_t *settings, void *context) {
    (void) settings;
    thermistor_logger_state_t *logger_state = (thermistor_logger_state_t *)context;
    logger_state->display_index = 0;
    logger_state->ts_ticks = 0;
}

bool thermistor_logging_face_loop(movement_event_t event, movement_settings_t *settings, void *context) {
    thermistor_logger_state_t *logger_state = (thermistor_logger_state_t *)context;
    switch (event.event_type) {
        case EVENT_TIMEOUT:
            movement_move_to_face(0);
            break;
        case EVENT_MODE_BUTTON_UP:
            movement_move_to_next_face();
            break;
        case EVENT_LIGHT_LONG_PRESS:
            // light button shows the timestamp, but if you need the light, long press it.
            movement_illuminate_led();
            break;
        case EVENT_LIGHT_BUTTON_DOWN:
            logger_state->ts_ticks = 2;
            _thermistor_logging_face_update_display(logger_state, settings->bit.use_imperial_units, settings->bit.clock_mode_24h);
            break;
        case EVENT_ALARM_BUTTON_DOWN:
            logger_state->display_index = (logger_state->display_index + 1) % THERMISTOR_LOGGING_NUM_DATA_POINTS;
            logger_state->ts_ticks = 0;
            // fall through
        case EVENT_ACTIVATE:
            _thermistor_logging_face_update_display(logger_state, settings->bit.use_imperial_units, settings->bit.clock_mode_24h);
            break;
        case EVENT_TICK:
            if (logger_state->ts_ticks && --logger_state->ts_ticks == 0) {
                _thermistor_logging_face_update_display(logger_state, settings->bit.use_imperial_units, settings->bit.clock_mode_24h);
            }
            break;
        case EVENT_BACKGROUND_TASK:
            _thermistor_logging_face_log_data(logger_state);
            break;
        default:
            break;
    }

    return true;
}

void thermistor_logging_face_resign(movement_settings_t *settings, void *context) {
    (void) settings;
    (void) context;
}

bool thermistor_logging_face_wants_background_task(movement_settings_t *settings, void *context) {
    (void) settings;
    (void) context;
    // this will get called at the top of each minute, so all we check is if we're at the top of the hour as well.
    // if we are, we ask for a background task.
    return watch_rtc_get_date_time().unit.minute == 0;
}