You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
2719 lines
85 KiB
2719 lines
85 KiB
6 years ago
|
/*
|
||
|
|
||
|
AlarmClock.ino
|
||
|
|
||
|
Universal 8bit Graphics Library (https://github.com/olikraus/u8g2/)
|
||
|
|
||
|
Copyright (c) 2017, olikraus@gmail.com
|
||
|
All rights reserved.
|
||
|
|
||
|
Redistribution and use in source and binary forms, with or without modification,
|
||
|
are permitted provided that the following conditions are met:
|
||
|
|
||
|
* Redistributions of source code must retain the above copyright notice, this list
|
||
|
of conditions and the following disclaimer.
|
||
|
|
||
|
* Redistributions in binary form must reproduce the above copyright notice, this
|
||
|
list of conditions and the following disclaimer in the documentation and/or other
|
||
|
materials provided with the distribution.
|
||
|
|
||
|
THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
|
||
|
CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
|
||
|
INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||
|
MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
||
|
DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
|
||
|
CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||
|
SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
||
|
NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
|
||
|
LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||
|
CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
||
|
STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
||
|
ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
||
|
ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||
|
|
||
|
*/
|
||
|
|
||
|
#include <Arduino.h>
|
||
|
#include <RTCZero.h>
|
||
|
#include <U8g2lib.h>
|
||
|
|
||
|
#ifdef U8X8_HAVE_HW_SPI
|
||
|
#include <SPI.h>
|
||
|
#endif
|
||
|
#ifdef U8X8_HAVE_HW_I2C
|
||
|
#include <Wire.h>
|
||
|
#endif
|
||
|
|
||
|
// Please UNCOMMENT one of the contructor lines below
|
||
|
// U8g2 Contructor List (Picture Loop Page Buffer)
|
||
|
// The complete list is available here: https://github.com/olikraus/u8g2/wiki/u8g2setupcpp
|
||
|
// Please update the pin numbers according to your setup. Use U8X8_PIN_NONE if the reset pin is not connected
|
||
|
//U8G2_NULL u8g2(U8G2_R0); // null device, a 8x8 pixel display which does nothing
|
||
|
//U8G2_SSD1306_128X64_NONAME_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1306_128X64_NONAME_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 12, /* dc=*/ 4, /* reset=*/ 6); // Arduboy (Production, Kickstarter Edition)
|
||
|
//U8G2_SSD1306_128X64_NONAME_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1306_128X64_NONAME_1_3W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* reset=*/ 8);
|
||
|
//U8G2_SSD1306_128X64_NONAME_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_SSD1306_128X64_ALT0_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE); // same as the NONAME variant, but may solve the "every 2nd line skipped" problem
|
||
|
//U8G2_SSD1306_128X64_NONAME_1_SW_I2C u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* reset=*/ 8);
|
||
|
//U8G2_SSD1306_128X64_NONAME_1_SW_I2C u8g2(U8G2_R0, /* clock=*/ SCL, /* data=*/ SDA, /* reset=*/ U8X8_PIN_NONE); // All Boards without Reset of the Display
|
||
|
//U8G2_SSD1306_128X64_NONAME_1_SW_I2C u8g2(U8G2_R0, /* clock=*/ 16, /* data=*/ 17, /* reset=*/ U8X8_PIN_NONE); // ESP32 Thing, pure SW emulated I2C
|
||
|
//U8G2_SSD1306_128X64_NONAME_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE, /* clock=*/ 16, /* data=*/ 17); // ESP32 Thing, HW I2C with pin remapping
|
||
|
//U8G2_SSD1306_128X64_NONAME_1_6800 u8g2(U8G2_R0, 13, 11, 2, 3, 4, 5, 6, A4, /*enable=*/ 7, /*cs=*/ 10, /*dc=*/ 9, /*reset=*/ 8);
|
||
|
//U8G2_SSD1306_128X64_NONAME_1_8080 u8g2(U8G2_R0, 13, 11, 2, 3, 4, 5, 6, A4, /*enable=*/ 7, /*cs=*/ 10, /*dc=*/ 9, /*reset=*/ 8);
|
||
|
//U8G2_SSD1306_128X64_VCOMH0_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // same as the NONAME variant, but maximizes setContrast() range
|
||
|
//U8G2_SSD1306_128X64_ALT0_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // same as the NONAME variant, but may solve the "every 2nd line skipped" problem
|
||
|
//U8G2_SH1106_128X64_NONAME_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SH1106_128X64_NONAME_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_SH1106_128X64_VCOMH0_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // same as the NONAME variant, but maximizes setContrast() range
|
||
|
//U8G2_SH1106_128X64_WINSTAR_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // same as the NONAME variant, but uses updated SH1106 init sequence
|
||
|
//U8G2_SH1106_72X40_WISE_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SH1107_64X128_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SH1107_128X128_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SH1107_128X128_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ 8);
|
||
|
//U8G2_SH1107_SEEED_96X96_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SH1108_160X160_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SH1122_256X64_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Enable U8G2_16BIT in u8g2.h
|
||
|
//U8G2_SSD1306_128X32_UNIVISION_1_SW_I2C u8g2(U8G2_R0, /* clock=*/ 21, /* data=*/ 20, /* reset=*/ U8X8_PIN_NONE); // Adafruit Feather M0 Basic Proto + FeatherWing OLED
|
||
|
//U8G2_SSD1306_128X32_UNIVISION_1_SW_I2C u8g2(U8G2_R0, /* clock=*/ SCL, /* data=*/ SDA, /* reset=*/ U8X8_PIN_NONE); // Adafruit Feather ESP8266/32u4 Boards + FeatherWing OLED
|
||
|
//U8G2_SSD1306_128X32_UNIVISION_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE); // Adafruit ESP8266/32u4/ARM Boards + FeatherWing OLED
|
||
|
//U8G2_SSD1306_128X32_UNIVISION_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE, /* clock=*/ SCL, /* data=*/ SDA); // pin remapping with ESP8266 HW I2C
|
||
|
//U8G2_SSD1306_64X48_ER_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE); // EastRising 0.66" OLED breakout board, Uno: A4=SDA, A5=SCL, 5V powered
|
||
|
//U8G2_SSD1306_48X64_WINSTAR_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_SSD1306_64X32_NONAME_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_SSD1306_64X32_1F_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_SSD1306_96X16_ER_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE); // EastRising 0.69" OLED
|
||
|
//U8G2_SSD1322_NHD_256X64_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Enable U8G2_16BIT in u8g2.h
|
||
|
//U8G2_SSD1322_NHD_256X64_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Enable U8G2_16BIT in u8g2.h
|
||
|
//U8G2_SSD1322_NHD_128X64_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1322_NHD_128X64_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1325_NHD_128X64_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1325_NHD_128X64_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD0323_OS128064_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD0323_OS128064_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1326_ER_256X32_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // experimental driver for ER-OLED018-1
|
||
|
//U8G2_SSD1327_SEEED_96X96_1_SW_I2C u8g2(U8G2_R0, /* clock=*/ SCL, /* data=*/ SDA, /* reset=*/ U8X8_PIN_NONE); // Seeedstudio Grove OLED 96x96
|
||
|
//U8G2_SSD1327_SEEED_96X96_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE); // Seeedstudio Grove OLED 96x96
|
||
|
//U8G2_SSD1327_EA_W128128_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1327_EA_W128128_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1327_EA_W128128_1_SW_I2C u8g2(U8G2_R0, /* clock=*/ 5, /* data=*/ 4, /* reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_SSD1327_EA_W128128_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_SSD1327_MIDAS_128X128_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1327_MIDAS_128X128_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1329_128X96_NONAME_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1329_128X96_NONAME_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1305_128X32_NONAME_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1305_128X32_NONAME_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1305_128X32_ADAFRUIT_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1305_128X32_ADAFRUIT_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1305_128X64_ADAFRUIT_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1305_128X64_ADAFRUIT_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1309_128X64_NONAME0_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1309_128X64_NONAME0_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1309_128X64_NONAME2_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1309_128X64_NONAME2_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1317_96X96_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_SSD1317_96X96_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_LD7032_60X32_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 11, /* data=*/ 12, /* cs=*/ 9, /* dc=*/ 10, /* reset=*/ 8); // SW SPI Nano Board
|
||
|
//U8G2_LD7032_60X32_1_4W_SW_I2C u8g2(U8G2_R0, /* clock=*/ 11, /* data=*/ 12, /* reset=*/ U8X8_PIN_NONE); // NOT TESTED!
|
||
|
//U8G2_UC1701_EA_DOGS102_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_UC1701_EA_DOGS102_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_PCD8544_84X48_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Nokia 5110 Display
|
||
|
//U8G2_PCD8544_84X48_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Nokia 5110 Display
|
||
|
//U8G2_PCF8812_96X65_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Could be also PCF8814
|
||
|
//U8G2_PCF8812_96X65_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Could be also PCF8814
|
||
|
//U8G2_HX1230_96X68_1_3W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* reset=*/ 8);
|
||
|
//U8G2_HX1230_96X68_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_KS0108_128X64_1 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 18, /*dc=*/ 17, /*cs0=*/ 14, /*cs1=*/ 15, /*cs2=*/ U8X8_PIN_NONE, /* reset=*/ U8X8_PIN_NONE); // Set R/W to low!
|
||
|
//U8G2_KS0108_ERM19264_1 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 18, /*dc=*/ 17, /*cs0=*/ 14, /*cs1=*/ 15, /*cs2=*/ 16, /* reset=*/ U8X8_PIN_NONE); // Set R/W to low!
|
||
|
//U8G2_ST7920_192X32_1_8080 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 18, /*cs=*/ U8X8_PIN_NONE, /*dc=*/ 17, /*reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_ST7920_192X32_1_SW_SPI u8g2(U8G2_R0, /* clock=*/ 18 /* A4 */ , /* data=*/ 16 /* A2 */, /* CS=*/ 17 /* A3 */, /* reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_ST7920_128X64_1_8080 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 18 /* A4 */, /*cs=*/ U8X8_PIN_NONE, /*dc/rs=*/ 17 /* A3 */, /*reset=*/ 15 /* A1 */); // Remember to set R/W to 0
|
||
|
//U8G2_ST7920_128X64_1_SW_SPI u8g2(U8G2_R0, /* clock=*/ 18 /* A4 */ , /* data=*/ 16 /* A2 */, /* CS=*/ 17 /* A3 */, /* reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_ST7920_128X64_1_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* CS=*/ 10, /* reset=*/ 8);
|
||
|
//U8G2_ST7920_128X64_1_HW_SPI u8g2(U8G2_R0, /* CS=*/ 10, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_EA_DOGM128_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_EA_DOGM128_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_64128N_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_64128N_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_EA_DOGM132_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ U8X8_PIN_NONE); // DOGM132 Shield
|
||
|
//U8G2_ST7565_EA_DOGM132_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ U8X8_PIN_NONE); // DOGM132 Shield
|
||
|
//U8G2_ST7565_ZOLEN_128X64_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_ZOLEN_128X64_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_LM6059_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Adafruit ST7565 GLCD
|
||
|
//U8G2_ST7565_LM6059_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Adafruit ST7565 GLCD
|
||
|
//U8G2_ST7565_LX12864_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_LX12864_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_ERC12864_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_ERC12864_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_NHD_C12832_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_NHD_C12832_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_NHD_C12864_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_NHD_C12864_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_JLX12864_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7565_JLX12864_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7567_PI_132X64_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 7, /* dc=*/ 9, /* reset=*/ 8); // Pax Instruments Shield, LCD_BL=6
|
||
|
//U8G2_ST7567_PI_132X64_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 7, /* dc=*/ 9, /* reset=*/ 8); // Pax Instruments Shield, LCD_BL=6
|
||
|
//U8G2_ST7567_JLX12864_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 7, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7567_JLX12864_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 7, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7567_ENH_DG128064_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7567_ENH_DG128064_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7567_ENH_DG128064I_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7567_ENH_DG128064I_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7567_64X32_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_ST75256_JLX172104_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST75256_JLX172104_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST75256_JLX256128_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Uno, Enable U8g2 16 bit mode for this display
|
||
|
//U8G2_ST75256_JLX256128_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Uno, Enable U8g2 16 bit mode for this display
|
||
|
//U8G2_ST75256_JLX256128_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 9, /* data=*/ 8, /* cs=*/ 7, /* dc=*/ 6, /* reset=*/ 5); // MKR Zero, Enable U8g2 16 bit mode for this display
|
||
|
//U8G2_ST75256_JLX256128_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 7, /* dc=*/ 6, /* reset=*/ 5); // MKR Zero, Enable U8g2 16 bit mode for this display
|
||
|
//U8G2_ST75256_JLX256160_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Uno, Enable U8g2 16 bit mode for this display
|
||
|
//U8G2_ST75256_JLX256160_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Uno, Enable U8g2 16 bit mode for this display
|
||
|
//U8G2_ST75256_JLX240160_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST75256_JLX240160_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST75256_JLX25664_1_2ND_HW_I2C u8g2(U8G2_R0, /* reset=*/ 8); // Due, 2nd I2C, enable U8g2 16 bit mode for this display
|
||
|
//U8G2_NT7534_TG12864R_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_NT7534_TG12864R_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_ST7588_JLX12864_1_SW_I2C u8g2(U8G2_R0, /* clock=*/ SCL, /* data=*/ SDA, /* reset=*/ 5);
|
||
|
//U8G2_ST7588_JLX12864_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ 5);
|
||
|
//U8G2_IST3020_ERC19264_1_6800 u8g2(U8G2_R0, 44, 43, 42, 41, 40, 39, 38, 37, /*enable=*/ 28, /*cs=*/ 32, /*dc=*/ 30, /*reset=*/ 31); // Connect WR pin with GND
|
||
|
//U8G2_IST3020_ERC19264_1_8080 u8g2(U8G2_R0, 44, 43, 42, 41, 40, 39, 38, 37, /*enable=*/ 29, /*cs=*/ 32, /*dc=*/ 30, /*reset=*/ 31); // Connect RD pin with 3.3V
|
||
|
//U8G2_IST3020_ERC19264_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_LC7981_160X80_1_6800 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 18, /*cs=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // Connect RW with GND
|
||
|
//U8G2_LC7981_160X160_1_6800 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 18, /*cs=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // Connect RW with GND
|
||
|
//U8G2_LC7981_240X128_1_6800 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 18, /*cs=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // Connect RW with GND
|
||
|
//U8G2_LC7981_240X64_1_6800 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 18, /*cs=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // Connect RW with GND
|
||
|
//U8G2_SED1520_122X32_1 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*dc=*/ A0, /*e1=*/ A3, /*e2=*/ A2, /* reset=*/ A4); // Set R/W to low!
|
||
|
//U8G2_T6963_240X128_1_8080 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable/wr=*/ 17, /*cs/ce=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // Connect RD with +5V, FS0 and FS1 with GND
|
||
|
//U8G2_T6963_256X64_1_8080 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable/wr=*/ 17, /*cs/ce=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // Connect RD with +5V, FS0 and FS1 with GND
|
||
|
//U8G2_T6963_160X80_1_8080 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable/wr=*/ 17, /*cs/ce=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // Connect RD with +5V, FS0 and FS1 with GND
|
||
|
//U8G2_T6963_128X64_1_8080 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable/wr=*/ 17, /*cs/ce=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // Connect RD with +5V, FS0 and FS1 with GND
|
||
|
//U8G2_T6963_128X64_ALT_1_8080 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable/wr=*/ 17, /*cs/ce=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // Connect RD with +5V, FS0 and FS1 with GND
|
||
|
//U8G2_SED1330_240X128_1_8080 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 17, /*cs=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // Connect /RD = E with +5V, enable is /WR = RW, FG with GND, 14=Uno Pin A0
|
||
|
//U8G2_SED1330_240X128_1_6800 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 17, /*cs=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // A0 is dc pin, /WR = RW = GND, enable is /RD = E
|
||
|
//U8G2_RA8835_NHD_240X128_1_8080 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 17, /*cs=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // Connect /RD = E with +5V, enable is /WR = RW, FG with GND, 14=Uno Pin A0
|
||
|
//U8G2_RA8835_NHD_240X128_1_6800 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 17, /*cs=*/ 14, /*dc=*/ 15, /*reset=*/ 16); // A0 is dc pin, /WR = RW = GND, enable is /RD = E
|
||
|
//U8G2_UC1601_128X32_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_UC1601_128X32_1_3W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* reset=*/ 8);
|
||
|
//U8G2_UC1604_JLX19264_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_UC1604_JLX19264_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_UC1608_ERC24064_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // SW SPI, Due ERC24064-1 Test Setup
|
||
|
//U8G2_UC1608_ERC240120_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8);
|
||
|
//U8G2_UC1608_240X128_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // SW SPI, Due ERC24064-1 Test Setup
|
||
|
//U8G2_UC1610_EA_DOGXL160_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_UC1610_EA_DOGXL160_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ U8X8_PIN_NONE);
|
||
|
//U8G2_UC1611_EA_DOGM240_1_2ND_HW_I2C u8g2(U8G2_R0, /* reset=*/ 8); // Due, 2nd I2C, DOGM240 Test Board
|
||
|
//U8G2_UC1611_EA_DOGM240_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // SW SPI, Due DOGXL240 Test Board
|
||
|
//U8G2_UC1611_EA_DOGXL240_1_2ND_HW_I2C u8g2(U8G2_R0, /* reset=*/ 8); // Due, 2nd I2C, DOGXL240 Test Board
|
||
|
//U8G2_UC1611_EA_DOGXL240_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // SW SPI, Due DOGXL240 Test Board
|
||
|
//U8G2_UC1611_EW50850_1_8080 u8g2(U8G2_R0, 8, 9, 10, 11, 4, 5, 6, 7, /*enable=*/ 18, /*cs=*/ 3, /*dc=*/ 16, /*reset=*/ 16); // 240x160, Connect RD/WR1 pin with 3.3V, CS is aktive high
|
||
|
//U8G2_UC1638_160X128_1_4W_HW_SPI u8g2(U8G2_R2, /* cs=*/ 2, /* dc=*/ 3, /* reset=*/ 4); // Not tested
|
||
|
//U8G2_SSD1606_172X72_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // eInk/ePaper Display
|
||
|
//U8G2_SSD1607_200X200_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // eInk/ePaper Display, original LUT from embedded artists
|
||
|
//U8G2_SSD1607_GD_200X200_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // Good Display
|
||
|
//U8G2_IL3820_296X128_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // WaveShare 2.9 inch eInk/ePaper Display, enable 16 bit mode for this display!
|
||
|
//U8G2_IL3820_V2_296X128_1_4W_SW_SPI u8g2(U8G2_R0, /* clock=*/ 13, /* data=*/ 11, /* cs=*/ 10, /* dc=*/ 9, /* reset=*/ 8); // ePaper Display, lesser flickering and faster speed, enable 16 bit mode for this display!
|
||
|
//U8G2_LS013B7DH03_128X128_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ U8X8_PIN_NONE, /* reset=*/ 8); // there is no DC line for this display
|
||
|
//U8G2_LS027B7DH01_400X240_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ U8X8_PIN_NONE, /* reset=*/ 8); // there is no DC line for this display
|
||
|
//U8G2_LS013B7DH05_144X168_1_4W_HW_SPI u8g2(U8G2_R0, /* cs=*/ 10, /* dc=*/ U8X8_PIN_NONE, /* reset=*/ 8); // there is no DC line for this display
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
// End of constructor list
|
||
|
|
||
|
U8G2_SSD1306_128X64_NONAME_1_HW_I2C u8g2(U8G2_R0, /* reset=*/ U8X8_PIN_NONE);
|
||
|
|
||
|
RTCZero rtc;
|
||
|
|
||
|
|
||
|
/*===================================================*/
|
||
|
|
||
|
uint8_t is_leap_year(uint16_t y);
|
||
|
uint16_t get_year_day_number(uint16_t y, uint8_t m, uint8_t d);
|
||
|
uint8_t get_month_by_year_day_number(uint16_t y, uint16_t ydn);
|
||
|
uint8_t get_day_by_year_day_number(uint16_t y, uint16_t ydn);
|
||
|
uint8_t get_weekday_by_year_day_number(uint16_t y, uint16_t ydn); /* returns day within a week: 0..6 with 0 = Sunday, 1 = Monday, ... */
|
||
|
uint16_t to_century_day_number(uint16_t y, uint16_t ydn);
|
||
|
void from_century_day_number(uint16_t cdn, uint16_t *year, uint16_t *ydn);
|
||
|
uint32_t to_time(uint16_t cdn, uint8_t h, uint8_t m, uint8_t s);
|
||
|
void from_time(uint32_t t, uint16_t *cdn, uint8_t *h, uint8_t *m, uint8_t *s);
|
||
|
|
||
|
uint32_t to_sec_since_2000(uint16_t y, uint8_t mo, uint8_t d, uint8_t h, uint8_t mi, uint8_t s);
|
||
|
|
||
|
uint32_t to_minutes(uint16_t cdn, uint8_t h, uint8_t m);
|
||
|
void from_minutes(uint32_t t, uint16_t *cdn, uint8_t *h, uint8_t *m);
|
||
|
uint32_t to_minutes_since_2000(uint16_t y, uint8_t mo, uint8_t d, uint8_t h, uint8_t mi);
|
||
|
|
||
|
|
||
|
/*===================================================*/
|
||
|
|
||
|
#define MENU_SMALL_FONT u8g2_font_baby_tr
|
||
|
#define MENU_NORMAL_FONT u8g2_font_ncenR08_tf
|
||
|
#define MENU_LARGE_FONT u8g2_font_ncenR10_tf
|
||
|
#define MENU_BIG_NUM u8g2_font_ncenR24_tn
|
||
|
#define MENU_BIG_NUM_FOCUS_XO 0
|
||
|
#define MENU_BIG_NUM_FOCUS_EXTRAX 1
|
||
|
|
||
|
typedef struct _menu_struct menu_t;
|
||
|
typedef struct _me_struct me_t;
|
||
|
|
||
|
typedef int (*me_cb)(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
|
||
|
|
||
|
|
||
|
struct _me_struct
|
||
|
{
|
||
|
me_cb cb;
|
||
|
void *val;
|
||
|
void *arg;
|
||
|
u8g2_uint_t x;
|
||
|
u8g2_uint_t y;
|
||
|
};
|
||
|
|
||
|
|
||
|
/* return 1, if this element can have focus */
|
||
|
#define ME_MSG_IS_FOCUS 1
|
||
|
|
||
|
/* draw focus graphics for the element */
|
||
|
#define ME_MSG_DRAW_FOCUS 2
|
||
|
|
||
|
/* user has pressed the select key */
|
||
|
#define ME_MSG_SELECT 3
|
||
|
|
||
|
/* advice for drawing */
|
||
|
#define ME_MSG_DRAW 4
|
||
|
|
||
|
|
||
|
struct _menu_struct
|
||
|
{
|
||
|
u8g2_t *u8g2;
|
||
|
|
||
|
volatile uint16_t current_index; /* element which is processed right now */
|
||
|
uint16_t focus_index; /* element which has the focus at the moment */
|
||
|
uint16_t radio_index; /* if elements for a radio selection, then this is set by the cb */
|
||
|
uint16_t me_count; /* total number of elements in the list */
|
||
|
|
||
|
/* pointer to the list of menu elements */
|
||
|
const me_t *me_list;
|
||
|
};
|
||
|
|
||
|
void menu_SetEdgePixel(menu_t *menu, u8g2_uint_t x, u8g2_uint_t y, u8g2_uint_t w, u8g2_uint_t h) U8G2_NOINLINE;
|
||
|
void menu_ClearEdgePixel(menu_t *menu, u8g2_uint_t x, u8g2_uint_t y, u8g2_uint_t w, u8g2_uint_t h) U8G2_NOINLINE;
|
||
|
void menu_DrawBoxFocus(menu_t *menu, u8g2_uint_t x, u8g2_uint_t y, u8g2_uint_t w, u8g2_uint_t h) U8G2_NOINLINE;
|
||
|
void menu_DrawFrameFocus(menu_t *menu, u8g2_uint_t x, u8g2_uint_t y, u8g2_uint_t w, u8g2_uint_t h) U8G2_NOINLINE;
|
||
|
|
||
|
void menu_Init(menu_t *menu, u8g2_t *u8g2);
|
||
|
void menu_SetMEList(menu_t *menu, const me_t *me_list, uint16_t initial_focus);
|
||
|
void menu_Draw(menu_t *menu);
|
||
|
void menu_NextFocus(menu_t *menu);
|
||
|
void menu_Select(menu_t *menu);
|
||
|
|
||
|
|
||
|
int me_cb_null(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_big_toggle(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_wd_toggle(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_0_5(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_0_9(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_0_9_ro(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_0_23(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_0_23_ro(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_0_9_small_ro(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_0_55(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_1_12(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_1_31(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_num_label(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_button_full_line(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_button_half_line(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_button_empty(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_scale_1_7(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_label(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_inv_label(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
int me_cb_16x16_bitmap_button(menu_t *menu, const me_t *me, uint8_t msg);
|
||
|
|
||
|
/*===================================================*/
|
||
|
|
||
|
u8g2_t *u8g2_ptr;
|
||
|
|
||
|
#define GUI_STATE_STOP 0
|
||
|
#define GUI_STATE_SIGNAL_ALARM 1
|
||
|
#define GUI_STATE_DISPLAY_TIME 2
|
||
|
#define GUI_STATE_MENU 3
|
||
|
|
||
|
#define GUI_ALARM_CNT 4
|
||
|
#define SNOOZE_MINUTES 5
|
||
|
#define ALLOW_SKIP_HOURS 10
|
||
|
|
||
|
struct _gui_data
|
||
|
{
|
||
|
uint16_t uptime; /* uptime in days, 10 bits, counts from 0 to 999, this value will be stored in the backup register */
|
||
|
|
||
|
|
||
|
uint16_t week_time; /* calculated: derived from h, mt, mo and weekday */
|
||
|
uint8_t gui_state; /* global running state, see guistate.c, defaults to 0 (GUI_STATE_STOP) */
|
||
|
uint8_t h, mt, mo, st, so; /* input: current time */
|
||
|
uint8_t day; /* input: 1 .. 31 current day in month */
|
||
|
uint8_t last_day; /* last day. This is used to check, whether the day has changed. Required for uptime calc. This is also stored in the backup register. */
|
||
|
uint8_t month; /* input: 1..12 */
|
||
|
uint8_t year_t, year_o; /* input: current year */
|
||
|
uint8_t weekday; /* calculated: 0 = Monday */
|
||
|
|
||
|
uint8_t next_alarm_index; /* calculated: index for the next alarm or GUI_ALARM_CNT if there is no next alarm */
|
||
|
uint8_t is_skip_possible; /* calculated: whether the next alarm (next_alarm_index) can be skipped */
|
||
|
|
||
|
uint8_t is_equal; /* calculated: whether the current time matches any alarm, will be set to 0 automatically */
|
||
|
uint8_t equal_h;
|
||
|
uint8_t equal_mt;
|
||
|
uint8_t equal_mo;
|
||
|
|
||
|
uint8_t is_alarm; /* input/calculated: set by the software, has to be reset by the user */
|
||
|
uint8_t active_alarm_idx; /* input/calculated: set by the software, has to be reset by the user */
|
||
|
uint8_t contrast; /* value 1..7, 0 is default (do not set) */
|
||
|
uint8_t display_voltage;
|
||
|
|
||
|
char s[16]; /* string buffer */
|
||
|
};
|
||
|
typedef struct _gui_data gui_data_t;
|
||
|
|
||
|
|
||
|
struct _gui_alarm_struct
|
||
|
{
|
||
|
/* next alarm, all na_ fields are derived from the alarm information */
|
||
|
uint16_t na_week_time_in_minutes;
|
||
|
uint16_t na_minutes_diff; /* calculated: time in minutes until next alarm, 0x0ffff = no alarm */
|
||
|
uint8_t na_h; /* calculated */
|
||
|
uint8_t na_m; /* calculated */
|
||
|
uint8_t na_wd; /* calculated: 0...7, 0=monday, 7=no alarm */
|
||
|
|
||
|
/* alarm information */
|
||
|
uint8_t snooze_count; /* input, 1 bit*/
|
||
|
volatile uint8_t enable; /* input, 1 bit */
|
||
|
uint8_t skip_wd; /* input 0 = no skip, 1 = Monday, ... 3 bits*/
|
||
|
uint8_t h; /* input 5 bits */
|
||
|
uint8_t m; /* input 6 bits */
|
||
|
uint8_t wd[7]; /* input: 0 or 1, 0=weekday not selected, 7 bits */
|
||
|
};
|
||
|
typedef struct _gui_alarm_struct gui_alarm_t;
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
extern const me_t melist_setup_time[];
|
||
|
extern const me_t melist_display_time[];
|
||
|
extern const me_t melist_setup_date[];
|
||
|
extern const me_t melist_setup_alarm[];
|
||
|
extern const me_t melist_alarm_menu[];
|
||
|
extern const me_t melist_setup_menu[];
|
||
|
extern const me_t melist_active_alarm_menu[];
|
||
|
extern const me_t melist_top_menu[];
|
||
|
|
||
|
extern gui_alarm_t gui_alarm_list[GUI_ALARM_CNT];
|
||
|
extern char gui_alarm_str[GUI_ALARM_CNT][8];
|
||
|
extern gui_data_t gui_data;
|
||
|
extern menu_t gui_menu;
|
||
|
|
||
|
void gui_date_adjust(void);
|
||
|
|
||
|
|
||
|
void gui_LoadData(void);
|
||
|
void gui_Recalculate(void);
|
||
|
void gui_SignalTimeChange(void);
|
||
|
void gui_Init(u8g2_t *u8g2, uint8_t is_por);
|
||
|
void gui_Draw(void);
|
||
|
void gui_Next(void);
|
||
|
void gui_Select(void);
|
||
|
|
||
|
void do_reset(void);
|
||
|
void store_gui_data(uint32_t *data);
|
||
|
void load_gui_data(uint32_t *data);
|
||
|
int is_por_reset(void);
|
||
|
int is_button_reset(void);
|
||
|
uint32_t get_boot_status_register(void);
|
||
|
uint32_t get_pwr_status_register(void);
|
||
|
uint32_t get_reset_reason(void);
|
||
|
uint32_t get_wakeup_count(void);
|
||
|
uint32_t get_dst_by_date(void);
|
||
|
uint32_t get_dst_by_RTC(void);
|
||
|
|
||
|
void enable_alarm(void);
|
||
|
void disable_alarm(void);
|
||
|
void set_time(uint8_t ht, uint8_t ho, uint8_t mt, uint8_t mo, uint8_t st, uint8_t so);
|
||
|
void set_date(uint8_t yt, uint8_t yo, uint8_t mt, uint8_t mo, uint8_t dayt, uint8_t dayo, uint8_t weekday);
|
||
|
void set_contrast(void); /* set contrast to gui_data.contrast, value 1..7, 0 is default (do not set) */
|
||
|
|
||
|
|
||
|
/*===================================================*/
|
||
|
|
||
|
/*
|
||
|
Prototype:
|
||
|
uint8_t is_leap_year(uint16_t y)
|
||
|
Description:
|
||
|
Calculate leap year
|
||
|
Arguments:
|
||
|
y year, e.g. 2011 for year 2011
|
||
|
Result:
|
||
|
0 not a leap year
|
||
|
1 leap year
|
||
|
*/
|
||
|
uint8_t is_leap_year(uint16_t y)
|
||
|
{
|
||
|
if (
|
||
|
((y % 4 == 0) && (y % 100 != 0)) ||
|
||
|
(y % 400 == 0)
|
||
|
)
|
||
|
return 1;
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Prototype:
|
||
|
uint16_t get_year_day_number(uint16_t y, uint8_t m, uint8_t d)
|
||
|
Description:
|
||
|
Calculate the day number within a year. 1st of Jan has the number 1.
|
||
|
"Robertson" Algorithm IDAY (CACM Vol 15/#10/Oct 1972)
|
||
|
Arguments:
|
||
|
y year, e.g. 2011 for year 2011
|
||
|
m month with 1 = january to 12 = december
|
||
|
d day starting with 1
|
||
|
Result:
|
||
|
The "day number" within the year: 1 for the 1st of Jan.
|
||
|
See also:
|
||
|
get_month_by_day_number()
|
||
|
|
||
|
*/
|
||
|
uint16_t get_year_day_number(uint16_t y, uint8_t m, uint8_t d)
|
||
|
{
|
||
|
uint8_t tmp1;
|
||
|
uint16_t tmp2;
|
||
|
tmp1 = 0;
|
||
|
if ( m >= 3 )
|
||
|
tmp1++;
|
||
|
tmp2 = m;
|
||
|
tmp2 +=2;
|
||
|
tmp2 *=611;
|
||
|
tmp2 /= 20;
|
||
|
tmp2 += d;
|
||
|
tmp2 -= 91;
|
||
|
tmp1 <<=1;
|
||
|
tmp2 -= tmp1;
|
||
|
if ( tmp1 != 0 )
|
||
|
tmp2 += is_leap_year(y);
|
||
|
return tmp2;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Prototype:
|
||
|
uint8_t get_month_by_year_day_number(uint16_t y, uint16_t ydn)
|
||
|
Description:
|
||
|
Get the month from year and day number within a year.
|
||
|
"R. A. Stone" Algorithm (CACM Vol 13/#10/Oct 1970)
|
||
|
Arguments:
|
||
|
y year, e.g. 2011 for year 2011
|
||
|
ydn year day number (1st of Jan has the number 1)
|
||
|
Result:
|
||
|
The month within the year: 1 for January.
|
||
|
See also:
|
||
|
get_year_day_number()
|
||
|
*/
|
||
|
|
||
|
static uint16_t corrected_year_day_number(uint16_t y, uint16_t ydn)
|
||
|
{
|
||
|
uint8_t a;
|
||
|
a = is_leap_year(y);
|
||
|
if ( ydn > 59+a )
|
||
|
{
|
||
|
ydn += 2;
|
||
|
ydn -= a;
|
||
|
}
|
||
|
ydn += 91;
|
||
|
return ydn;
|
||
|
}
|
||
|
|
||
|
uint8_t get_month_by_year_day_number(uint16_t y, uint16_t ydn)
|
||
|
{
|
||
|
uint8_t a;
|
||
|
ydn = corrected_year_day_number(y, ydn);
|
||
|
ydn *= 20;
|
||
|
ydn /= 611;
|
||
|
a = ydn;
|
||
|
a -= 2;
|
||
|
return a;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Prototype:
|
||
|
uint8_t get_day_by_year_day_number(uint16_t y, uint16_t ydn)
|
||
|
Description:
|
||
|
Get the day within month from year and day number within a year.
|
||
|
"R. A. Stone" Algorithm (CACM Vol 13/#10/Oct 1970)
|
||
|
Arguments:
|
||
|
y year, e.g. 2011 for year 2011
|
||
|
ydn year day number (1st of Jan has the number 1)
|
||
|
Result:
|
||
|
The day within a month: 1 for the first day of a month.
|
||
|
See also:
|
||
|
get_year_day_number()
|
||
|
*/
|
||
|
uint8_t get_day_by_year_day_number(uint16_t y, uint16_t ydn)
|
||
|
{
|
||
|
uint8_t m;
|
||
|
uint16_t tmp;
|
||
|
m = get_month_by_year_day_number(y, ydn);
|
||
|
m += 2;
|
||
|
ydn = corrected_year_day_number(y, ydn);
|
||
|
tmp = 611;
|
||
|
tmp *= m;
|
||
|
tmp /= 20;
|
||
|
ydn -= tmp;
|
||
|
return ydn;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Prototype:
|
||
|
uint8_t get_weekday_by_year_day_number(uint16_t y, uint16_t ydn)
|
||
|
Description:
|
||
|
Get the day within week from year and day number within a year.
|
||
|
"Zeller" Algorithm
|
||
|
https://de.wikisource.org/wiki/Index:Acta_Mathematica_vol._009_(1886)
|
||
|
https://ia801407.us.archive.org/8/items/actamathematica09upps/actamathematica09upps.pdf
|
||
|
Arguments:
|
||
|
y year, e.g. 2011 for year 2011
|
||
|
ydn year day number (1st of Jan has the number 1)
|
||
|
Result:
|
||
|
The day within a week: 0..6 with 0 = Sunday, 1 = Monday, ...
|
||
|
See also:
|
||
|
get_year_day_number()
|
||
|
*/
|
||
|
uint8_t get_weekday_by_year_day_number(uint16_t y, uint16_t ydn)
|
||
|
{
|
||
|
uint8_t j, c, tmp8;
|
||
|
uint16_t tmp16;
|
||
|
y--;
|
||
|
j = y % 100;
|
||
|
c = y / 100;
|
||
|
tmp16 = c;
|
||
|
tmp16 *= 5;
|
||
|
tmp16 += ydn;
|
||
|
tmp8 = j;
|
||
|
j >>= 2;
|
||
|
c >>= 2;
|
||
|
tmp8 += j;
|
||
|
tmp8 += c;
|
||
|
tmp8 += 28;
|
||
|
tmp16 += tmp8;
|
||
|
tmp16 %= 7;
|
||
|
return tmp16;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Prototype:
|
||
|
uint16_t to_century_day_number(uint16_t y, uint16_t ydn)
|
||
|
Description:
|
||
|
Calculate days since January, 1st, 2000
|
||
|
Arguments:
|
||
|
y year, e.g. 2011 for year 2011
|
||
|
ydn year day number (1st of Jan has the number 1)
|
||
|
*/
|
||
|
uint16_t to_century_day_number(uint16_t y, uint16_t ydn)
|
||
|
{
|
||
|
uint16_t cdn;
|
||
|
cdn = ydn;
|
||
|
cdn--;
|
||
|
while( y > 2000 )
|
||
|
{
|
||
|
y--;
|
||
|
cdn += 365;
|
||
|
cdn += is_leap_year(y);
|
||
|
}
|
||
|
return cdn;
|
||
|
}
|
||
|
|
||
|
void from_century_day_number(uint16_t cdn, uint16_t *year, uint16_t *ydn)
|
||
|
{
|
||
|
uint16_t y, days_per_year;
|
||
|
y = 2000;
|
||
|
for(;;)
|
||
|
{
|
||
|
days_per_year = 365;
|
||
|
days_per_year += is_leap_year(y);
|
||
|
if ( cdn >= days_per_year )
|
||
|
{
|
||
|
cdn -= days_per_year;
|
||
|
y++;
|
||
|
}
|
||
|
else
|
||
|
break;
|
||
|
}
|
||
|
cdn++;
|
||
|
*year = y;
|
||
|
*ydn = cdn;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Calculate the seconds after 2000-01-01 00:00. The largest possible
|
||
|
time is 2136-02-07 06:28:15
|
||
|
*/
|
||
|
uint32_t to_time(uint16_t cdn, uint8_t h, uint8_t m, uint8_t s)
|
||
|
{
|
||
|
uint32_t t;
|
||
|
t = cdn;
|
||
|
t *= 24;
|
||
|
t += h;
|
||
|
t *= 60;
|
||
|
t += m;
|
||
|
t *= 60;
|
||
|
t += s;
|
||
|
return t;
|
||
|
}
|
||
|
|
||
|
|
||
|
void from_time(uint32_t t, uint16_t *cdn, uint8_t *h, uint8_t *m, uint8_t *s)
|
||
|
{
|
||
|
*s = t % 60;
|
||
|
t /= 60;
|
||
|
*m = t % 60;
|
||
|
t /= 60;
|
||
|
*h = t % 24;
|
||
|
t /= 24;
|
||
|
*cdn = t;
|
||
|
}
|
||
|
|
||
|
uint32_t to_sec_since_2000(uint16_t y, uint8_t mo, uint8_t d, uint8_t h, uint8_t mi, uint8_t s)
|
||
|
{
|
||
|
uint16_t ydn = get_year_day_number(y, mo, d);
|
||
|
uint16_t cdn = to_century_day_number(y, ydn);
|
||
|
return to_time(cdn, h, mi, s);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Calculate the minutes after 2000-01-01 00:00.
|
||
|
*/
|
||
|
uint32_t to_minutes(uint16_t cdn, uint8_t h, uint8_t m)
|
||
|
{
|
||
|
uint32_t t;
|
||
|
t = cdn;
|
||
|
t *= 24;
|
||
|
t += h;
|
||
|
t *= 60;
|
||
|
t += m;
|
||
|
return t;
|
||
|
}
|
||
|
|
||
|
|
||
|
void from_minutes(uint32_t t, uint16_t *cdn, uint8_t *h, uint8_t *m)
|
||
|
{
|
||
|
*m = t % 60;
|
||
|
t /= 60;
|
||
|
*h = t % 24;
|
||
|
t /= 24;
|
||
|
*cdn = t;
|
||
|
}
|
||
|
|
||
|
uint32_t to_minutes_since_2000(uint16_t y, uint8_t mo, uint8_t d, uint8_t h, uint8_t mi)
|
||
|
{
|
||
|
uint16_t ydn = get_year_day_number(y, mo, d);
|
||
|
uint16_t cdn = to_century_day_number(y, ydn);
|
||
|
return to_minutes(cdn, h, mi);
|
||
|
}
|
||
|
|
||
|
/*===================================================*/
|
||
|
|
||
|
/* common menu functions */
|
||
|
|
||
|
void menu_DrawEdgePixel(menu_t *menu, u8g2_uint_t x, u8g2_uint_t y, u8g2_uint_t w, u8g2_uint_t h)
|
||
|
{
|
||
|
w--;
|
||
|
h--;
|
||
|
u8g2_DrawPixel(menu->u8g2, x,y);
|
||
|
u8g2_DrawPixel(menu->u8g2, x+w,y);
|
||
|
u8g2_DrawPixel(menu->u8g2, x,y+h);
|
||
|
u8g2_DrawPixel(menu->u8g2, x+w,y+h);
|
||
|
}
|
||
|
|
||
|
void menu_ClearEdgePixel(menu_t *menu, u8g2_uint_t x, u8g2_uint_t y, u8g2_uint_t w, u8g2_uint_t h)
|
||
|
{
|
||
|
u8g2_SetDrawColor(menu->u8g2, 0);
|
||
|
menu_DrawEdgePixel(menu, x, y, w, h);
|
||
|
u8g2_SetDrawColor(menu->u8g2, 1);
|
||
|
}
|
||
|
|
||
|
void menu_DrawBoxFocus(menu_t *menu, u8g2_uint_t x, u8g2_uint_t y, u8g2_uint_t w, u8g2_uint_t h)
|
||
|
{
|
||
|
u8g2_SetDrawColor(menu->u8g2, 2);
|
||
|
u8g2_DrawBox(menu->u8g2, x, y, w, h);
|
||
|
menu_ClearEdgePixel(menu, x, y, w, h);
|
||
|
}
|
||
|
|
||
|
void menu_DrawFrameFocus(menu_t *menu, u8g2_uint_t x, u8g2_uint_t y, u8g2_uint_t w, u8g2_uint_t h)
|
||
|
{
|
||
|
menu_DrawEdgePixel(menu, x, y, w, h);
|
||
|
x--;
|
||
|
y--;
|
||
|
w+=2;
|
||
|
h+=2;
|
||
|
u8g2_DrawFrame(menu->u8g2, x, y, w, h);
|
||
|
menu_ClearEdgePixel(menu, x, y, w, h);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*===================================================*/
|
||
|
// menu callbacks
|
||
|
|
||
|
/* this function must be the last function in the list. it also marks the end of a list */
|
||
|
int me_cb_null(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_big_toggle
|
||
|
Val: uint8_t *
|
||
|
Arg: Not used
|
||
|
*/
|
||
|
int me_cb_big_toggle(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
uint8_t val = *(uint8_t *)(me->val);
|
||
|
u8g2_uint_t x, y, w, h, w2;
|
||
|
|
||
|
w = 16;
|
||
|
w2 = 6;
|
||
|
h = 10;
|
||
|
x = me->x;
|
||
|
y = me->y;
|
||
|
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_IS_FOCUS:
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW_FOCUS:
|
||
|
//menu_DrawFrameFocus(menu, x-1, y-1, w+2, h+2);
|
||
|
menu_DrawFrameFocus(menu, x, y, w, h);
|
||
|
return 1;
|
||
|
case ME_MSG_SELECT:
|
||
|
{
|
||
|
val++;
|
||
|
if ( val > 1 )
|
||
|
val = 0;
|
||
|
*(uint8_t *)(me->val) = val;
|
||
|
}
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW:
|
||
|
menu_DrawFrameFocus(menu, x+1,y+1,w-2,h-2);
|
||
|
if ( val == 0 )
|
||
|
{
|
||
|
menu_DrawFrameFocus(menu, x+3,y+3,w2-2,h-6);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
menu_DrawBoxFocus(menu, x+w/2,y+2,w2,h-4);
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_wd_toggle
|
||
|
Val: uint8_t *
|
||
|
Arg: char *
|
||
|
*/
|
||
|
int me_cb_wd_toggle(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
uint8_t val = *(uint8_t *)(me->val);
|
||
|
u8g2_uint_t x, y, w, h;
|
||
|
|
||
|
u8g2_SetFont(menu->u8g2, MENU_SMALL_FONT);
|
||
|
|
||
|
w = 13;
|
||
|
h = u8g2_GetAscent(menu->u8g2)+2;
|
||
|
x = me->x-2;
|
||
|
y = me->y - u8g2_GetAscent(menu->u8g2)-1;
|
||
|
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_IS_FOCUS:
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW_FOCUS:
|
||
|
menu_DrawFrameFocus(menu, x, y, w, h);
|
||
|
return 1;
|
||
|
case ME_MSG_SELECT:
|
||
|
{
|
||
|
val++;
|
||
|
if ( val > 1 )
|
||
|
val = 0;
|
||
|
*(uint8_t *)(me->val) = val;
|
||
|
}
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW:
|
||
|
u8g2_DrawUTF8(menu->u8g2, me->x, me->y, (const char *)(me->arg));
|
||
|
if ( val > 0 )
|
||
|
{
|
||
|
menu_DrawBoxFocus(menu, x,y,w,h);
|
||
|
}
|
||
|
//u8g2_DrawRFrame(menu->u8g2, x, y, w, h, 1);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_0_9
|
||
|
Val: uint8_t *
|
||
|
*/
|
||
|
int me_cb_0_9(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_IS_FOCUS:
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW_FOCUS:
|
||
|
u8g2_SetFont(menu->u8g2, MENU_BIG_NUM);
|
||
|
menu_DrawBoxFocus(menu,
|
||
|
me->x+MENU_BIG_NUM_FOCUS_XO,
|
||
|
me->y - u8g2_GetAscent(menu->u8g2)-1,
|
||
|
u8g2_GetGlyphWidth(menu->u8g2, '0')+MENU_BIG_NUM_FOCUS_EXTRAX,
|
||
|
u8g2_GetAscent(menu->u8g2) + 2);
|
||
|
return 1;
|
||
|
case ME_MSG_SELECT:
|
||
|
{
|
||
|
uint8_t val = *(uint8_t *)(me->val);
|
||
|
val++;
|
||
|
if ( val > 9 )
|
||
|
val = 0;
|
||
|
*(uint8_t *)(me->val) = val;
|
||
|
}
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW:
|
||
|
u8g2_SetFont(menu->u8g2, MENU_BIG_NUM);
|
||
|
u8g2_DrawGlyph(menu->u8g2, me->x, me->y, *(uint8_t *)(me->val) + '0');
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int me_cb_0_9_ro(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_IS_FOCUS )
|
||
|
return 0;
|
||
|
return me_cb_0_9(menu, me, msg);
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_0_5
|
||
|
Val: uint8_t *
|
||
|
*/
|
||
|
int me_cb_0_5(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_SELECT:
|
||
|
{
|
||
|
uint8_t val = *(uint8_t *)(me->val);
|
||
|
val++;
|
||
|
if ( val > 5 )
|
||
|
val = 0;
|
||
|
*(uint8_t *)(me->val) = val;
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
return me_cb_0_9(menu, me, msg);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_0_23
|
||
|
Val: uint8_t *
|
||
|
*/
|
||
|
int me_cb_0_23(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
char s[4];
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_IS_FOCUS:
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW_FOCUS:
|
||
|
u8g2_SetFont(menu->u8g2, MENU_BIG_NUM);
|
||
|
menu_DrawBoxFocus(menu,
|
||
|
me->x+MENU_BIG_NUM_FOCUS_XO,
|
||
|
me->y - u8g2_GetAscent(menu->u8g2)-1,
|
||
|
u8g2_GetGlyphWidth(menu->u8g2, '0')*2+MENU_BIG_NUM_FOCUS_EXTRAX,
|
||
|
u8g2_GetAscent(menu->u8g2) + 2);
|
||
|
return 1;
|
||
|
case ME_MSG_SELECT:
|
||
|
{
|
||
|
uint8_t val = *(uint8_t *)(me->val);
|
||
|
val++;
|
||
|
if ( val > 23 )
|
||
|
val = 0;
|
||
|
*(uint8_t *)(me->val) = val;
|
||
|
}
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW:
|
||
|
u8g2_SetFont(menu->u8g2, MENU_BIG_NUM);
|
||
|
s[0] = *(uint8_t *)(me->val);
|
||
|
s[1] = s[0];
|
||
|
s[1] %= 10;
|
||
|
s[1] += '0';
|
||
|
s[0] /= 10;
|
||
|
s[0] += '0';
|
||
|
s[2] = '\0';
|
||
|
u8g2_DrawUTF8(menu->u8g2, me->x, me->y, s);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int me_cb_0_23_ro(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_IS_FOCUS )
|
||
|
return 0;
|
||
|
return me_cb_0_23(menu, me, msg);
|
||
|
}
|
||
|
|
||
|
|
||
|
int me_cb_0_9_small_ro(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_IS_FOCUS:
|
||
|
return 0;
|
||
|
case ME_MSG_DRAW_FOCUS:
|
||
|
return 1;
|
||
|
case ME_MSG_SELECT:
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW:
|
||
|
u8g2_SetFont(menu->u8g2, MENU_SMALL_FONT);
|
||
|
u8g2_DrawGlyph(menu->u8g2, me->x, me->y, *(uint8_t *)(me->val) + '0');
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_0_55
|
||
|
Val: uint8_t *
|
||
|
*/
|
||
|
int me_cb_0_55(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_SELECT:
|
||
|
{
|
||
|
uint8_t val = *(uint8_t *)(me->val);
|
||
|
val+=5;
|
||
|
if ( val > 55 )
|
||
|
val = 0;
|
||
|
*(uint8_t *)(me->val) = val;
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
return me_cb_0_23(menu, me, msg);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_1_12
|
||
|
Val: uint8_t *
|
||
|
*/
|
||
|
int me_cb_1_12(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_SELECT:
|
||
|
{
|
||
|
uint8_t val = *(uint8_t *)(me->val);
|
||
|
val++;
|
||
|
if ( val > 12 )
|
||
|
val = 1;
|
||
|
*(uint8_t *)(me->val) = val;
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
return me_cb_0_23(menu, me, msg);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_1_31
|
||
|
Val: uint8_t *
|
||
|
*/
|
||
|
int me_cb_1_31(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_SELECT:
|
||
|
{
|
||
|
uint8_t val = *(uint8_t *)(me->val);
|
||
|
val++;
|
||
|
if ( val > 31 )
|
||
|
val = 1;
|
||
|
*(uint8_t *)(me->val) = val;
|
||
|
}
|
||
|
return 1;
|
||
|
}
|
||
|
return me_cb_0_23(menu, me, msg);
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_num_label
|
||
|
can not get focus
|
||
|
Arg: char *
|
||
|
*/
|
||
|
int me_cb_num_label(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_IS_FOCUS:
|
||
|
case ME_MSG_DRAW_FOCUS:
|
||
|
case ME_MSG_SELECT:
|
||
|
break;
|
||
|
case ME_MSG_DRAW:
|
||
|
u8g2_SetFont(menu->u8g2, MENU_BIG_NUM);
|
||
|
u8g2_DrawUTF8(menu->u8g2, me->x, me->y, (char *)(me->arg) );
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_button_full_line
|
||
|
Val: callback function
|
||
|
Arg: char *
|
||
|
*/
|
||
|
int me_cb_button_full_line(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
int r = 0;
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_IS_FOCUS:
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW_FOCUS:
|
||
|
menu_DrawBoxFocus(menu,
|
||
|
0,
|
||
|
me->y - u8g2_GetAscent(menu->u8g2)-1,
|
||
|
u8g2_GetDisplayWidth(menu->u8g2) ,
|
||
|
u8g2_GetAscent(menu->u8g2) - u8g2_GetDescent(menu->u8g2) +1);
|
||
|
r = 1;
|
||
|
break;
|
||
|
case ME_MSG_DRAW:
|
||
|
u8g2_SetFont(menu->u8g2, MENU_NORMAL_FONT);
|
||
|
u8g2_DrawUTF8(menu->u8g2, me->x, me->y, (char *)(me->arg) );
|
||
|
r = 1;
|
||
|
break;
|
||
|
}
|
||
|
/* pass all messages except for the IS_FOCUS also to the callback function */
|
||
|
if ( me->val != NULL )
|
||
|
return ((me_cb)(me->val))(menu, me, msg) | r;
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_button_full_line
|
||
|
Val: callback function
|
||
|
Arg: char *
|
||
|
*/
|
||
|
int me_cb_button_half_line(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
int r = 0;
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_IS_FOCUS:
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW_FOCUS:
|
||
|
menu_DrawBoxFocus(menu,
|
||
|
me->x,
|
||
|
me->y - u8g2_GetAscent(menu->u8g2)-1,
|
||
|
u8g2_GetDisplayWidth(menu->u8g2)/2,
|
||
|
u8g2_GetAscent(menu->u8g2) - u8g2_GetDescent(menu->u8g2) +1);
|
||
|
r = 1;
|
||
|
break;
|
||
|
case ME_MSG_DRAW:
|
||
|
u8g2_SetFont(menu->u8g2, MENU_NORMAL_FONT);
|
||
|
u8g2_DrawUTF8(menu->u8g2, me->x+4, me->y, (char *)(me->arg) );
|
||
|
r = 1;
|
||
|
break;
|
||
|
}
|
||
|
/* pass all messages except for the IS_FOCUS also to the callback function */
|
||
|
if ( me->val != NULL )
|
||
|
return ((me_cb)(me->val))(menu, me, msg) | r;
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_button_empty
|
||
|
Val: callback function
|
||
|
Arg: not used
|
||
|
*/
|
||
|
int me_cb_button_empty(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_IS_FOCUS )
|
||
|
return 1;
|
||
|
if ( me->val != NULL )
|
||
|
return ((me_cb)(me->val))(menu, me, msg);
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_scale_1_7
|
||
|
Val: uint8_t *
|
||
|
*/
|
||
|
|
||
|
void set_contrast(void);
|
||
|
|
||
|
|
||
|
int me_cb_scale_1_7(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
u8g2_uint_t x;
|
||
|
uint8_t val = *(uint8_t *)(me->val);
|
||
|
|
||
|
if ( val <= 0 )
|
||
|
val = 1;
|
||
|
x = me->x+(val-1)*5;
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_IS_FOCUS:
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW_FOCUS:
|
||
|
/*
|
||
|
u8g2_SetFont(menu->u8g2, MENU_BIG_NUM);
|
||
|
menu_DrawBoxFocus(menu,
|
||
|
me->x+MENU_BIG_NUM_FOCUS_XO,
|
||
|
me->y - u8g2_GetAscent(menu->u8g2)-1,
|
||
|
u8g2_GetGlyphWidth(menu->u8g2, '0')+MENU_BIG_NUM_FOCUS_EXTRAX,
|
||
|
u8g2_GetAscent(menu->u8g2) + 2);
|
||
|
*/
|
||
|
u8g2_DrawBox(menu->u8g2, x-2 , me->y-2, 5, 5);
|
||
|
return 1;
|
||
|
case ME_MSG_SELECT:
|
||
|
{
|
||
|
val++;
|
||
|
if ( val > 7 )
|
||
|
val = 1;
|
||
|
*(uint8_t *)(me->val) = val;
|
||
|
}
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW:
|
||
|
set_contrast(); /* give user feedback... not so nice: We assume *(uint8_t *)(me->val) points to gui_data.contrast */
|
||
|
|
||
|
//u8g2_SetFont(menu->u8g2, MENU_NORMAL_FONT);
|
||
|
//u8g2_DrawGlyph(menu->u8g2, me->x, me->y-2, *(uint8_t *)(me->val) + '0');
|
||
|
|
||
|
u8g2_DrawHLine(menu->u8g2, me->x, me->y, 6*5+1);
|
||
|
|
||
|
u8g2_DrawVLine(menu->u8g2, me->x, me->y-2, 5);
|
||
|
//u8g2_DrawVLine(menu->u8g2, me->x+1*5, me->y-1, 3);
|
||
|
//u8g2_DrawVLine(menu->u8g2, me->x+2*5, me->y-1, 3);
|
||
|
u8g2_DrawVLine(menu->u8g2, me->x+3*5, me->y-2, 5);
|
||
|
//u8g2_DrawVLine(menu->u8g2, me->x+4*5, me->y-1, 3);
|
||
|
//u8g2_DrawVLine(menu->u8g2, me->x+5*5, me->y-1, 3);
|
||
|
u8g2_DrawVLine(menu->u8g2, me->x+6*5, me->y-2, 5);
|
||
|
|
||
|
u8g2_DrawFrame(menu->u8g2, x-3 , me->y-3, 7, 7);
|
||
|
u8g2_SetDrawColor(menu->u8g2, 0);
|
||
|
u8g2_DrawBox(menu->u8g2, x-2 , me->y-2, 5, 5);
|
||
|
/* draw color is set to 1 in the following function */
|
||
|
menu_ClearEdgePixel(menu, x-3 , me->y-3, 7, 7);
|
||
|
menu_DrawEdgePixel(menu, x-2 , me->y-3+1, 5, 5);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_label
|
||
|
can not get focus
|
||
|
Arg: char *
|
||
|
*/
|
||
|
int me_cb_label(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_IS_FOCUS:
|
||
|
case ME_MSG_DRAW_FOCUS:
|
||
|
case ME_MSG_SELECT:
|
||
|
break;
|
||
|
case ME_MSG_DRAW:
|
||
|
u8g2_SetFont(menu->u8g2, MENU_NORMAL_FONT);
|
||
|
u8g2_DrawUTF8(menu->u8g2, me->x, me->y, (char *)(me->arg) );
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int me_cb_inv_label(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
int r = me_cb_label(menu, me, msg);
|
||
|
if ( msg == ME_MSG_DRAW )
|
||
|
{
|
||
|
menu_DrawBoxFocus(menu,
|
||
|
me->x-1,
|
||
|
me->y - u8g2_GetAscent(menu->u8g2)-1,
|
||
|
u8g2_GetUTF8Width(menu->u8g2, (char *)(me->arg))+2,
|
||
|
u8g2_GetAscent(menu->u8g2) + 2);
|
||
|
}
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
Name: me_cb_button_full_line
|
||
|
Val: callback function
|
||
|
Arg: bitmap
|
||
|
*/
|
||
|
int me_cb_16x16_bitmap_button(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
int r = 0;
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_IS_FOCUS:
|
||
|
return 1;
|
||
|
case ME_MSG_DRAW_FOCUS:
|
||
|
/*
|
||
|
menu_DrawFrameFocus(menu,
|
||
|
me->x-1,
|
||
|
me->y-1,
|
||
|
16+2,
|
||
|
16+2);
|
||
|
*/
|
||
|
menu_DrawFrameFocus(menu,
|
||
|
me->x,
|
||
|
me->y,
|
||
|
16,
|
||
|
16);
|
||
|
|
||
|
r = 1;
|
||
|
break;
|
||
|
case ME_MSG_DRAW:
|
||
|
u8g2_DrawXBM(menu->u8g2, me->x, me->y, 16, 16, (const uint8_t *)(me->arg));
|
||
|
r = 1;
|
||
|
break;
|
||
|
}
|
||
|
/* pass all messages except for the IS_FOCUS also to the callback function */
|
||
|
if ( me->val != NULL )
|
||
|
return ((me_cb)(me->val))(menu, me, msg) | r;
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
/*===================================================*/
|
||
|
// menu API
|
||
|
|
||
|
/* call menu element from menu->current_index */
|
||
|
int menu_CallME(menu_t *menu, uint8_t msg)
|
||
|
{
|
||
|
const me_t *me;
|
||
|
me = menu->me_list+menu->current_index;
|
||
|
return me->cb(menu, me, msg);
|
||
|
}
|
||
|
|
||
|
/* stay on current focus if valid, move to next valid focus */
|
||
|
static void menu_CalcNextValidFocus(menu_t *menu) U8G2_NOINLINE;
|
||
|
static void menu_CalcNextValidFocus(menu_t *menu)
|
||
|
{
|
||
|
for(;;)
|
||
|
{
|
||
|
menu->current_index = menu->focus_index;
|
||
|
if ( menu->current_index >= menu->me_count )
|
||
|
break;
|
||
|
if ( menu_CallME(menu, ME_MSG_IS_FOCUS) != 0 )
|
||
|
break;
|
||
|
menu->focus_index++;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/* advance current focus to the next element */
|
||
|
void menu_NextFocus(menu_t *menu)
|
||
|
{
|
||
|
menu->focus_index++;
|
||
|
if ( menu->focus_index >= menu->me_count )
|
||
|
menu->focus_index = 0;
|
||
|
menu_CalcNextValidFocus(menu);
|
||
|
}
|
||
|
|
||
|
/* send select message to the element which has the current focus */
|
||
|
void menu_Select(menu_t *menu)
|
||
|
{
|
||
|
menu->current_index = menu->focus_index;
|
||
|
menu_CallME(menu, ME_MSG_SELECT);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void menu_SetMEList(menu_t *menu, const me_t *me_list, uint16_t initial_focus)
|
||
|
{
|
||
|
menu->me_list = me_list;
|
||
|
|
||
|
menu->me_count = 0;
|
||
|
while( me_list[menu->me_count].cb != me_cb_null )
|
||
|
menu->me_count++;
|
||
|
|
||
|
menu->focus_index = 0;
|
||
|
menu_CalcNextValidFocus(menu);
|
||
|
|
||
|
while( initial_focus > 0 )
|
||
|
{
|
||
|
menu_NextFocus(menu);
|
||
|
initial_focus--;
|
||
|
}
|
||
|
|
||
|
menu->radio_index = menu->me_count;
|
||
|
}
|
||
|
|
||
|
me_t melist_emty[] =
|
||
|
{
|
||
|
{ me_cb_null, NULL, 0, 0 }
|
||
|
};
|
||
|
|
||
|
|
||
|
void menu_Init(menu_t *menu, u8g2_t *u8g2)
|
||
|
{
|
||
|
menu->u8g2 = u8g2;
|
||
|
menu_SetMEList(menu, melist_emty, 0);
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void menu_Draw(menu_t *menu)
|
||
|
{
|
||
|
for( menu->current_index = 0; menu->current_index < menu->me_count; menu->current_index++ )
|
||
|
{
|
||
|
menu_CallME(menu, ME_MSG_DRAW);
|
||
|
if ( menu->current_index == menu->focus_index )
|
||
|
{
|
||
|
menu_CallME(menu, ME_MSG_DRAW_FOCUS);
|
||
|
}
|
||
|
}
|
||
|
// u8g2_DrawHLine(menu->u8g2, 0, 32, 128);
|
||
|
}
|
||
|
|
||
|
/*===================================================*/
|
||
|
|
||
|
/*============================================*/
|
||
|
/* reset */
|
||
|
void do_reset(void)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
/*============================================*/
|
||
|
/* load & store from/to permanent memory */
|
||
|
|
||
|
|
||
|
/* 5x 32bit */
|
||
|
|
||
|
void store_gui_data(uint32_t *data)
|
||
|
{
|
||
|
/*
|
||
|
FILE *fp;
|
||
|
fp = fopen("clock.dat", "w");
|
||
|
if ( fp != NULL )
|
||
|
{
|
||
|
fwrite(data, sizeof(uint32_t), 5, fp);
|
||
|
fclose(fp);
|
||
|
}
|
||
|
*/
|
||
|
}
|
||
|
|
||
|
void load_gui_data(uint32_t *data)
|
||
|
{
|
||
|
/*
|
||
|
FILE *fp;
|
||
|
|
||
|
int i;
|
||
|
|
||
|
for( i = 0; i < GUI_ALARM_CNT; i++ )
|
||
|
data[i] = 0;
|
||
|
|
||
|
fp = fopen("clock.dat", "r");
|
||
|
if ( fp != NULL )
|
||
|
{
|
||
|
fread(data, sizeof(uint32_t), 5, fp);
|
||
|
fclose(fp);
|
||
|
}
|
||
|
*/
|
||
|
}
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
/* input */
|
||
|
|
||
|
int is_por_reset(void)
|
||
|
{
|
||
|
return 1;
|
||
|
}
|
||
|
|
||
|
int is_button_reset(void)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
uint32_t get_boot_status_register(void)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
uint32_t get_pwr_status_register(void)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
uint32_t get_reset_reason(void)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
uint32_t get_wakeup_count(void)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
uint32_t get_dst_by_date(void)
|
||
|
{
|
||
|
return -1;
|
||
|
}
|
||
|
|
||
|
uint32_t get_dst_by_RTC(void)
|
||
|
{
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
/*============================================*/
|
||
|
/* output */
|
||
|
|
||
|
void enable_alarm(void)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void disable_alarm(void)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void set_time(uint8_t ht, uint8_t ho, uint8_t mt, uint8_t mo, uint8_t st, uint8_t so)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
void set_date(uint8_t yt, uint8_t yo, uint8_t mt, uint8_t mo, uint8_t dayt, uint8_t dayo, uint8_t weekday)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
/* set contrast to gui_data.contrast, value 1..7, 0 is default (do not set) */
|
||
|
void set_contrast(void)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
/*===================================================*/
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
|
||
|
/* global variable for the gui menues */
|
||
|
|
||
|
uint8_t gui_alarm_index = 0;
|
||
|
gui_alarm_t gui_alarm_current;
|
||
|
|
||
|
const char weekdaystr[7][4] = {
|
||
|
"Mo", "Di", "Mi", "Do", "Fr", "Sa", "So"
|
||
|
};
|
||
|
|
||
|
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
|
||
|
const static uint8_t ok_xbm[] = { /* 16x16 */
|
||
|
0xfe, 0x7f, 0x03, 0xc0, 0x01, 0x80, 0x01, 0xb8, 0x01, 0x9c, 0x01, 0x8e,
|
||
|
0x01, 0x87, 0x01, 0x87, 0x9d, 0x83, 0xb9, 0x83, 0xf1, 0x81, 0xe1, 0x81,
|
||
|
0xc1, 0x80, 0x01, 0x80, 0x03, 0xc0, 0xfe, 0x7f };
|
||
|
|
||
|
const static uint8_t alarm_xbm[] = { /* 12x12 */
|
||
|
0x00, 0x00, 0x0c, 0x06, 0xf6, 0x0d, 0x1a, 0x0b, 0x4c, 0x06, 0x44, 0x04,
|
||
|
0xc4, 0x05, 0x04, 0x04, 0x0c, 0x06, 0x18, 0x03, 0xf0, 0x01, 0x00, 0x00 };
|
||
|
|
||
|
|
||
|
int me_action_to_top_menu(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_top_menu, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int me_action_to_setup_menu(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_setup_menu, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
int me_action_save_time(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
set_time(gui_data.h / 10, gui_data.h % 10, gui_data.mt, gui_data.mo, gui_data.st, gui_data.so);
|
||
|
//menu_SetMEList(menu, melist_top_menu, 0); /* first set the normal menu */
|
||
|
menu_SetMEList(menu, melist_setup_menu, 0);
|
||
|
gui_Recalculate(); /* because it might be overwritten with the alarm menu */
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
#define ME_TIME_Y 26
|
||
|
|
||
|
|
||
|
#define ME_TIME_XO 11
|
||
|
const me_t melist_setup_time[] =
|
||
|
{
|
||
|
{ me_cb_0_23, &gui_data.h, NULL, 1,ME_TIME_Y },
|
||
|
{ me_cb_num_label, NULL, (void *)":", 37,ME_TIME_Y-3 },
|
||
|
{ me_cb_0_5, &gui_data.mt, NULL, 45,ME_TIME_Y },
|
||
|
{ me_cb_0_9, &gui_data.mo, NULL, 63,ME_TIME_Y },
|
||
|
{ me_cb_num_label, NULL, (void *)":", 81,ME_TIME_Y-3 },
|
||
|
{ me_cb_0_5, &gui_data.st, NULL, 89,ME_TIME_Y },
|
||
|
{ me_cb_0_9, &gui_data.so, NULL, 107,ME_TIME_Y },
|
||
|
{ me_cb_button_half_line, (void *)me_action_to_setup_menu, (void *)"Abbrechen", 0,42 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_save_time, (void *)"Speichern", 64,42 },
|
||
|
|
||
|
//{ me_cb_button_full_line, (void *)me_action_save_time, "Speichern", 40,42 },
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
/* Display Time */
|
||
|
|
||
|
void gui_alarm_to_str(uint8_t idx)
|
||
|
{
|
||
|
strcpy(gui_data.s, weekdaystr[gui_alarm_list[gui_data.next_alarm_index].na_wd]);
|
||
|
gui_data.s[2] = ',';
|
||
|
gui_data.s[3] = ' ';
|
||
|
strcpy(gui_data.s+4, u8x8_u8toa(gui_alarm_list[gui_data.next_alarm_index].na_h, 2));
|
||
|
gui_data.s[6] = ':';
|
||
|
strcpy(gui_data.s+7, u8x8_u8toa(gui_alarm_list[gui_data.next_alarm_index].na_m, 2));
|
||
|
if ( gui_alarm_list[gui_data.next_alarm_index].snooze_count != 0 )
|
||
|
{
|
||
|
gui_data.s[9] = '+';
|
||
|
gui_data.s[10] = '\0';
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
gui_data.s[9] = '\0';
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int me_action_handle_display_time(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_DRAW )
|
||
|
{
|
||
|
char s[14];
|
||
|
u8g2_uint_t w;
|
||
|
|
||
|
|
||
|
u8g2_uint_t x = 27;
|
||
|
u8g2_uint_t y = 61;
|
||
|
u8g2_SetFont(menu->u8g2, MENU_LARGE_FONT);
|
||
|
|
||
|
if ( gui_data.next_alarm_index < GUI_ALARM_CNT )
|
||
|
{
|
||
|
gui_alarm_to_str(gui_data.next_alarm_index);
|
||
|
w = u8g2_GetUTF8Width(menu->u8g2, gui_data.s)+14;
|
||
|
//u8g2_DrawUTF8(menu->u8g2, x+14, y, gui_data.s);
|
||
|
u8g2_DrawXBM(menu->u8g2, (128-w)/2, y-11, 12, 12, (const uint8_t *)(alarm_xbm));
|
||
|
u8g2_DrawUTF8(menu->u8g2, (128-w)/2+14, y, gui_data.s);
|
||
|
}
|
||
|
|
||
|
y -= 17;
|
||
|
x -= 3;
|
||
|
|
||
|
strcpy(s, weekdaystr[gui_data.weekday]);
|
||
|
s[2] = ',';
|
||
|
s[3] = ' ';
|
||
|
strcpy(s+4, u8x8_u8toa(gui_data.day, 2));
|
||
|
s[6] = '.';
|
||
|
strcpy(s+7, u8x8_u8toa(gui_data.month, 2));
|
||
|
s[9] = '.';
|
||
|
s[10] = gui_data.year_t+'0';
|
||
|
s[11] = gui_data.year_o+'0';
|
||
|
s[12] = '\0';
|
||
|
|
||
|
w = u8g2_GetUTF8Width(menu->u8g2, s);
|
||
|
|
||
|
u8g2_DrawUTF8(menu->u8g2, (128-w)/2, y, s);
|
||
|
|
||
|
return 1;
|
||
|
}
|
||
|
else if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_top_menu, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
#define ME_TIME_DXO 29
|
||
|
const me_t melist_display_time[] =
|
||
|
{
|
||
|
{ me_cb_0_23_ro, &gui_data.h, NULL, ME_TIME_DXO-7,ME_TIME_Y },
|
||
|
{ me_cb_num_label, NULL, (void *)":", ME_TIME_DXO+30,ME_TIME_Y-3 },
|
||
|
{ me_cb_0_9_ro, &gui_data.mt, NULL, ME_TIME_DXO+39,ME_TIME_Y },
|
||
|
{ me_cb_0_9_ro, &gui_data.mo, NULL, ME_TIME_DXO+57,ME_TIME_Y },
|
||
|
|
||
|
{ me_cb_0_9_small_ro, &gui_data.st, NULL, 118,ME_TIME_Y },
|
||
|
{ me_cb_0_9_small_ro, &gui_data.so, NULL, 123,ME_TIME_Y },
|
||
|
|
||
|
|
||
|
{ me_cb_button_empty, (void *)me_action_handle_display_time, NULL, 0, 0 },
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
/*============================================*/
|
||
|
/* Date Edit Dialog */
|
||
|
|
||
|
int me_action_save_date(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
gui_date_adjust(); /* calculate the weekday */
|
||
|
|
||
|
set_date(gui_data.year_t, gui_data.year_o, gui_data.month / 10, gui_data.month % 10, gui_data.day / 10 , gui_data.day % 10, gui_data.weekday);
|
||
|
|
||
|
menu_SetMEList(menu, melist_setup_menu, 0); /* first set the normal menu */
|
||
|
gui_Recalculate(); /* because it might be overwritten with the alarm menu */
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
const me_t melist_setup_date[] =
|
||
|
{
|
||
|
{ me_cb_1_31, &gui_data.day, NULL, 1,ME_TIME_Y },
|
||
|
{ me_cb_num_label, NULL, (void *)".", 37,ME_TIME_Y },
|
||
|
{ me_cb_1_12, &gui_data.month, NULL, 45,ME_TIME_Y },
|
||
|
{ me_cb_num_label, NULL, (void *)".", 81,ME_TIME_Y },
|
||
|
{ me_cb_0_9, &gui_data.year_t, NULL, 89,ME_TIME_Y },
|
||
|
{ me_cb_0_9, &gui_data.year_o, NULL, 107,ME_TIME_Y },
|
||
|
//{ me_cb_button_full_line, (void *)me_action_save_date, (void *)"Speichern", 40,42 },
|
||
|
|
||
|
{ me_cb_button_half_line, (void *)me_action_to_setup_menu, (void *)"Abbrechen", 0,42 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_save_date, (void *)"Speichern", 64,42 },
|
||
|
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
/*============================================*/
|
||
|
/* Alarm Edit Dialog */
|
||
|
|
||
|
|
||
|
|
||
|
int me_action_alarm_done(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
gui_alarm_list[gui_alarm_index] = gui_alarm_current;
|
||
|
gui_alarm_list[gui_alarm_index].skip_wd = 0; /* clear the skip alarm (if any) */
|
||
|
gui_alarm_list[gui_alarm_index].snooze_count = 0; /* clear snooze (if any) */
|
||
|
//gui_alarm_calc_str_time(gui_alarm_index);
|
||
|
menu_SetMEList(menu, melist_alarm_menu, gui_alarm_index); /* first set the normal menu */
|
||
|
gui_Recalculate(); /* because it might be overwritten with the alarm menu */
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
#define ME_ALARM_TIME_XO 28
|
||
|
#define ME_ALARM_TIME_Y 26
|
||
|
#define ME_ALARM_WD_Y 36
|
||
|
#define ME_ALARM_WD_XO 8
|
||
|
const me_t melist_setup_alarm[] =
|
||
|
{
|
||
|
|
||
|
{ me_cb_0_23, &(gui_alarm_current.h), NULL, ME_ALARM_TIME_XO-7,ME_ALARM_TIME_Y },
|
||
|
{ me_cb_num_label, NULL, (void *)":", ME_ALARM_TIME_XO+30,ME_ALARM_TIME_Y-3 },
|
||
|
{ me_cb_0_55, &(gui_alarm_current.m), NULL, ME_ALARM_TIME_XO+39,ME_ALARM_TIME_Y },
|
||
|
|
||
|
{ me_cb_wd_toggle, &(gui_alarm_current.wd[0]), (void *)weekdaystr[0], ME_ALARM_WD_XO+17*0, ME_ALARM_WD_Y},
|
||
|
{ me_cb_wd_toggle, &(gui_alarm_current.wd[1]), (void *)weekdaystr[1], ME_ALARM_WD_XO+17*1, ME_ALARM_WD_Y},
|
||
|
{ me_cb_wd_toggle, &(gui_alarm_current.wd[2]), (void *)weekdaystr[2], ME_ALARM_WD_XO+17*2, ME_ALARM_WD_Y},
|
||
|
{ me_cb_wd_toggle, &(gui_alarm_current.wd[3]), (void *)weekdaystr[3], ME_ALARM_WD_XO+17*3, ME_ALARM_WD_Y},
|
||
|
{ me_cb_wd_toggle, &(gui_alarm_current.wd[4]), (void *)weekdaystr[4], ME_ALARM_WD_XO+17*4, ME_ALARM_WD_Y},
|
||
|
{ me_cb_wd_toggle, &(gui_alarm_current.wd[5]), (void *)weekdaystr[5], ME_ALARM_WD_XO+17*5, ME_ALARM_WD_Y},
|
||
|
{ me_cb_wd_toggle, &(gui_alarm_current.wd[6]), (void *)weekdaystr[6], ME_ALARM_WD_XO+17*6, ME_ALARM_WD_Y},
|
||
|
|
||
|
{ me_cb_big_toggle, (void *)&(gui_alarm_current.enable), NULL, 28, 47},
|
||
|
|
||
|
{ me_cb_16x16_bitmap_button, (void *)me_action_alarm_done, (void *)ok_xbm, 80, 44 },
|
||
|
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
/* Alarm Setup Menu */
|
||
|
|
||
|
|
||
|
static int me_action_alarm_common(menu_t *menu, const me_t *me, uint8_t msg) U8G2_NOINLINE;
|
||
|
static int me_action_alarm_common(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
gui_alarm_current = gui_alarm_list[gui_alarm_index];
|
||
|
menu_SetMEList(menu, melist_setup_alarm, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int me_action_alarm1(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
gui_alarm_index = 0;
|
||
|
return me_action_alarm_common(menu, me, msg);
|
||
|
}
|
||
|
|
||
|
int me_action_alarm2(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
gui_alarm_index = 1;
|
||
|
return me_action_alarm_common(menu, me, msg);
|
||
|
}
|
||
|
int me_action_alarm3(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
gui_alarm_index = 2;
|
||
|
return me_action_alarm_common(menu, me, msg);
|
||
|
}
|
||
|
int me_action_alarm4(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
gui_alarm_index = 3;
|
||
|
return me_action_alarm_common(menu, me, msg);
|
||
|
}
|
||
|
|
||
|
int me_action_handle_alarm_list(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_DRAW )
|
||
|
{
|
||
|
uint8_t y, x;
|
||
|
uint8_t ym;
|
||
|
u8g2_SetFont(menu->u8g2, MENU_SMALL_FONT);
|
||
|
for( y = 0; y < 4; y++ )
|
||
|
{
|
||
|
ym = y*8+7;
|
||
|
for( x = 0; x < 7; x++ )
|
||
|
{
|
||
|
u8g2_DrawGlyph(menu->u8g2, 0, ym, y+'1');
|
||
|
u8g2_DrawGlyph(menu->u8g2, 5, ym, ':');
|
||
|
u8g2_DrawStr(menu->u8g2, 9, ym, gui_alarm_str[y]);
|
||
|
if ( gui_alarm_list[y].wd[x] )
|
||
|
{
|
||
|
u8g2_DrawStr(menu->u8g2, 40+x*12, ym, weekdaystr[x]);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
else if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_alarm_menu, 4);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
const me_t melist_alarm_list_menu[] =
|
||
|
{
|
||
|
{ me_cb_button_empty, (void *)me_action_handle_alarm_list, NULL, 0, 0 },
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
int me_action_goto_alarm_list(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_alarm_list_menu, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
const me_t melist_alarm_menu[] =
|
||
|
{
|
||
|
{ me_cb_button_half_line, (void *)me_action_alarm1, gui_alarm_str[0], 0,12 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_alarm2, gui_alarm_str[1], 64,12 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_alarm3, gui_alarm_str[2], 0,24 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_alarm4, gui_alarm_str[3], 64,24 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_goto_alarm_list, (void *)"Liste", 0,36 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_to_top_menu, (void *)"Zurück", 64,36 },
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
/* Reset Menu */
|
||
|
|
||
|
int me_action_reset_no(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_setup_menu, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int me_action_reset_yes(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
do_reset();
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
const me_t melist_reset_menu[] =
|
||
|
{
|
||
|
{ me_cb_label, NULL, (void *)"Reset?", 44, 13},
|
||
|
{ me_cb_button_half_line, (void *)me_action_reset_no, (void *)"Nein", 0,30 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_reset_yes, (void *)"Ja", 64,30 },
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
/*============================================*/
|
||
|
/* Boot Info */
|
||
|
|
||
|
const char *bitstr(uint32_t v, int bitcnt)
|
||
|
{
|
||
|
static char s[34];
|
||
|
char *t = s;
|
||
|
uint32_t mask;
|
||
|
|
||
|
mask = 1<<(bitcnt-1);
|
||
|
while( mask > 0 )
|
||
|
{
|
||
|
*t = ((v & mask) == 0 ? '0' : '1');
|
||
|
t++;
|
||
|
mask >>= 1;
|
||
|
}
|
||
|
*t = '\0';
|
||
|
return s;
|
||
|
}
|
||
|
|
||
|
static const char *reset_reason_str[] = { "POR", "NVIC", "T2", "T3", "WUF" };
|
||
|
|
||
|
int me_action_handle_boot_info(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_DRAW )
|
||
|
{
|
||
|
u8g2_SetFont(menu->u8g2, MENU_SMALL_FONT);
|
||
|
u8g2_DrawStr(menu->u8g2, 0, 8-1, "RCC_CSR 31..24");
|
||
|
u8g2_DrawStr(menu->u8g2, 64, 8-1, bitstr(get_boot_status_register(), 8));
|
||
|
|
||
|
u8g2_DrawStr(menu->u8g2, 0, 16-1, "PWR_CSR 7..0");
|
||
|
u8g2_DrawStr(menu->u8g2, 64, 16-1, bitstr(get_pwr_status_register(), 8));
|
||
|
|
||
|
u8g2_DrawStr(menu->u8g2, 0, 24-1, "ResetReason");
|
||
|
u8g2_DrawStr(menu->u8g2, 64, 24-1, reset_reason_str[get_reset_reason()]);
|
||
|
|
||
|
u8g2_DrawStr(menu->u8g2, 0, 32-1, "Uptime");
|
||
|
u8g2_DrawStr(menu->u8g2, 64, 32-1, u8x8_u16toa(gui_data.uptime, 3));
|
||
|
|
||
|
u8g2_DrawStr(menu->u8g2, 0, 40-1, "Wakeups");
|
||
|
u8g2_DrawStr(menu->u8g2, 64, 40-1, u8x8_u16toa(get_wakeup_count(), 5));
|
||
|
|
||
|
u8g2_DrawStr(menu->u8g2, 0, 48-1, "DST by Date:");
|
||
|
u8g2_DrawStr(menu->u8g2, 50, 48-1, u8x8_u16toa(get_dst_by_date(), 2));
|
||
|
u8g2_DrawStr(menu->u8g2, 64, 48-1, "by RTC:");
|
||
|
u8g2_DrawStr(menu->u8g2, 95, 48-1, u8x8_u16toa(get_dst_by_RTC(), 1));
|
||
|
|
||
|
|
||
|
|
||
|
}
|
||
|
else if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_display_time, 4);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
const me_t melist_boot_info_menu[] =
|
||
|
{
|
||
|
{ me_cb_button_empty, (void *)me_action_handle_boot_info, NULL, 0, 0 },
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
/*============================================*/
|
||
|
/* System Menu */
|
||
|
|
||
|
|
||
|
const me_t melist_system_menu[] =
|
||
|
{
|
||
|
{ me_cb_label, NULL, (void *)"Helligkeit:", 0, 10},
|
||
|
{ me_cb_scale_1_7, &(gui_data.contrast), NULL, 103-10, 8},
|
||
|
{ me_cb_label, NULL, (void *)"Batteriespannung:", 0, 10+12},
|
||
|
{ me_cb_big_toggle, &(gui_data.display_voltage), NULL, 100-7, 10+12-8},
|
||
|
|
||
|
{ me_cb_button_full_line, (void *)me_action_to_setup_menu, (void *)"Speichern", 40,10+2*12 },
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
/*============================================*/
|
||
|
/* System 2 Menu */
|
||
|
|
||
|
int me_action_goto_boot_info(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_boot_info_menu, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int me_action_goto_reset(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_reset_menu, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
const me_t melist_system_2_menu[] =
|
||
|
{
|
||
|
//{ me_cb_button_half_line, (void *)me_action_setup_time, "Uhrzeit", 0,10 },
|
||
|
//{ me_cb_button_half_line, (void *)me_action_setup_date, "Datum", 64,10 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_goto_boot_info, (void *)"Info", 0,20 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_goto_reset, (void *)"Reset", 64,20 },
|
||
|
{ me_cb_button_full_line, (void *)me_action_to_setup_menu, (void *)"Zurück", 40,30 },
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
/* Setup Menu */
|
||
|
|
||
|
int me_action_setup_time(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_setup_time, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int me_action_setup_date(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_setup_date, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
int me_action_goto_system(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_system_menu, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int me_action_goto_system_2(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
//menu_SetMEList(menu, melist_reset_menu, 0);
|
||
|
menu_SetMEList(menu, melist_system_2_menu, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
const me_t melist_setup_menu[] =
|
||
|
{
|
||
|
{ me_cb_button_half_line, (void *)me_action_setup_time, (void *)"Uhrzeit", 0,10 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_setup_date, (void *)"Datum", 64,10 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_goto_system, (void *)"Anzeige", 0,20 },
|
||
|
{ me_cb_button_half_line, (void *)me_action_goto_system_2, (void *)"System", 64,20 },
|
||
|
{ me_cb_button_full_line, (void *)me_action_to_top_menu, (void *)"Zurück", 40,31 },
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
/*============================================*/
|
||
|
/* Alarm Menu */
|
||
|
|
||
|
|
||
|
int me_action_deactivate_alarm(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_display_time, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int me_action_do_snooze(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_display_time, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int me_cb_cond_inv_label(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( gui_data.is_alarm == 0 )
|
||
|
return me_cb_inv_label(menu, me, msg);
|
||
|
if ( gui_alarm_list[gui_data.active_alarm_idx].snooze_count == 0 )
|
||
|
return me_cb_inv_label(menu, me, msg);
|
||
|
return 0;
|
||
|
|
||
|
}
|
||
|
|
||
|
const me_t melist_active_alarm_menu[] =
|
||
|
{
|
||
|
{ me_cb_label, NULL, (void *)"Alarm", 2, 13},
|
||
|
|
||
|
{ me_cb_0_23_ro, &gui_data.h, NULL, ME_TIME_DXO+2+12,ME_TIME_Y },
|
||
|
{ me_cb_num_label, NULL, (void *)":", ME_TIME_DXO+30+20,ME_TIME_Y-3 },
|
||
|
{ me_cb_0_9_ro, &gui_data.mt, NULL, ME_TIME_DXO+39+19,ME_TIME_Y },
|
||
|
{ me_cb_0_9_ro, &gui_data.mo, NULL, ME_TIME_DXO+52+24,ME_TIME_Y },
|
||
|
|
||
|
{ me_cb_button_empty, (void *)me_action_deactivate_alarm, NULL, 0,0 },
|
||
|
//{ me_cb_button_half_line, (void *)me_action_deactivate_alarm, (void *)"Alarm aus", 0,30 },
|
||
|
//{ me_cb_button_half_line, (void *)me_action_do_snooze, (void *)"+5 Min ", 64,30 },
|
||
|
|
||
|
{ me_cb_inv_label, NULL, (void *)"Alarm aus", 4,40 },
|
||
|
{ me_cb_cond_inv_label, NULL, (void *)"+5 Min", 76,40 },
|
||
|
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
/* toplevel menu */
|
||
|
|
||
|
|
||
|
int me_action_to_display_time(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_display_time, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
|
||
|
int me_action_to_alarm_menu(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
if ( msg == ME_MSG_SELECT )
|
||
|
{
|
||
|
menu_SetMEList(menu, melist_alarm_menu, 0);
|
||
|
return 1;
|
||
|
}
|
||
|
return 0;
|
||
|
}
|
||
|
|
||
|
int me_cb_button_skip_alarm(menu_t *menu, const me_t *me, uint8_t msg)
|
||
|
{
|
||
|
int r = 0;
|
||
|
u8g2_uint_t x;
|
||
|
switch(msg)
|
||
|
{
|
||
|
case ME_MSG_IS_FOCUS:
|
||
|
return gui_data.is_skip_possible;
|
||
|
case ME_MSG_DRAW_FOCUS:
|
||
|
u8g2_SetFont(menu->u8g2, MENU_NORMAL_FONT);
|
||
|
menu_DrawBoxFocus(menu,
|
||
|
0,
|
||
|
me->y - u8g2_GetAscent(menu->u8g2)-1,
|
||
|
u8g2_GetDisplayWidth(menu->u8g2) ,
|
||
|
u8g2_GetAscent(menu->u8g2) - u8g2_GetDescent(menu->u8g2) +1);
|
||
|
r = 1;
|
||
|
break;
|
||
|
case ME_MSG_DRAW:
|
||
|
if ( gui_data.is_skip_possible )
|
||
|
{
|
||
|
u8g2_SetFont(menu->u8g2, MENU_NORMAL_FONT);
|
||
|
gui_alarm_to_str(gui_data.next_alarm_index);
|
||
|
x = u8g2_DrawUTF8(menu->u8g2, me->x, me->y, gui_data.s );
|
||
|
u8g2_DrawUTF8(menu->u8g2, me->x+x, me->y, " deaktvieren" );
|
||
|
}
|
||
|
r = 1;
|
||
|
break;
|
||
|
case ME_MSG_SELECT:
|
||
|
//printf("me_cb_button_skip_alarm ME_MSG_SELECT\n");
|
||
|
gui_alarm_list[gui_data.next_alarm_index].skip_wd =
|
||
|
gui_alarm_list[gui_data.next_alarm_index].na_wd + 1;
|
||
|
gui_alarm_list[gui_data.next_alarm_index].snooze_count = 0; /* clear snooze (if any) */
|
||
|
menu_SetMEList(menu, melist_display_time, 0); /* first set the normal menu */
|
||
|
gui_Recalculate(); /* it might be changed here to the alarm menu */
|
||
|
r = 1;
|
||
|
break;
|
||
|
}
|
||
|
return r;
|
||
|
}
|
||
|
|
||
|
|
||
|
const me_t melist_top_menu[] =
|
||
|
{
|
||
|
{ me_cb_button_full_line, (void *)me_action_to_display_time, (void *)"Zurück", 3,12 },
|
||
|
{ me_cb_button_full_line, (void *)me_action_to_alarm_menu, (void *)"Alarm", 3,24 },
|
||
|
{ me_cb_button_skip_alarm, NULL, NULL, 3,36 },
|
||
|
{ me_cb_button_full_line, (void *)me_action_to_setup_menu, (void *)"Weitere Funktionen", 3,48 },
|
||
|
{ me_cb_null, NULL, NULL, 0, 0 },
|
||
|
};
|
||
|
|
||
|
|
||
|
/*===================================================*/
|
||
|
|
||
|
|
||
|
/*============================================*/
|
||
|
|
||
|
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;
|
||
|
}
|
||
|
}
|
||
|
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);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
/*===================================================*/
|
||
|
|
||
|
void read_rtc(void)
|
||
|
{
|
||
|
gui_data.h = rtc.getHours();
|
||
|
gui_data.mt = rtc.getMinutes()/10;
|
||
|
gui_data.mo = rtc.getMinutes()%10;
|
||
|
gui_data.st = rtc.getSeconds()/10;
|
||
|
gui_data.so = rtc.getSeconds()%10;
|
||
|
|
||
|
gui_data.day = rtc.getDay();
|
||
|
gui_data.month = rtc.getMonth();
|
||
|
gui_data.year_o = rtc.getYear()/10;
|
||
|
gui_data.year_t = rtc.getYear()%10;
|
||
|
|
||
|
//gui_date_adjust(); /* calculate weekday */
|
||
|
gui_Recalculate();
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
void setup(void) {
|
||
|
|
||
|
rtc.begin();
|
||
|
|
||
|
// U8g2 SH1106 Proto-Shield
|
||
|
//u8g2.begin(/* menu_select_pin= */ 2, /* menu_next_pin= */ 4, /* menu_prev_pin= */ 7, /* menu_up_pin= */ 6, /* menu_down_pin= */ 5, /* menu_home_pin= */ 3);
|
||
|
|
||
|
// DOGS102 Shield (http://shieldlist.org/controlconnection/dogs102)
|
||
|
// u8g2.begin(/* menu_select_pin= */ 5, /* menu_next_pin= */ 4, /* menu_prev_pin= */ 2, /* menu_up_pin= */ U8X8_PIN_NONE, /* menu_down_pin= */ U8X8_PIN_NONE, /* menu_home_pin= */ 3);
|
||
|
|
||
|
// DOGM128 Shield (http://shieldlist.org/schmelle2/dogm128) + DOGXL160 Shield
|
||
|
//u8g2.begin(/* menu_select_pin= */ 2, /* menu_next_pin= */ 3, /* menu_prev_pin= */ 7, /* menu_up_pin= */ U8X8_PIN_NONE, /* menu_down_pin= */ U8X8_PIN_NONE, /* menu_home_pin= */ 8);
|
||
|
|
||
|
// Arduboy
|
||
|
//u8g2.begin(/*Select=*/ A0, /*Right/Next=*/ 5, /*Left/Prev=*/ 9, /*Up=*/ 8, /*Down=*/ 10, /*Home/Cancel=*/ A1); // Arduboy DevKit
|
||
|
//u8g2.begin(/*Select=*/ 7, /*Right/Next=*/ A1, /*Left/Prev=*/ A2, /*Up=*/ A0, /*Down=*/ A3, /*Home/Cancel=*/ 8); // Arduboy 10 (Production)
|
||
|
|
||
|
u8g2.begin(/* menu_select_pin= */ 6, /* menu_next_pin= */ 7, /* menu_prev_pin= */ U8X8_PIN_NONE, /* menu_up_pin= */ U8X8_PIN_NONE, /* menu_down_pin= */ U8X8_PIN_NONE, /* menu_home_pin= */ U8X8_PIN_NONE);
|
||
|
u8g2.setFont(u8g2_font_6x12_tr);
|
||
|
u8g2.setFlipMode(1);
|
||
|
|
||
|
u8g2_ptr = u8g2.getU8g2();
|
||
|
|
||
|
gui_Init(u8g2_ptr, 0);
|
||
|
|
||
|
}
|
||
|
|
||
|
|
||
|
|
||
|
void loop(void) {
|
||
|
uint8_t event;
|
||
|
unsigned long m;
|
||
|
|
||
|
for(;;)
|
||
|
{
|
||
|
read_rtc();
|
||
|
u8g2.firstPage();
|
||
|
do
|
||
|
{
|
||
|
gui_Draw();
|
||
|
} while(u8g2.nextPage());
|
||
|
|
||
|
m = millis() + 400UL;
|
||
|
for(;;)
|
||
|
{
|
||
|
event = u8g2.getMenuEvent();
|
||
|
if ( event == U8X8_MSG_GPIO_MENU_SELECT )
|
||
|
{
|
||
|
gui_Select();
|
||
|
break;
|
||
|
}
|
||
|
else if ( event == U8X8_MSG_GPIO_MENU_NEXT )
|
||
|
{
|
||
|
gui_Next();
|
||
|
break;
|
||
|
}
|
||
|
if ( millis() > m )
|
||
|
break;
|
||
|
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
|