CamTimer/Src/main.c

900 lines
27 KiB
C
Raw Permalink Normal View History

2021-03-08 18:53:17 +01:00
/* USER CODE BEGIN Header */
2020-01-15 18:02:31 +01:00
/**
******************************************************************************
* @file : main.c
* @brief : Main program body
******************************************************************************
** This notice applies to any and all portions of this file
* that are not between comment pairs USER CODE BEGIN and
* USER CODE END. Other portions of this file, whether
* inserted by the user or by software development tools
* are owned by their respective copyright owners.
*
* COPYRIGHT(c) 2020 STMicroelectronics
*
* Redistribution and use in source and binary forms, with or without modification,
* are permitted provided that the following conditions are met:
* 1. Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
* 2. 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.
* 3. Neither the name of STMicroelectronics nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* 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.
*
******************************************************************************
*/
/* USER CODE END Header */
/* Includes ------------------------------------------------------------------*/
#include "main.h"
/* Private includes ----------------------------------------------------------*/
/* USER CODE BEGIN Includes */
2021-03-08 18:53:17 +01:00
#include "EPD_1in54_V2.h"
2020-01-15 18:02:31 +01:00
#include "DEV_Config.h"
#include "GUI_Paint.h"
#include "ImageData.h"
2020-02-06 13:29:37 +01:00
#include "eeprom.h"
2020-01-15 18:02:31 +01:00
#include <stdlib.h>
2020-01-16 15:32:21 +01:00
#include <stdbool.h>
2020-01-15 18:02:31 +01:00
/* USER CODE END Includes */
/* Private typedef -----------------------------------------------------------*/
/* USER CODE BEGIN PTD */
/* USER CODE END PTD */
/* Private define ------------------------------------------------------------*/
/* USER CODE BEGIN PD */
/* USER CODE END PD */
/* Private macro -------------------------------------------------------------*/
/* USER CODE BEGIN PM */
/* USER CODE END PM */
/* Private variables ---------------------------------------------------------*/
RTC_HandleTypeDef hrtc;
SPI_HandleTypeDef hspi1;
UART_HandleTypeDef huart1;
/* USER CODE BEGIN PV */
2020-01-15 18:02:31 +01:00
PAINT_TIME sPaint_time;
2020-01-15 18:02:31 +01:00
RTC_TimeTypeDef sTime;
RTC_TimeTypeDef sTimeSet = {0,0,0};
RTC_TimeTypeDef sSetStart[7] = {{0,0,0},{0,0,0},{0,0,0},{0,0,0},{0,0,0},{0,0,0},{0,0,0}};
RTC_TimeTypeDef sSetEnd[7] = {{0,0,0},{0,0,0},{0,0,0},{0,0,0},{0,0,0},{0,0,0},{0,0,0}};
RTC_DateTypeDef sDate = {0,1,1,20};
const uint8_t WeekLetter[7][2]={
"Mo",
"Di",
"Mi",
"Do",
"Fr",
"Sa",
"So"
};
2021-03-08 18:53:17 +01:00
enum Mode{Running, Choose, Set /*, Offscreen*/};
uint8_t system_mode = Running;
uint8_t setWeekday,weekday,hour,minsec;
2020-02-06 13:29:37 +01:00
uint8_t selector_pos = 0;
2020-01-15 18:02:31 +01:00
//Create a new image cache
UBYTE *BlackImage;
2021-03-08 18:53:17 +01:00
UWORD Imagesize = ((EPD_1IN54_V2_WIDTH % 8 == 0) ? (EPD_1IN54_V2_WIDTH / 8) : (EPD_1IN54_V2_WIDTH / 8 + 1)) * EPD_1IN54_V2_HEIGHT;
2020-01-15 18:02:31 +01:00
/* USER CODE END PV */
/* Private function prototypes -----------------------------------------------*/
void SystemClock_Config(void);
static void MX_GPIO_Init(void);
static void MX_SPI1_Init(void);
static void MX_USART1_UART_Init(void);
static void MX_RTC_Init(void);
/* USER CODE BEGIN PFP */
2020-01-21 15:23:22 +01:00
void Paint_DrawArray(void);
void Paint_DrawCurrent(void);
void Paint_Drawonoff(bool);
void Paint_Selector(bool, uint8_t,uint8_t);
2020-02-06 13:29:37 +01:00
void System_bjt_output(bool);
void System_screenfullupdate(bool);
bool System_fire(void);
bool System_CustomHeartbeat(uint8_t);
2020-01-21 15:21:05 +01:00
void ButtonLeft_Pressed(void);
void ButtonPress_Pressed(void);
void ButtonRight_Pressed(void);
void Change_systemsetmode(void);
void Change_systemmode(void);
void Change_selectorpos(bool);
void Change_time(uint8_t,bool);
void Read_intosettime(void);
void Write_intostime(void);
2020-02-06 13:29:37 +01:00
void Eeprom_loadsettings(void);
void Eeprom_savesettings(void);
2020-01-15 18:02:31 +01:00
/* USER CODE END PFP */
/* Private user code ---------------------------------------------------------*/
/* USER CODE BEGIN 0 */
/* USER CODE END 0 */
/**
* @brief The application entry point.
* @retval int
*/
int main(void)
{
/* USER CODE BEGIN 1 */
/* USER CODE END 1 */
2021-03-08 18:53:17 +01:00
2020-01-15 18:02:31 +01:00
/* MCU Configuration--------------------------------------------------------*/
/* Reset of all peripherals, Initializes the Flash interface and the Systick. */
HAL_Init();
/* USER CODE BEGIN Init */
/* USER CODE END Init */
/* Configure the system clock */
SystemClock_Config();
/* USER CODE BEGIN SysInit */
/* USER CODE END SysInit */
/* Initialize all configured peripherals */
MX_GPIO_Init();
MX_SPI1_Init();
MX_USART1_UART_Init();
MX_RTC_Init();
/* USER CODE BEGIN 2 */
2021-03-08 18:53:17 +01:00
DEV_Module_Init();
printf("e-Paper Init and Clear...\r\n");
EPD_1IN54_V2_Init();
EPD_1IN54_V2_Clear();
DEV_Delay_ms(500);
if ((BlackImage = (UBYTE *)malloc(Imagesize)) == NULL){
printf("Failed to apply for black memory...\r\n");
return 1;
}
2021-03-08 18:53:17 +01:00
Paint_NewImage(BlackImage, EPD_1IN54_V2_WIDTH, EPD_1IN54_V2_HEIGHT, 270, WHITE);
Paint_SelectImage(BlackImage);
Paint_Clear(WHITE);
sPaint_time.Hour = 0;
sPaint_time.Min = 0;
sPaint_time.Sec = 0;
2020-01-15 18:02:31 +01:00
//Partial refresh, example shows time
Paint_SelectImage(BlackImage);
2020-02-06 13:29:37 +01:00
Eeprom_loadsettings();
2020-01-15 18:02:31 +01:00
for (;;){
HAL_RTC_GetTime(&hrtc, &sTime, RTC_FORMAT_BIN); // Get Time
HAL_RTC_GetDate(&hrtc, &sDate, RTC_FORMAT_BIN); // Get Date
sPaint_time.Hour = sTime.Hours;
sPaint_time.Min = sTime.Minutes;
sPaint_time.Sec = sTime.Seconds;
2021-03-08 18:53:17 +01:00
// System_screenfullupdate(System_CustomHeartbeat(1));
Paint_SelectImage(BlackImage);
Paint_Clear(WHITE);
2020-02-06 13:29:37 +01:00
2021-03-08 18:53:17 +01:00
Paint_DrawArray();
Paint_DrawCurrent();
Paint_Selector(System_CustomHeartbeat(2),selector_pos, system_mode);
Paint_Drawonoff(System_fire());
EPD_1IN54_V2_DisplayPart(BlackImage);
// DEV_Delay_ms(50); //Analog clock 1s
2020-02-06 13:29:37 +01:00
System_bjt_output(System_fire());
2021-03-08 18:53:17 +01:00
//printout current system
// if(System_fire()){
// printf("HOT \r\n");
// }else{
// printf("COLD \r\n");
// }
// if(HAL_GPIO_ReadPin(FLT_GPIO_Port,FLT_Pin)){
// printf("FLT HIGH\r\n");
// }else{
// printf("FLT LOW\r\n");
// }
2020-02-06 13:29:37 +01:00
// else{
// Paint_Clear(WHITE);
// Paint_DrawString_EN(0,0,"You can remove the monitor",&Font16,WHITE,BLACK);
// EPD_Display(BlackImage);
// DEV_Delay_ms(50); //Analog clock 1s
// EPD_Sleep();
// }
2020-01-15 18:02:31 +01:00
// printf("500ms..\r\n");
2021-03-08 18:53:17 +01:00
if(!HAL_GPIO_ReadPin(ButtonLeft_GPIO_Port,ButtonLeft_Pin)){
2020-01-21 15:21:40 +01:00
ButtonLeft_Pressed();
}
2021-03-08 18:53:17 +01:00
if(!HAL_GPIO_ReadPin(ButtonPress_GPIO_Port,ButtonPress_Pin)){
2020-01-21 15:21:40 +01:00
ButtonPress_Pressed();
}
2021-03-08 18:53:17 +01:00
if(!HAL_GPIO_ReadPin(ButtonRight_GPIO_Port,ButtonRight_Pin)){
2020-01-21 15:21:40 +01:00
ButtonRight_Pressed();
}
2021-03-08 18:53:17 +01:00
if(!HAL_GPIO_ReadPin(EINK_Refresh_GPIO_Port,EINK_Refresh_Pin)){
HAL_Delay(150);
if(!HAL_GPIO_ReadPin(EINK_Refresh_GPIO_Port,EINK_Refresh_Pin)){
EPD_1IN54_V2_Init();
EPD_1IN54_V2_Clear();
DEV_Delay_ms(20);
}
}
2021-03-08 18:53:17 +01:00
2020-01-15 18:02:31 +01:00
}
/* USER CODE END 2 */
2021-03-08 18:53:17 +01:00
2020-01-15 18:02:31 +01:00
/* Infinite loop */
/* USER CODE BEGIN WHILE */
2020-01-15 18:02:31 +01:00
/* USER CODE END WHILE */
/* USER CODE BEGIN 3 */
/* USER CODE END 3 */
}
2020-01-15 18:02:31 +01:00
/**
* @brief System Clock Configuration
* @retval None
*/
void SystemClock_Config(void)
{
RCC_OscInitTypeDef RCC_OscInitStruct = {0};
RCC_ClkInitTypeDef RCC_ClkInitStruct = {0};
RCC_PeriphCLKInitTypeDef PeriphClkInit = {0};
2021-03-08 18:53:17 +01:00
/** Initializes the CPU, AHB and APB busses clocks
2020-01-15 18:02:31 +01:00
*/
2021-03-08 18:53:17 +01:00
RCC_OscInitStruct.OscillatorType = RCC_OSCILLATORTYPE_HSE;
2020-01-15 18:02:31 +01:00
RCC_OscInitStruct.HSEState = RCC_HSE_ON;
RCC_OscInitStruct.HSEPredivValue = RCC_HSE_PREDIV_DIV1;
RCC_OscInitStruct.HSIState = RCC_HSI_ON;
RCC_OscInitStruct.PLL.PLLState = RCC_PLL_ON;
RCC_OscInitStruct.PLL.PLLSource = RCC_PLLSOURCE_HSE;
RCC_OscInitStruct.PLL.PLLMUL = RCC_PLL_MUL9;
if (HAL_RCC_OscConfig(&RCC_OscInitStruct) != HAL_OK)
{
Error_Handler();
}
2021-03-08 18:53:17 +01:00
/** Initializes the CPU, AHB and APB busses clocks
2020-01-15 18:02:31 +01:00
*/
RCC_ClkInitStruct.ClockType = RCC_CLOCKTYPE_HCLK|RCC_CLOCKTYPE_SYSCLK
|RCC_CLOCKTYPE_PCLK1|RCC_CLOCKTYPE_PCLK2;
RCC_ClkInitStruct.SYSCLKSource = RCC_SYSCLKSOURCE_PLLCLK;
RCC_ClkInitStruct.AHBCLKDivider = RCC_SYSCLK_DIV1;
RCC_ClkInitStruct.APB1CLKDivider = RCC_HCLK_DIV2;
RCC_ClkInitStruct.APB2CLKDivider = RCC_HCLK_DIV1;
if (HAL_RCC_ClockConfig(&RCC_ClkInitStruct, FLASH_LATENCY_2) != HAL_OK)
{
Error_Handler();
}
PeriphClkInit.PeriphClockSelection = RCC_PERIPHCLK_RTC;
2021-03-08 18:53:17 +01:00
PeriphClkInit.RTCClockSelection = RCC_RTCCLKSOURCE_HSE_DIV128;
2020-01-15 18:02:31 +01:00
if (HAL_RCCEx_PeriphCLKConfig(&PeriphClkInit) != HAL_OK)
{
Error_Handler();
}
}
/**
* @brief RTC Initialization Function
* @param None
* @retval None
*/
static void MX_RTC_Init(void)
{
/* USER CODE BEGIN RTC_Init 0 */
/* USER CODE END RTC_Init 0 */
RTC_TimeTypeDef sTime = {0};
RTC_DateTypeDef DateToUpdate = {0};
2021-03-08 18:53:17 +01:00
RTC_TamperTypeDef sTamper = {0};
2020-01-15 18:02:31 +01:00
/* USER CODE BEGIN RTC_Init 1 */
/* USER CODE END RTC_Init 1 */
2021-03-08 18:53:17 +01:00
/** Initialize RTC Only
2020-01-15 18:02:31 +01:00
*/
hrtc.Instance = RTC;
hrtc.Init.AsynchPrediv = RTC_AUTO_1_SECOND;
hrtc.Init.OutPut = RTC_OUTPUTSOURCE_NONE;
if (HAL_RTC_Init(&hrtc) != HAL_OK)
{
Error_Handler();
}
/* USER CODE BEGIN Check_RTC_BKUP */
2021-03-08 18:53:17 +01:00
2020-01-15 18:02:31 +01:00
/* USER CODE END Check_RTC_BKUP */
2021-03-08 18:53:17 +01:00
/** Initialize RTC and set the Time and Date
2020-01-15 18:02:31 +01:00
*/
sTime.Hours = 0x0;
sTime.Minutes = 0x0;
sTime.Seconds = 0x0;
if (HAL_RTC_SetTime(&hrtc, &sTime, RTC_FORMAT_BCD) != HAL_OK)
{
Error_Handler();
}
DateToUpdate.WeekDay = RTC_WEEKDAY_MONDAY;
DateToUpdate.Month = RTC_MONTH_JANUARY;
DateToUpdate.Date = 0x1;
2020-01-15 18:02:31 +01:00
DateToUpdate.Year = 0x0;
if (HAL_RTC_SetDate(&hrtc, &DateToUpdate, RTC_FORMAT_BCD) != HAL_OK)
{
Error_Handler();
}
2021-03-08 18:53:17 +01:00
/** Enable the RTC Tamper
*/
sTamper.Tamper = RTC_TAMPER_1;
sTamper.Trigger = RTC_TAMPERTRIGGER_LOWLEVEL;
if (HAL_RTCEx_SetTamper(&hrtc, &sTamper) != HAL_OK)
{
Error_Handler();
}
2020-01-15 18:02:31 +01:00
/* USER CODE BEGIN RTC_Init 2 */
/* USER CODE END RTC_Init 2 */
}
/**
* @brief SPI1 Initialization Function
* @param None
* @retval None
*/
static void MX_SPI1_Init(void)
{
/* USER CODE BEGIN SPI1_Init 0 */
/* USER CODE END SPI1_Init 0 */
/* USER CODE BEGIN SPI1_Init 1 */
/* USER CODE END SPI1_Init 1 */
/* SPI1 parameter configuration*/
hspi1.Instance = SPI1;
hspi1.Init.Mode = SPI_MODE_MASTER;
hspi1.Init.Direction = SPI_DIRECTION_2LINES;
hspi1.Init.DataSize = SPI_DATASIZE_8BIT;
hspi1.Init.CLKPolarity = SPI_POLARITY_LOW;
hspi1.Init.CLKPhase = SPI_PHASE_1EDGE;
hspi1.Init.NSS = SPI_NSS_SOFT;
hspi1.Init.BaudRatePrescaler = SPI_BAUDRATEPRESCALER_64;
hspi1.Init.FirstBit = SPI_FIRSTBIT_MSB;
hspi1.Init.TIMode = SPI_TIMODE_DISABLE;
hspi1.Init.CRCCalculation = SPI_CRCCALCULATION_DISABLE;
hspi1.Init.CRCPolynomial = 10;
if (HAL_SPI_Init(&hspi1) != HAL_OK)
{
Error_Handler();
}
/* USER CODE BEGIN SPI1_Init 2 */
/* USER CODE END SPI1_Init 2 */
}
/**
* @brief USART1 Initialization Function
* @param None
* @retval None
*/
static void MX_USART1_UART_Init(void)
{
/* USER CODE BEGIN USART1_Init 0 */
/* USER CODE END USART1_Init 0 */
/* USER CODE BEGIN USART1_Init 1 */
/* USER CODE END USART1_Init 1 */
huart1.Instance = USART1;
huart1.Init.BaudRate = 115200;
huart1.Init.WordLength = UART_WORDLENGTH_8B;
huart1.Init.StopBits = UART_STOPBITS_1;
huart1.Init.Parity = UART_PARITY_NONE;
huart1.Init.Mode = UART_MODE_TX_RX;
huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE;
huart1.Init.OverSampling = UART_OVERSAMPLING_16;
if (HAL_UART_Init(&huart1) != HAL_OK)
{
Error_Handler();
}
/* USER CODE BEGIN USART1_Init 2 */
/* USER CODE END USART1_Init 2 */
}
/**
* @brief GPIO Initialization Function
* @param None
* @retval None
*/
static void MX_GPIO_Init(void)
{
GPIO_InitTypeDef GPIO_InitStruct = {0};
/* GPIO Ports Clock Enable */
__HAL_RCC_GPIOC_CLK_ENABLE();
__HAL_RCC_GPIOD_CLK_ENABLE();
__HAL_RCC_GPIOA_CLK_ENABLE();
__HAL_RCC_GPIOB_CLK_ENABLE();
/*Configure GPIO pin Output Level */
HAL_GPIO_WritePin(GPIOA, RST_Pin|DC_Pin|BUSY_Pin|SPI_CS_Pin, GPIO_PIN_RESET);
/*Configure GPIO pin Output Level */
2020-02-06 13:29:37 +01:00
HAL_GPIO_WritePin(GPIOB, TimerControl_Pin|LED_Pin, GPIO_PIN_RESET);
2020-01-15 18:02:31 +01:00
/*Configure GPIO pins : RST_Pin DC_Pin BUSY_Pin SPI_CS_Pin */
GPIO_InitStruct.Pin = RST_Pin|DC_Pin|BUSY_Pin|SPI_CS_Pin;
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
GPIO_InitStruct.Pull = GPIO_NOPULL;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
HAL_GPIO_Init(GPIOA, &GPIO_InitStruct);
2020-02-06 13:29:37 +01:00
/*Configure GPIO pins : TimerControl_Pin LED_Pin */
GPIO_InitStruct.Pin = TimerControl_Pin|LED_Pin;
GPIO_InitStruct.Mode = GPIO_MODE_OUTPUT_PP;
GPIO_InitStruct.Pull = GPIO_NOPULL;
GPIO_InitStruct.Speed = GPIO_SPEED_FREQ_LOW;
HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
2021-03-08 18:53:17 +01:00
/*Configure GPIO pins : ButtonLeft_Pin ButtonPress_Pin ButtonRight_Pin EINK_Refresh_Pin */
GPIO_InitStruct.Pin = ButtonLeft_Pin|ButtonPress_Pin|ButtonRight_Pin|EINK_Refresh_Pin;
GPIO_InitStruct.Mode = GPIO_MODE_INPUT;
GPIO_InitStruct.Pull = GPIO_PULLUP;
HAL_GPIO_Init(GPIOB, &GPIO_InitStruct);
2021-03-08 18:53:17 +01:00
/*Configure GPIO pin : FLT_Pin */
GPIO_InitStruct.Pin = FLT_Pin;
GPIO_InitStruct.Mode = GPIO_MODE_INPUT;
GPIO_InitStruct.Pull = GPIO_NOPULL;
HAL_GPIO_Init(FLT_GPIO_Port, &GPIO_InitStruct);
2020-01-15 18:02:31 +01:00
}
/* USER CODE BEGIN 4 */
2020-01-21 15:23:22 +01:00
void Paint_DrawArray(){
for(int i=0; i<7; i++){
sPaint_time.Hour = sSetStart[i].Hours;
sPaint_time.Min = sSetStart[i].Minutes;
sPaint_time.Sec = sSetStart[i].Seconds;
Paint_ClearWindows(5, i*25 + 5, 5 + Font20.Width * 7, i*25 + 5 + Font20.Height, WHITE);
Paint_DrawTime(5, i*25 + 5, &sPaint_time, &Font20, WHITE, BLACK);
sPaint_time.Hour = sSetEnd[i].Hours;
sPaint_time.Min = sSetEnd[i].Minutes;
sPaint_time.Sec = sSetEnd[i].Seconds;
Paint_ClearWindows(100, i*25 + 5, 100 + Font20.Width * 7, i*25 + 5 + Font20.Height, WHITE);
Paint_DrawTime(100, i*25 + 5, &sPaint_time, &Font20, WHITE, BLACK);
}
for(uint8_t i=0; i<7 ; i++){
Paint_DrawChar(176,25*i+5,WeekLetter[i][0],&Font20,WHITE,BLACK);
Paint_DrawChar(187,25*i+5,WeekLetter[i][1],&Font20,WHITE,BLACK);
}
}
void Paint_DrawCurrent(){
weekday = sDate.WeekDay;
sPaint_time.Hour = sTime.Hours;
sPaint_time.Min = sTime.Minutes;
sPaint_time.Sec = sTime.Seconds;
if(system_mode == Set && (selector_pos == 28 || selector_pos == 29 || selector_pos == 30)){
sPaint_time.Hour = sTimeSet.Hours;
sPaint_time.Min = sTimeSet.Minutes;
sPaint_time.Sec = sTimeSet.Seconds;
weekday = setWeekday;
}
Paint_ClearWindows(50, 175, 50 + Font24.Width * 7, 175 + Font24.Height, WHITE);
uint8_t value[10] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9'};
UWORD Dx = Font24.Width;
//Write data into the cache
Paint_DrawChar(50 , 175, value[sPaint_time.Hour / 10], &Font24, WHITE, BLACK);
Paint_DrawChar(50 + Dx , 175, value[sPaint_time.Hour % 10], &Font24, WHITE, BLACK);
if(System_CustomHeartbeat(2)){
Paint_DrawChar(50 + Dx + Dx / 4 + Dx / 2 , 175, ':' , &Font24, WHITE, BLACK);
}else{
Paint_ClearWindows(80 , 175, 80 + Font24.Width * 7, 175 + Font24.Height, WHITE);
}
Paint_DrawChar(50 + Dx * 2 + Dx / 2 , 175, value[sPaint_time.Min / 10] , &Font24, WHITE, BLACK);
Paint_DrawChar(50 + Dx * 3 + Dx / 2 , 175, value[sPaint_time.Min % 10] , &Font24, WHITE, BLACK);
// Paint_DrawTime(50,175, &sPaint_time, &Font24, WHITE, BLACK);
Paint_DrawChar(140,175,WeekLetter[weekday][0],&Font20,WHITE,BLACK);
Paint_DrawChar(152,175,WeekLetter[weekday][1],&Font20,WHITE,BLACK);
2020-02-06 13:29:37 +01:00
//debug
Paint_DrawChar(0 , 175, value[sPaint_time.Sec / 10] , &Font24, WHITE, BLACK);
Paint_DrawChar(15 , 175, value[sPaint_time.Sec % 10] , &Font24, WHITE, BLACK);
}
void Paint_Drawonoff(bool draw_yes){
Paint_ClearWindows(180, 180, 190, 190, WHITE);
Paint_DrawRectangle(180, 180, 190, 190, BLACK, draw_yes, DOT_PIXEL_1X1);
}
void Paint_Selector(bool heartbeat,uint8_t pos, uint8_t runningmode){
if(runningmode == Choose || runningmode == Set){
if(runningmode == Choose){
heartbeat = true;
}
if(pos< MAX_SSET_POS){
const uint8_t sset_selectorpos[4]={10,45,105,140};
const uint8_t sset_selectorlength=20;
uint8_t sset_selectorrow= pos/4 + 1;
Paint_DrawLine(sset_selectorpos[pos%4],25*sset_selectorrow,sset_selectorpos[pos%4]+sset_selectorlength,25*sset_selectorrow,BLACK,heartbeat,DOT_PIXEL_2X2);
}
if(pos == 28){
Paint_DrawLine(55,195,80,195,BLACK,heartbeat,DOT_PIXEL_2X2);
}
if(pos == 29){
Paint_DrawLine(95,195,120,195,BLACK,heartbeat,DOT_PIXEL_2X2);
}
if(pos == 30){
Paint_DrawLine(140,195,165,195,BLACK,heartbeat,DOT_PIXEL_2X2);
}
for(uint8_t i=1 ; i<8 ; i++){
Paint_ClearWindows(0, 25*i, 200, 25*i + 2, WHITE);
}
}
}
2020-02-06 13:29:37 +01:00
void System_bjt_output(bool on){
if(on){
HAL_GPIO_WritePin(GPIOB,GPIO_PIN_10,GPIO_PIN_RESET);
}else{
HAL_GPIO_WritePin(GPIOB,GPIO_PIN_10,GPIO_PIN_SET);
2020-02-06 13:29:37 +01:00
}
}
void System_screenfullupdate(bool heartbeat){
2021-03-08 18:53:17 +01:00
// if(heartbeat){
// if (EPD_Init(lut_full_update) != 0){
// printf("e-Paper init failed\r\n");
// }
// }else{
// if (EPD_Init(lut_partial_update) != 0){
// printf("e-Paper init failed\r\n");
// }
// }
2020-02-06 13:29:37 +01:00
}
bool System_fire(){
uint16_t Activetime_length_min=0, Currenttime_length_min=0;
if(sSetEnd[weekday].Hours > sSetStart[weekday].Hours){
Activetime_length_min = (sSetEnd[weekday].Hours * 60 + sSetEnd[weekday].Minutes) - (sSetStart[weekday].Hours * 60 + sSetStart[weekday].Minutes);
}
if(sSetEnd[weekday].Hours == sSetStart[weekday].Hours){
if(sSetEnd[weekday].Minutes > sSetStart[weekday].Minutes){
Activetime_length_min = sSetEnd[weekday].Minutes - sSetStart[weekday].Minutes;
}
if(sSetEnd[weekday].Minutes < sSetStart[weekday].Minutes){
Activetime_length_min = 60 * 24 - (sSetStart[weekday].Minutes - sSetEnd[weekday].Minutes);
}
}
if(sSetEnd[weekday].Hours < sSetStart[weekday].Hours){
Activetime_length_min = ( (sSetEnd[weekday].Hours + 24) * 60 + sSetEnd[weekday].Minutes ) - (sSetStart[weekday].Hours*60 + sSetStart[weekday].Minutes);
}
if(sTime.Hours > sSetStart[weekday].Hours){
Currenttime_length_min = (sTime.Hours * 60 + sTime.Minutes) - (sSetStart[weekday].Hours * 60 + sSetStart[weekday].Minutes);
}
if(sTime.Hours == sSetStart[weekday].Hours){
if(sTime.Minutes > sSetStart[weekday].Minutes){
Currenttime_length_min = sTime.Minutes - sSetStart[weekday].Minutes;
}
if(sTime.Minutes < sSetStart[weekday].Minutes){
Currenttime_length_min = 60 * 24 - (sSetStart[weekday].Minutes - sTime.Minutes);
}
}
if(sTime.Hours < sSetStart[weekday].Hours){
Currenttime_length_min = ( (sTime.Hours + 24) * 60 + sTime.Minutes ) - (sSetStart[weekday].Hours*60 + sSetStart[weekday].Minutes);
}
if(Currenttime_length_min < Activetime_length_min){
return true;
}
return false;
// if(sTime.Hours > sSetStart[weekday].Hours && sTime.Hours < sSetEnd[weekday].Hours){
// return true;
// }else{
// if(sTime.Hours == sSetStart[weekday].Hours && sTime.Minutes >= sSetStart[weekday].Minutes){
// if(sTime.Hours <= sSetEnd[weekday].Hours){
// return true;
// }
// if(sTime.Hours == sSetEnd[weekday].Hours && sTime.Minutes <= sSetEnd[weekday].Minutes){
// return true;
// }
// }
// if(sTime.Hours == sSetEnd[weekday].Hours && sTime.Minutes <= sSetEnd[weekday].Minutes){
// if(sTime.Hours >= sSetStart[weekday].Hours){
// return true;
// }
// if(sTime.Hours == sSetStart[weekday].Hours && sTime.Minutes >= sSetStart[weekday].Minutes){
// return true;
// }
// }
// }
//
// return false;
}
bool System_CustomHeartbeat(uint8_t var){
if(sTime.Seconds%var == 0){
return true;
}else{
return false;
}
}
2020-01-21 15:21:05 +01:00
//keep it short, executed in ISR
void ButtonLeft_Pressed(void){
if(system_mode==Choose){
2020-08-15 14:43:43 +02:00
Change_selectorpos(false);
2020-01-21 15:21:05 +01:00
}
if(system_mode==Set){
2020-08-15 14:43:43 +02:00
Change_time(selector_pos,false);
2020-01-21 15:21:05 +01:00
}
};
//keep it short, executed in ISR
void ButtonPress_Pressed(void){
if(selector_pos==28 || selector_pos == 29 || selector_pos == 30){
if(system_mode == Choose){ //read time to sTimeSet
Read_intosettime();
}
if(system_mode == Set){ //write time back to RTC
Write_intostime();
if (HAL_RTC_SetTime(&hrtc, &sTime, RTC_FORMAT_BIN) != HAL_OK)
{
Error_Handler();
}
if (HAL_RTC_SetDate(&hrtc, &sDate, RTC_FORMAT_BCD) != HAL_OK)
{
Error_Handler();
}
}
}
2020-02-06 13:29:37 +01:00
if(system_mode == Set){
Eeprom_savesettings();
}
2020-01-21 15:21:05 +01:00
Change_systemmode();
};
//keep it short, executed in ISR
void ButtonRight_Pressed(void){
if(system_mode==Choose){
2020-08-15 14:43:43 +02:00
Change_selectorpos(true);
2020-01-21 15:21:05 +01:00
}
if(system_mode==Set){
2020-08-15 14:43:43 +02:00
Change_time(selector_pos,true);
2020-01-21 15:21:05 +01:00
}
};
void Change_systemsetmode(){
if(system_mode == Choose){
system_mode = Set;
}
if(system_mode == Set){
system_mode = Choose;
}
}
void Change_systemmode(){
system_mode == 2 ? system_mode = 0 : system_mode++;
}
void Change_time(uint8_t selectorpos, bool increase){
if(selectorpos < 28){
uint8_t setrow = selectorpos/4;
uint8_t setpos = selectorpos%4;
if(increase){
//ssetstart
if(setpos<2){
if(setpos==0){
sSetStart[setrow].Hours == 23 ? sSetStart[setrow].Hours = 0 : sSetStart[setrow].Hours ++;
}
if(setpos==1){
sSetStart[setrow].Minutes == 59 ? sSetStart[setrow].Minutes = 0 : sSetStart[setrow].Minutes ++;
}
}else{ //ssetend
if(setpos==2){
sSetEnd[setrow].Hours == 23 ? sSetEnd[setrow].Hours = 0 : sSetEnd[setrow].Hours ++;
}
if(setpos==3){
sSetEnd[setrow].Minutes == 59 ? sSetEnd[setrow].Minutes = 0 : sSetEnd[setrow].Minutes ++;
}
}
}else{
//ssetstart
if(setpos<2){
if(setpos==0){
sSetStart[setrow].Hours == 0 ? sSetStart[setrow].Hours = 23 : sSetStart[setrow].Hours --;
}
if(setpos==1){
sSetStart[setrow].Minutes == 0 ? sSetStart[setrow].Minutes = 59 : sSetStart[setrow].Minutes --;
}
}else{ //ssetend
if(setpos==2){
sSetEnd[setrow].Hours == 0 ? sSetEnd[setrow].Hours = 23 : sSetEnd[setrow].Hours --;
}
if(setpos==3){
sSetEnd[setrow].Minutes == 0 ? sSetEnd[setrow].Minutes = 59 : sSetEnd[setrow].Minutes --;
}
}
}
}
if(selectorpos >= 28 && selectorpos < 31){
if(selectorpos == 28){
if(increase){
sTimeSet.Hours == 23 ? sTimeSet.Hours = 0 : sTimeSet.Hours++;
}else{
sTimeSet.Hours == 0 ? sTimeSet.Hours = 23 : sTimeSet.Hours--;
}
}
if(selectorpos == 29){
if(increase){
sTimeSet.Minutes == 59 ? sTimeSet.Minutes = 0 : sTimeSet.Minutes++;
}else{
sTimeSet.Minutes == 0 ? sTimeSet.Minutes = 59 : sTimeSet.Minutes--;
}
}
if(selectorpos == 30){
if(increase){
setWeekday == 6 ? setWeekday = 0 : setWeekday++;
}else{
setWeekday == 0 ? setWeekday = 6 : setWeekday--;
}
}
}
}
void Change_selectorpos(bool increase){
if(increase){
selector_pos == 30 ? selector_pos = 0 : selector_pos++;
}else{
selector_pos == 0 ? selector_pos = 30 : selector_pos--;
}
};
void Read_intosettime(){
sTimeSet = sTime;
setWeekday = sDate.WeekDay;
};
void Write_intostime(){
sTime = sTimeSet;
sDate.Date = setWeekday + 2; //date hack because the RTC auto detect the weekday from the date
}
2020-02-06 13:29:37 +01:00
void Eeprom_loadsettings(){
for(uint8_t i=0 ; i < 7 ; i++){
if((uint8_t) (readEEPROMHalfWord(2*i) >> 8) >= 0 && (uint8_t) (readEEPROMHalfWord(2*i) >> 8) < 24){
sSetStart[i].Hours = (uint8_t) (readEEPROMHalfWord(2*i) >> 8);
}else{
sSetStart[i].Hours = 0;
}
if((uint8_t) readEEPROMHalfWord(2*i) >= 0 && (uint8_t) readEEPROMHalfWord(2*i) < 60){
sSetStart[i].Minutes = (uint8_t) readEEPROMHalfWord(2*i);
}else{
sSetStart[i].Minutes = 0;
}
if((uint8_t) (readEEPROMHalfWord(2*i + 14) >> 8) >= 0 && (uint8_t) (readEEPROMHalfWord(2*i + 14) >> 8) < 24){
sSetEnd[i].Hours = (uint8_t) (readEEPROMHalfWord(2*i + 14) >> 8);
}else{
sSetEnd[i].Hours = 0;
}
if((uint8_t) readEEPROMHalfWord(2*i + 14) >= 0 && (uint8_t) readEEPROMHalfWord(2*i + 14) < 60){
sSetEnd[i].Minutes = (uint8_t) readEEPROMHalfWord(2*i + 14);
}else{
sSetEnd[i].Minutes = 0;
}
2020-02-06 13:29:37 +01:00
}
}
void Eeprom_savesettings(){
enableEEPROMWriting();
for(uint8_t i = 0; i < 7; i++){
writeEEPROMHalfWord(2*i,sSetStart[i].Hours << 8 | sSetStart[i].Minutes);
writeEEPROMHalfWord(2*i + 14,sSetEnd[i].Hours << 8 | sSetEnd[i].Minutes);
}
disableEEPROMWriting();
}
2020-01-21 15:19:49 +01:00
int __io_putchar(int ch){
uint8_t c[1];
c[0] = ch & 0x00FF;
HAL_UART_Transmit(&huart1, &*c, 1, 10);
return ch;
}
int _write(int file, char *ptr, int len){
int DataIdx;
for (DataIdx = 0; DataIdx < len; DataIdx++){
__io_putchar(*ptr++);
}
return len;
}
2020-01-15 18:02:31 +01:00
/* USER CODE END 4 */
/**
* @brief This function is executed in case of error occurrence.
* @retval None
*/
void Error_Handler(void)
{
/* USER CODE BEGIN Error_Handler_Debug */
/* User can add his own implementation to report the HAL error return state */
/* USER CODE END Error_Handler_Debug */
}
#ifdef USE_FULL_ASSERT
/**
* @brief Reports the name of the source file and the source line number
* where the assert_param error has occurred.
* @param file: pointer to the source file name
* @param line: assert_param error line source number
* @retval None
*/
void assert_failed(uint8_t *file, uint32_t line)
{
/* USER CODE BEGIN 6 */
/* User can add his own implementation to report the file name and line number,
tex: printf("Wrong parameters value: file %s on line %d\r\n", file, line) */
/* USER CODE END 6 */
}
#endif /* USE_FULL_ASSERT */
/************************ (C) COPYRIGHT STMicroelectronics *****END OF FILE****/