194 lines
6.0 KiB
C

/*
* timesel.c - Time selector
* Copyright (C) 2024 Alexander Rosenberg
*
* This program is free software: you can redistribute it and/or modify it under
* the terms of the GNU General Public License as published by the Free Software
* Foundation, either version 3 of the License, or (at your option) any later
* version. See the LICENSE file for more information.
*/
#include "timesel.h"
#include <err.h>
void time_sel_init(TimeSelection *ts, DateSelection *ds, bool do_second) {
ts->ds = ds;
ts->do_second = do_second;
time_sel_reset(ts);
}
static void time_sel_clamp(TimeSelection *ts) {
if (ts->ds) {
if (ts->ds->end_time.local_year == ts->ds->year &&
ts->ds->end_time.local_month == ts->ds->month &&
ts->ds->end_time.local_day == ts->ds->day) {
if (ts->hour > ts->ds->end_time.local_hour) {
ts->hour = ts->ds->end_time.local_hour;
}
if (ts->hour == ts->ds->end_time.local_hour &&
ts->minute > ts->ds->end_time.local_minute) {
ts->minute = ts->ds->end_time.local_minute;
}
if (ts->hour == ts->ds->end_time.local_hour &&
ts->minute == ts->ds->end_time.local_minute &&
ts->second > ts->ds->end_time.local_second) {
ts->second = ts->ds->end_time.local_second;
}
} else if (ts->ds->start_time.local_year == ts->ds->year &&
ts->ds->start_time.local_month == ts->ds->month &&
ts->ds->start_time.local_day == ts->ds->day) {
if (ts->hour < ts->ds->start_time.local_hour) {
ts->hour = ts->ds->start_time.local_hour;
}
if (ts->hour == ts->ds->start_time.local_hour &&
ts->minute < ts->ds->start_time.local_minute) {
ts->minute = ts->ds->start_time.local_minute;
}
if (ts->hour == ts->ds->start_time.local_hour &&
ts->minute == ts->ds->start_time.local_minute &&
ts->second > ts->ds->start_time.local_second) {
ts->second = ts->ds->start_time.local_second;
}
}
}
}
static void time_sel_add(TimeSelection *ts, int amount) {
switch (ts->stage) {
case TS_STAGE_HOUR:
ts->hour += amount;
break;
case TS_STAGE_MINUTE:
ts->minute += amount;
break;
case TS_STAGE_SECOND:
ts->second += amount;
break;
}
if (ts->second >= 60) {
++ts->minute;
} else if (ts->second < 0) {
--ts->minute;
}
if (ts->minute >= 60) {
++ts->hour;
} else if (ts->minute < 0) {
--ts->hour;
}
ts->hour %= 24;
ts->minute %= 60;
ts->second %= 60;
if (ts->hour < 0) {
ts->hour += 24;
}
if (ts->minute < 0) {
ts->minute += 60;
}
if (ts->second < 0) {
ts->second += 60;
}
time_sel_clamp(ts);
}
TimeSelState time_sel_dispatch(TimeSelection *ts, SensorState *state,
const char *label) {
if (state->back_down) {
if (ts->stage != TS_STAGE_HOUR) {
--ts->stage;
} else {
lcd_display_control(state->lcd, LCD_CURSOR_NO_BLINK, LCD_CURSOR_OFF,
LCD_DISPLAY_ON);
return TIME_SEL_BACK;
}
}
if (state->sel_down) {
if ((ts->do_second && ts->stage == TS_STAGE_SECOND) ||
(!ts->do_second && ts->stage == TS_STAGE_MINUTE)) {
lcd_display_control(state->lcd, LCD_CURSOR_NO_BLINK, LCD_CURSOR_OFF,
LCD_DISPLAY_ON);
return TIME_SEL_DONE;
} else {
++ts->stage;
}
}
if (ts->ds) {
UtilDate start, end;
if (!get_database_limits(state->db, PERIOD_DAY, &start, &end)) {
warnx("failed to query database limits");
lcd_display_control(state->lcd, LCD_CURSOR_NO_BLINK, LCD_CURSOR_OFF,
LCD_DISPLAY_ON);
return TIME_SEL_ERROR;
}
ts->ds->start_time = start;
ts->ds->end_time = end;
}
time_sel_add(ts, state->up_down - state->down_down);
lcd_clear(state->lcd);
lcd_move_to(state->lcd, 0, 0);
lcd_write_string(state->lcd, label);
char time_line[17];
int cursor_pos;
if (ts->do_second) {
const char *format;
switch (ts->stage) {
case TS_STAGE_HOUR:
format = ">%02d:%02d:%02d";
cursor_pos = 0;
break;
case TS_STAGE_MINUTE:
format = "%02d:>%02d:%02d";
cursor_pos = 3;
break;
case TS_STAGE_SECOND:
format = "%02d:%02d:>%02d";
cursor_pos = 6;
break;
}
snprintf(time_line, 17, format, ts->hour, ts->minute, ts->second);
} else {
const char *format;
switch (ts->stage) {
case TS_STAGE_HOUR:
format = ">%02d:%02d";
cursor_pos = 0;
break;
default: // minute or second (which shouldn't happen)
format = "%02d:>%02d";
cursor_pos = 3;
break;
}
snprintf(time_line, 17, format, ts->hour, ts->minute);
}
lcd_move_to(state->lcd, 1, 0);
lcd_write_string(state->lcd, time_line);
lcd_move_to(state->lcd, 1, cursor_pos);
lcd_display_control(state->lcd, LCD_CURSOR_BLINK, LCD_CURSOR_ON,
LCD_DISPLAY_ON);
return TIME_SEL_CONTINUE;
}
void time_sel_reset(TimeSelection *ts) {
ts->hour = 0;
ts->minute = 0;
ts->stage = TS_STAGE_HOUR;
}
int64_t timedate_sel_to_uts_timestamp(DateSelection *ds, TimeSelection *ts) {
struct tm broken_time = {
.tm_sec = ts->second,
.tm_min = ts->minute,
.tm_hour = ts->hour,
.tm_mday = ds->day,
.tm_mon = ds->month - 1,
.tm_year = ds->year - 1900,
.tm_isdst = -1,
};
time_t it = mktime(&broken_time);
if (it < 0) {
warnx("failed to convert to UTC timestamp");
return 0;
}
return it;
}