mirror of
https://github.com/Valeh2012/PersonalVotingMachine
synced 2024-11-23 09:41:01 +02:00
454 lines
12 KiB
C
454 lines
12 KiB
C
|
/*
|
||
|
|
||
|
guifn.c
|
||
|
|
||
|
*/
|
||
|
|
||
|
#include "gui.h"
|
||
|
#include "datecalc.h"
|
||
|
#include <string.h>
|
||
|
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
|
||
|
gui_alarm_t gui_alarm_list[GUI_ALARM_CNT];
|
||
|
char gui_alarm_str[GUI_ALARM_CNT][8];
|
||
|
uint32_t gui_backup_array[5];
|
||
|
|
||
|
|
||
|
gui_data_t gui_data;
|
||
|
|
||
|
menu_t gui_menu;
|
||
|
|
||
|
/*============================================*/
|
||
|
/* local functions */
|
||
|
|
||
|
uint32_t get_u32_by_alarm_data(gui_alarm_t *alarm);
|
||
|
void set_alarm_data_by_u32(gui_alarm_t *alarm, uint32_t u);
|
||
|
void gui_alarm_calc_next_wd_alarm(uint8_t idx, uint16_t current_week_time_in_minutes);
|
||
|
void gui_alarm_calc_str_time(uint8_t idx) U8G2_NOINLINE;
|
||
|
void gui_date_adjust(void) U8G2_NOINLINE;
|
||
|
void gui_calc_week_time(void);
|
||
|
void gui_calc_next_alarm(void);
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
|
||
|
|
||
|
uint32_t get_u32_by_alarm_data(gui_alarm_t *alarm)
|
||
|
{
|
||
|
uint32_t u;
|
||
|
int i;
|
||
|
u = 0;
|
||
|
for( i = 0; i < 7; i++ )
|
||
|
{
|
||
|
if ( alarm->wd[i] )
|
||
|
{
|
||
|
u |= 1<<i;
|
||
|
}
|
||
|
}
|
||
|
/* 17 Sep 2017: fixed 32 bit handling for the next statements */
|
||
|
u |= ((uint32_t)alarm->m&63) << (7);
|
||
|
u |= ((uint32_t)alarm->h&31) << (7+6);
|
||
|
u |= ((uint32_t)alarm->skip_wd&7) << (7+6+5);
|
||
|
u |= ((uint32_t)alarm->enable&1) << (7+6+5+3);
|
||
|
u |= ((uint32_t)alarm->snooze_count&1) << (7+6+5+3+1);
|
||
|
return u;
|
||
|
}
|
||
|
|
||
|
void set_alarm_data_by_u32(gui_alarm_t *alarm, uint32_t u)
|
||
|
{
|
||
|
int i;
|
||
|
for( i = 0; i < 7; i++ )
|
||
|
{
|
||
|
if ( (u & (1<<i)) != 0 )
|
||
|
alarm->wd[i] = 1;
|
||
|
else
|
||
|
alarm->wd[i] = 0;
|
||
|
}
|
||
|
u>>=7;
|
||
|
alarm->m = u & 63;
|
||
|
u>>=6;
|
||
|
alarm->h = u & 31;
|
||
|
u>>=5;
|
||
|
alarm->skip_wd = u & 7;
|
||
|
u>>=3;
|
||
|
alarm->enable = u & 1;
|
||
|
u>>=1;
|
||
|
alarm->snooze_count = u & 1;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
|
||
|
void gui_alarm_calc_next_wd_alarm(uint8_t idx, uint16_t current_week_time_in_minutes)
|
||
|
{
|
||
|
uint8_t i;
|
||
|
uint16_t week_time_abs;
|
||
|
uint16_t week_time_diff; /* difference to current_week_time_in_minutes */
|
||
|
uint16_t best_diff = 0x0ffff;
|
||
|
gui_alarm_list[idx].na_week_time_in_minutes = 0x0ffff; /* not found */
|
||
|
gui_alarm_list[idx].na_minutes_diff = 0x0ffff; /* not found */
|
||
|
gui_alarm_list[idx].na_wd = 7; /* not found */
|
||
|
|
||
|
//printf("gui_alarm_calc_next_wd_alarm: %d\n", idx);
|
||
|
|
||
|
if ( gui_alarm_list[idx].enable != 0 )
|
||
|
{
|
||
|
//printf("gui_alarm_calc_next_wd_alarm: %d enabled\n", idx);
|
||
|
for( i = 0; i < 7; i++ )
|
||
|
{
|
||
|
if ( gui_alarm_list[idx].wd[i] != 0 )
|
||
|
{
|
||
|
//printf("gui_alarm_calc_next_wd_alarm: %d i=%d gui_alarm_list[idx].skip_wd=%d \n", idx, i, gui_alarm_list[idx].skip_wd);
|
||
|
if ( gui_alarm_list[idx].skip_wd != i+1 )
|
||
|
{
|
||
|
week_time_abs = i;
|
||
|
week_time_abs *= 24;
|
||
|
week_time_abs += gui_alarm_list[idx].h;
|
||
|
week_time_abs *= 60;
|
||
|
week_time_abs += gui_alarm_list[idx].m;
|
||
|
week_time_abs += gui_alarm_list[idx].snooze_count*(uint16_t)SNOOZE_MINUTES;
|
||
|
|
||
|
|
||
|
if ( current_week_time_in_minutes <= week_time_abs )
|
||
|
week_time_diff = week_time_abs - current_week_time_in_minutes;
|
||
|
else
|
||
|
week_time_diff = week_time_abs + 7*24*60 - current_week_time_in_minutes;
|
||
|
|
||
|
//printf("gui_alarm_calc_next_wd_alarm: %d week_time_abs=%d current_week_time_in_minutes=%d week_time_diff=%d\n", idx, week_time_abs, current_week_time_in_minutes,week_time_diff);
|
||
|
|
||
|
if ( best_diff > week_time_diff )
|
||
|
{
|
||
|
best_diff = week_time_diff;
|
||
|
/* found for this alarm */
|
||
|
gui_alarm_list[idx].na_minutes_diff = week_time_diff;
|
||
|
gui_alarm_list[idx].na_week_time_in_minutes = week_time_abs;
|
||
|
gui_alarm_list[idx].na_h = gui_alarm_list[idx].h;
|
||
|
gui_alarm_list[idx].na_m = gui_alarm_list[idx].m;
|
||
|
gui_alarm_list[idx].na_wd = i;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
//printf("gui_alarm_calc_next_wd_alarm: %d na_minutes_diff=%d\n", idx, gui_alarm_list[idx].na_minutes_diff);
|
||
|
//printf("gui_alarm_calc_next_wd_alarm: %d na_wd=%d\n", idx, gui_alarm_list[idx].na_wd);
|
||
|
}
|
||
|
|
||
|
|
||
|
void gui_alarm_calc_str_time(uint8_t idx)
|
||
|
{
|
||
|
gui_alarm_str[idx][0] = ' ';
|
||
|
strcpy(gui_alarm_str[idx]+1, u8x8_u8toa(gui_alarm_list[idx].h, 2));
|
||
|
strcpy(gui_alarm_str[idx]+4, u8x8_u8toa(gui_alarm_list[idx].m, 2));
|
||
|
gui_alarm_str[idx][3] = ':';
|
||
|
if ( gui_alarm_list[idx].enable == 0 )
|
||
|
{
|
||
|
gui_alarm_str[idx][0] = '(';
|
||
|
gui_alarm_str[idx][6] = ')';
|
||
|
gui_alarm_str[idx][7] = '\0';
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* adjust day/month and calculates the weekday */
|
||
|
/* this function must be called after reading from RTC or after setting the input vars by the user */
|
||
|
void gui_date_adjust(void)
|
||
|
{
|
||
|
uint16_t ydn;
|
||
|
uint16_t year;
|
||
|
//uint16_t cdn;
|
||
|
//uint32_t minutes_since_2000;
|
||
|
|
||
|
if ( gui_data.month == 0 )
|
||
|
gui_data.month++;
|
||
|
if ( gui_data.day == 0 )
|
||
|
gui_data.day++;
|
||
|
year = 2000+gui_data.year_t*10 + gui_data.year_o;
|
||
|
ydn = get_year_day_number(year, gui_data.month, gui_data.day);
|
||
|
//cdn = to_century_day_number(year, ydn);
|
||
|
//minutes_since_2000 = to_minutes(cdn, gui_data.h, gui_data.mt*10+gui_data.mo);
|
||
|
|
||
|
// maybe adjust time by +/- 1h based on argument given to gui_date_adjust... but then it should be renamed also
|
||
|
|
||
|
gui_data.month = get_month_by_year_day_number(year, ydn);
|
||
|
gui_data.day = get_day_by_year_day_number(year, ydn);
|
||
|
gui_data.weekday = get_weekday_by_year_day_number(year, ydn); /* 0 = Sunday */
|
||
|
/* adjust the weekday so that 0 will be Monday */
|
||
|
gui_data.weekday += 6;
|
||
|
if ( gui_data.weekday >= 7 )
|
||
|
gui_data.weekday -= 7;
|
||
|
|
||
|
if ( gui_data.day != gui_data.last_day )
|
||
|
{
|
||
|
gui_data.uptime++;
|
||
|
gui_data.last_day = gui_data.day;
|
||
|
}
|
||
|
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
calculate the minute within the week.
|
||
|
this must be called after gui_date_adjust(), because the weekday is used here
|
||
|
*/
|
||
|
void gui_calc_week_time(void)
|
||
|
{
|
||
|
gui_data.week_time = gui_data.weekday;
|
||
|
gui_data.week_time *= 24;
|
||
|
gui_data.week_time += gui_data.h;
|
||
|
gui_data.week_time *= 60;
|
||
|
gui_data.week_time += gui_data.mt * 10 + gui_data.mo;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
/*
|
||
|
calculate the next alarm.
|
||
|
this must be called after gui_calc_week_time() because, we need week_time
|
||
|
*/
|
||
|
void gui_calc_next_alarm(void)
|
||
|
{
|
||
|
uint8_t i;
|
||
|
uint8_t lowest_i;
|
||
|
uint16_t lowest_diff;
|
||
|
uint8_t redo;
|
||
|
|
||
|
do
|
||
|
{
|
||
|
redo = 0;
|
||
|
|
||
|
/* step 1: Calculate the difference to current weektime for each alarm */
|
||
|
/* result is stored in gui_alarm_list[i].na_minutes_diff */
|
||
|
for( i = 0; i < GUI_ALARM_CNT; i++ )
|
||
|
gui_alarm_calc_next_wd_alarm(i, gui_data.week_time+(uint16_t)gui_data.is_equal); /* is_equal flag is used as a offset */
|
||
|
|
||
|
/* step 2: find the index with the lowest difference */
|
||
|
lowest_diff = 0x0ffff;
|
||
|
lowest_i = GUI_ALARM_CNT;
|
||
|
for( i = 0; i < GUI_ALARM_CNT; i++ )
|
||
|
{
|
||
|
if ( lowest_diff > gui_alarm_list[i].na_minutes_diff )
|
||
|
{
|
||
|
lowest_diff = gui_alarm_list[i].na_minutes_diff;
|
||
|
lowest_i = i;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/* step 3: store the result */
|
||
|
gui_data.next_alarm_index = lowest_i; /* this can be GUI_ALARM_CNT */
|
||
|
//printf("gui_calc_next_alarm gui_data.next_alarm_index=%d\n", gui_data.next_alarm_index);
|
||
|
|
||
|
/* calculate the is_skip_possible and the is_alarm flag */
|
||
|
gui_data.is_skip_possible = 0;
|
||
|
if ( lowest_i < GUI_ALARM_CNT )
|
||
|
{
|
||
|
|
||
|
if ( gui_alarm_list[lowest_i].na_minutes_diff == 0 )
|
||
|
{
|
||
|
if ( gui_data.is_equal == 0 )
|
||
|
{
|
||
|
gui_data.is_alarm = 1;
|
||
|
gui_data.is_equal = 1;
|
||
|
gui_data.active_alarm_idx = lowest_i;
|
||
|
//gui_data.active_equal_idx = lowest_i;
|
||
|
|
||
|
gui_data.equal_h = gui_data.h;
|
||
|
gui_data.equal_mt = gui_data.mt;
|
||
|
gui_data.equal_mo = gui_data.mo;
|
||
|
|
||
|
redo = 1;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
|
||
|
/* valid next alarm time */
|
||
|
if ( gui_alarm_list[lowest_i].skip_wd == 0 )
|
||
|
{
|
||
|
/* skip flag not yet set */
|
||
|
if ( gui_alarm_list[lowest_i].na_minutes_diff <= (uint16_t)60*(uint16_t)ALLOW_SKIP_HOURS )
|
||
|
{
|
||
|
/* within the limit before alarm */
|
||
|
gui_data.is_skip_possible = 1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
} while( redo );
|
||
|
|
||
|
/* reset the equal flag */
|
||
|
if ( gui_data.is_equal != 0 )
|
||
|
{
|
||
|
if ( gui_data.equal_h != gui_data.h ||
|
||
|
gui_data.equal_mt != gui_data.mt ||
|
||
|
gui_data.equal_mo != gui_data.mo )
|
||
|
{
|
||
|
gui_data.is_equal = 0;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
|
||
|
void gui_LoadData(void)
|
||
|
{
|
||
|
uint32_t data[5];
|
||
|
int i;
|
||
|
|
||
|
//printf("Load Data\n");
|
||
|
|
||
|
load_gui_data(data);
|
||
|
for( i = 0; i < GUI_ALARM_CNT; i++ )
|
||
|
{
|
||
|
set_alarm_data_by_u32(gui_alarm_list+i, data[i]);
|
||
|
}
|
||
|
gui_data.uptime = data[4] & (uint32_t)0x03ff;
|
||
|
gui_data.last_day = (data[4]>>10) & (uint32_t)31;
|
||
|
gui_data.contrast = (data[4]>>15) & (uint32_t)7;
|
||
|
gui_data.display_voltage = (data[4]>>16) & (uint32_t)1;
|
||
|
}
|
||
|
|
||
|
void gui_StoreData(void)
|
||
|
{
|
||
|
uint32_t data[5];
|
||
|
int i;
|
||
|
for( i = 0; i < GUI_ALARM_CNT; i++ )
|
||
|
{
|
||
|
data[i] = get_u32_by_alarm_data(gui_alarm_list+i);
|
||
|
//printf("%d: %08lx\n", i, data[i]);
|
||
|
}
|
||
|
data[4] = 0;
|
||
|
data[4] |= gui_data.uptime & (uint32_t)0x03ff; /* 0...1023 */
|
||
|
data[4] |= (gui_data.last_day & (uint32_t)31)<<10;
|
||
|
data[4] |= (gui_data.contrast & (uint32_t)7)<<15;
|
||
|
data[4] |= (gui_data.display_voltage & (uint32_t)1)<<16;
|
||
|
|
||
|
store_gui_data(data);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
/* recalculate all internal data */
|
||
|
void gui_Recalculate(void)
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
gui_date_adjust();
|
||
|
gui_calc_week_time();
|
||
|
gui_calc_next_alarm();
|
||
|
for ( i = 0; i < GUI_ALARM_CNT; i++ )
|
||
|
{
|
||
|
gui_alarm_calc_str_time(i);
|
||
|
}
|
||
|
gui_StoreData();
|
||
|
}
|
||
|
|
||
|
/* minute and/or hour has changed */
|
||
|
/* additionally the active alarm menu might be set by this function */
|
||
|
void gui_SignalTimeChange(void)
|
||
|
{
|
||
|
/* recalculate dependent values */
|
||
|
gui_Recalculate();
|
||
|
|
||
|
/* setup menu */
|
||
|
menu_SetMEList(&gui_menu, melist_display_time, 0);
|
||
|
|
||
|
/* enable alarm */
|
||
|
if ( gui_data.is_alarm != 0 )
|
||
|
{
|
||
|
menu_SetMEList(&gui_menu, melist_active_alarm_menu, 0);
|
||
|
enable_alarm();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void gui_Init(u8g2_t *u8g2, uint8_t is_por)
|
||
|
{
|
||
|
if ( is_por == 0 )
|
||
|
{
|
||
|
/* not a POR reset, so load current values */
|
||
|
gui_LoadData();
|
||
|
/* do NOT init the display, otherwise there will be some flicker visible */
|
||
|
/* however, the GPIO subsystem still has to be setup, so call the other init procedures */
|
||
|
/* this acts like a warm start for the display */
|
||
|
/* the display setup code for the display is NOT send */
|
||
|
u8x8_gpio_Init(u8g2_GetU8x8(u8g2));
|
||
|
u8x8_cad_Init(u8g2_GetU8x8(u8g2));
|
||
|
u8x8_gpio_SetReset(u8g2_GetU8x8(u8g2), 1);
|
||
|
|
||
|
//u8g2_InitDisplay(u8g2);
|
||
|
//u8x8_d_helper_display_init(u8g2_GetU8x8(u8g2));
|
||
|
|
||
|
// u8g2_SetPowerSave(u8g2, 0); // this will be done later
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
/* POR reset, so do NOT load any values (they will be 0 in the best case) */
|
||
|
/* instead do a proper reset of the display */
|
||
|
// u8x8_InitDisplay(u8g2_GetU8x8(&u8g2));
|
||
|
u8g2_InitDisplay(u8g2);
|
||
|
|
||
|
// u8x8_SetPowerSave(u8g2_GetU8x8(&u8g2), 0);
|
||
|
u8g2_SetPowerSave(u8g2, 0);
|
||
|
}
|
||
|
|
||
|
menu_Init(&gui_menu, u8g2);
|
||
|
|
||
|
gui_SignalTimeChange();
|
||
|
}
|
||
|
|
||
|
|
||
|
void gui_Draw(void)
|
||
|
{
|
||
|
menu_Draw(&gui_menu);
|
||
|
}
|
||
|
|
||
|
void gui_Next(void)
|
||
|
{
|
||
|
if ( gui_menu.me_list == melist_active_alarm_menu )
|
||
|
{
|
||
|
disable_alarm();
|
||
|
if ( gui_alarm_list[gui_data.active_alarm_idx].snooze_count != 0 )
|
||
|
{
|
||
|
int i;
|
||
|
/* this is already the snooze alarm, so clear all and behave like the normal alarm off */
|
||
|
for( i = 0; i < GUI_ALARM_CNT; i++ )
|
||
|
gui_alarm_list[i].snooze_count = 0;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
/* enable snooze */
|
||
|
gui_alarm_list[gui_data.active_alarm_idx].snooze_count = 1;
|
||
|
}
|
||
|
gui_data.is_alarm = 0;
|
||
|
gui_Recalculate();
|
||
|
menu_SetMEList(&gui_menu, melist_display_time, 0);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
menu_NextFocus(&gui_menu);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void gui_Select(void)
|
||
|
{
|
||
|
if ( gui_menu.me_list == melist_active_alarm_menu )
|
||
|
{
|
||
|
int i;
|
||
|
disable_alarm();
|
||
|
for( i = 0; i < GUI_ALARM_CNT; i++ )
|
||
|
gui_alarm_list[i].snooze_count = 0;
|
||
|
gui_data.is_alarm = 0;
|
||
|
gui_Recalculate();
|
||
|
menu_SetMEList(&gui_menu, melist_display_time, 0);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
menu_Select(&gui_menu);
|
||
|
}
|
||
|
}
|