Massive rework of the timer setup and a graphical presentation of the timer status
This commit is contained in:
parent
cc98073eac
commit
fefb84a87a
10 changed files with 511 additions and 49 deletions
|
@ -10,6 +10,7 @@
|
|||
#include "RebootScreen.h"
|
||||
#include "HeaterSettingsScreen.h"
|
||||
#include "SettingsScreen.h"
|
||||
#include "TimerChartScreen.h"
|
||||
#include <Wire.h>
|
||||
#include "../cfg/pins.h"
|
||||
#include "../cfg/BTCConfig.h"
|
||||
|
@ -161,8 +162,21 @@ CScreenManager::begin(bool bNoClock)
|
|||
_Screens.push_back(new CWiFiScreen(*_pDisplay, *this)); // comms info
|
||||
_Screens.push_back(new CSettingsScreen(*_pDisplay, *this)); // tuning info
|
||||
_SetTimeScreen = new CSetClockScreen(*_pDisplay, *this); // clock set
|
||||
_TimerScreens.push_back(new CTimerChartScreen(*_pDisplay, *this, 0)); // timer chart
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 0)); // set timer 1
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 1)); // set timer 2
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 2)); // set timer 3
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 3)); // set timer 4
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 4)); // set timer 5
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 5)); // set timer 6
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 6)); // set timer 7
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 7)); // set timer 8
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 8)); // set timer 9
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 9)); // set timer 10
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 10)); // set timer 11
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 11)); // set timer 12
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 12)); // set timer 13
|
||||
_TimerScreens.push_back(new CSetTimerScreen(*_pDisplay, *this, 13)); // set timer 14
|
||||
_SettingsScreens.push_back(new CFuelMixtureScreen(*_pDisplay, *this)); // tuning
|
||||
_SettingsScreens.push_back(new CHeaterSettingsScreen(*_pDisplay, *this)); // tuning
|
||||
|
||||
|
|
|
@ -33,6 +33,7 @@
|
|||
#include "../Protocol/helpers.h"
|
||||
#include "../Utility/NVStorage.h"
|
||||
#include <RTClib.h>
|
||||
#include "../RTC/TimerManager.h"
|
||||
|
||||
const char* briefDOW[] = { "S", "M", "T", "W", "T", "F", "S" };
|
||||
|
||||
|
@ -41,13 +42,15 @@ CSetTimerScreen::CSetTimerScreen(C128x64_OLED& display, CScreenManager& mgr, int
|
|||
_rowSel = 0;
|
||||
_colSel = 0;
|
||||
_SaveTime = 0;
|
||||
_instance = instance;
|
||||
_ConflictTime = 0;
|
||||
_conflictID = 0;
|
||||
_timerID = instance;
|
||||
}
|
||||
|
||||
void
|
||||
CSetTimerScreen::onSelect()
|
||||
{
|
||||
NVstore.getTimerInfo(_instance, _timer);
|
||||
NVstore.getTimerInfo(_timerID, _timerInfo);
|
||||
}
|
||||
|
||||
bool
|
||||
|
@ -55,13 +58,13 @@ CSetTimerScreen::show()
|
|||
{
|
||||
CScreenHeader::show();
|
||||
|
||||
char str[16];
|
||||
char str[20];
|
||||
int xPos, yPos;
|
||||
|
||||
if(_rowSel == 0) {
|
||||
NVstore.getTimerInfo(_instance, _timer);
|
||||
NVstore.getTimerInfo(_timerID, _timerInfo);
|
||||
}
|
||||
sprintf(str, " Set Timer %d ", _instance + 1);
|
||||
sprintf(str, " Set Timer %d ", _timerID + 1);
|
||||
_printInverted(0, 16, str, true);
|
||||
|
||||
if(_SaveTime) {
|
||||
|
@ -72,6 +75,25 @@ CSetTimerScreen::show()
|
|||
_printInverted(_display.xCentre(), 39, " ", true, eCentreJustify);
|
||||
_printInverted(_display.xCentre(), 34, " STORING ", true, eCentreJustify);
|
||||
}
|
||||
else if(_ConflictTime) {
|
||||
long tDelta = millis() - _ConflictTime;
|
||||
if(tDelta > 0)
|
||||
_ConflictTime = 0;
|
||||
sprintf(str, " with Timer %d ", _conflictID);
|
||||
if(_conflictID >= 10) {
|
||||
// extra space
|
||||
_printInverted(_display.xCentre(), 26, " ", true, eCentreJustify);
|
||||
_printInverted(_display.xCentre(), 45, " ", true, eCentreJustify);
|
||||
_printInverted(_display.xCentre(), 30, " Conflicts ", true, eCentreJustify);
|
||||
_printInverted(_display.xCentre(), 38, str, true, eCentreJustify);
|
||||
}
|
||||
else {
|
||||
_printInverted(_display.xCentre(), 26, " ", true, eCentreJustify);
|
||||
_printInverted(_display.xCentre(), 45, " ", true, eCentreJustify);
|
||||
_printInverted(_display.xCentre(), 30, " Conflicts ", true, eCentreJustify);
|
||||
_printInverted(_display.xCentre(), 38, str, true, eCentreJustify);
|
||||
}
|
||||
}
|
||||
else {
|
||||
// start
|
||||
xPos = 18;
|
||||
|
@ -79,10 +101,10 @@ CSetTimerScreen::show()
|
|||
_printMenuText(xPos, yPos, "On", false, eRightJustify);
|
||||
_printMenuText(xPos+18, yPos, ":");
|
||||
xPos += 6;
|
||||
sprintf(str, "%02d", _timer.start.hour);
|
||||
sprintf(str, "%02d", _timerInfo.start.hour);
|
||||
_printMenuText(xPos, yPos, str, _rowSel==1 && _colSel==0);
|
||||
xPos += 17;
|
||||
sprintf(str, "%02d", _timer.start.min);
|
||||
sprintf(str, "%02d", _timerInfo.start.min);
|
||||
_printMenuText(xPos, yPos, str, _rowSel==1 && _colSel==1);
|
||||
|
||||
// stop
|
||||
|
@ -91,10 +113,10 @@ CSetTimerScreen::show()
|
|||
_printMenuText(xPos, yPos, "Off", false, eRightJustify);
|
||||
_printMenuText(xPos+18, yPos, ":");
|
||||
xPos += 6;
|
||||
sprintf(str, "%02d", _timer.stop.hour);
|
||||
sprintf(str, "%02d", _timerInfo.stop.hour);
|
||||
_printMenuText(xPos, yPos, str, _rowSel==1 && _colSel==2);
|
||||
xPos += 17;
|
||||
sprintf(str, "%02d", _timer.stop.min);
|
||||
sprintf(str, "%02d", _timerInfo.stop.min);
|
||||
_printMenuText(xPos, yPos, str, _rowSel==1 && _colSel==3);
|
||||
|
||||
// control
|
||||
|
@ -103,14 +125,14 @@ CSetTimerScreen::show()
|
|||
_printEnabledTimers();
|
||||
|
||||
yPos = 40;
|
||||
if(_timer.repeat)
|
||||
if(_timerInfo.repeat)
|
||||
msg = "Repeat";
|
||||
else
|
||||
msg = "Once";
|
||||
if(_rowSel == 1)
|
||||
_printMenuText(xPos, yPos, msg, _colSel==5, eRightJustify);
|
||||
else
|
||||
_printInverted(xPos, yPos, msg, _timer.repeat, eRightJustify);
|
||||
_printInverted(xPos, yPos, msg, _timerInfo.repeat, eRightJustify);
|
||||
}
|
||||
// navigation line
|
||||
yPos = 53;
|
||||
|
@ -139,12 +161,22 @@ CSetTimerScreen::keyHandler(uint8_t event)
|
|||
_colSel = 4;
|
||||
}
|
||||
else { // in config fields, save new settings
|
||||
_SaveTime = millis() + 1500;
|
||||
NVstore.setTimerInfo(_instance, _timer);
|
||||
NVstore.save();
|
||||
_rowSel = 0;
|
||||
NVstore.setTimerInfo(_timerID, _timerInfo);
|
||||
_conflictID = CTimerManager::conflictTest(_timerID);
|
||||
if(_conflictID) {
|
||||
_timerInfo.enabled = 0; // cancel enabled status
|
||||
_ConflictTime = millis() + 1500;
|
||||
_ScreenManager.reqUpdate();
|
||||
}
|
||||
else {
|
||||
_SaveTime = millis() + 1500;
|
||||
_ScreenManager.reqUpdate();
|
||||
}
|
||||
_rowSel = 0;
|
||||
_colSel = 0;
|
||||
NVstore.setTimerInfo(_timerID, _timerInfo); // may have got disabled
|
||||
NVstore.save();
|
||||
}
|
||||
}
|
||||
// press LEFT - navigate fields, or screens
|
||||
if(event & key_Left) {
|
||||
|
@ -212,7 +244,7 @@ CSetTimerScreen::keyHandler(uint8_t event)
|
|||
}
|
||||
else if(_colSel == 4) {
|
||||
if(event & key_Right) {
|
||||
_timer.enabled &= 0x7f; // strip next day flag
|
||||
_timerInfo.enabled &= 0x7f; // strip next day flag
|
||||
_rowSel = 2;
|
||||
_colSel = 0;
|
||||
}
|
||||
|
@ -248,8 +280,8 @@ CSetTimerScreen::keyHandler(uint8_t event)
|
|||
break;*/
|
||||
case 2:
|
||||
// adjust selected item
|
||||
_timer.enabled ^= maskDOW;
|
||||
_timer.enabled &= 0x7f;
|
||||
_timerInfo.enabled ^= maskDOW;
|
||||
_timerInfo.enabled &= 0x7f;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
@ -271,8 +303,8 @@ CSetTimerScreen::keyHandler(uint8_t event)
|
|||
break;*/
|
||||
case 2:
|
||||
// adjust selected item
|
||||
_timer.enabled ^= maskDOW;
|
||||
_timer.enabled &= 0x7f;
|
||||
_timerInfo.enabled ^= maskDOW;
|
||||
_timerInfo.enabled &= 0x7f;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
@ -292,37 +324,37 @@ CSetTimerScreen::adjust(int dir)
|
|||
|
||||
switch(_colSel) {
|
||||
case 0:
|
||||
_timer.start.hour += dir;
|
||||
ROLLUPPERLIMIT(_timer.start.hour, 23, 0);
|
||||
ROLLLOWERLIMIT(_timer.start.hour, 0, 23);
|
||||
_timerInfo.start.hour += dir;
|
||||
ROLLUPPERLIMIT(_timerInfo.start.hour, 23, 0);
|
||||
ROLLLOWERLIMIT(_timerInfo.start.hour, 0, 23);
|
||||
break;
|
||||
case 1:
|
||||
_timer.start.min += dir;
|
||||
ROLLUPPERLIMIT(_timer.start.min, 59, 0);
|
||||
ROLLLOWERLIMIT(_timer.start.min, 0, 59);
|
||||
_timerInfo.start.min += dir;
|
||||
ROLLUPPERLIMIT(_timerInfo.start.min, 59, 0);
|
||||
ROLLLOWERLIMIT(_timerInfo.start.min, 0, 59);
|
||||
break;
|
||||
case 2:
|
||||
_timer.stop.hour += dir;
|
||||
ROLLUPPERLIMIT(_timer.stop.hour, 23, 0);
|
||||
ROLLLOWERLIMIT(_timer.stop.hour, 0, 23);
|
||||
_timerInfo.stop.hour += dir;
|
||||
ROLLUPPERLIMIT(_timerInfo.stop.hour, 23, 0);
|
||||
ROLLLOWERLIMIT(_timerInfo.stop.hour, 0, 23);
|
||||
break;
|
||||
case 3:
|
||||
_timer.stop.min += dir;
|
||||
ROLLUPPERLIMIT(_timer.stop.min, 59, 0);
|
||||
ROLLLOWERLIMIT(_timer.stop.min, 0, 59);
|
||||
_timerInfo.stop.min += dir;
|
||||
ROLLUPPERLIMIT(_timerInfo.stop.min, 59, 0);
|
||||
ROLLLOWERLIMIT(_timerInfo.stop.min, 0, 59);
|
||||
break;
|
||||
case 4:
|
||||
if(_rowSel == 1) {
|
||||
_timer.enabled &= 0x80; // ensure specific day flags are cleared
|
||||
_timer.enabled ^= 0x80; // toggle next day flag
|
||||
_timerInfo.enabled &= 0x80; // ensure specific day flags are cleared
|
||||
_timerInfo.enabled ^= 0x80; // toggle next day flag
|
||||
}
|
||||
if(_rowSel == 2) {
|
||||
_timer.enabled &= 0x7f; // ensure next day flag is cleared
|
||||
_timer.enabled ^= maskDOW; // toggle flag for day of week
|
||||
_timerInfo.enabled &= 0x7f; // ensure next day flag is cleared
|
||||
_timerInfo.enabled ^= maskDOW; // toggle flag for day of week
|
||||
}
|
||||
break;
|
||||
case 5:
|
||||
_timer.repeat = !_timer.repeat;
|
||||
_timerInfo.repeat = !_timerInfo.repeat;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
@ -334,10 +366,10 @@ CSetTimerScreen::_printEnabledTimers()
|
|||
int xPos = _display.width() - border;
|
||||
int yPos = 28;
|
||||
|
||||
if(_timer.enabled == 0x00 && _rowSel != 2) {
|
||||
if(_timerInfo.enabled == 0x00 && _rowSel != 2) {
|
||||
_printMenuText(xPos, yPos, "Disabled", _colSel==4, eRightJustify);
|
||||
}
|
||||
else if(_timer.enabled & 0x80) {
|
||||
else if(_timerInfo.enabled & 0x80) {
|
||||
if(_rowSel==1 && _colSel==4)
|
||||
_printMenuText(xPos, yPos, "Enabled", true, eRightJustify);
|
||||
else
|
||||
|
@ -352,7 +384,7 @@ CSetTimerScreen::_printEnabledTimers()
|
|||
int xSel = xPos + _colSel * dayWidth;
|
||||
for(int i=0; i<7; i++) {
|
||||
int dayMask = 0x01 << i;
|
||||
_printInverted(xPos, yPos, briefDOW[i], _timer.enabled & dayMask);
|
||||
_printInverted(xPos, yPos, briefDOW[i], _timerInfo.enabled & dayMask);
|
||||
xPos += dayWidth;
|
||||
}
|
||||
if(_rowSel == 2) {
|
||||
|
|
|
@ -33,9 +33,11 @@ class CProtocol;
|
|||
class CSetTimerScreen : public CScreenHeader {
|
||||
int _rowSel;
|
||||
int _colSel;
|
||||
int _instance;
|
||||
int _timerID;
|
||||
unsigned long _SaveTime;
|
||||
sTimer _timer;
|
||||
unsigned long _ConflictTime;
|
||||
int _conflictID;
|
||||
sTimer _timerInfo;
|
||||
void adjust(int dir);
|
||||
void _printEnabledTimers();
|
||||
|
||||
|
|
175
Arduino/BTCDieselHeater/src/OLED/TimerChartScreen.cpp
Normal file
175
Arduino/BTCDieselHeater/src/OLED/TimerChartScreen.cpp
Normal file
|
@ -0,0 +1,175 @@
|
|||
/*
|
||||
* This file is part of the "bluetoothheater" distribution
|
||||
* (https://gitlab.com/mrjones.id.au/bluetoothheater)
|
||||
*
|
||||
* Copyright (C) 2018 Ray Jones <ray@mrjones.id.au>
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
*
|
||||
*/
|
||||
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// CTimerChartScreen
|
||||
//
|
||||
// This screen shows the timers as a chart for the entire week
|
||||
//
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
|
||||
#include "TimerChartScreen.h"
|
||||
#include "KeyPad.h"
|
||||
#include "../Protocol/helpers.h"
|
||||
#include "../Utility/NVStorage.h"
|
||||
#include <RTClib.h>
|
||||
#include "fonts/MiniFont.h"
|
||||
#include "../RTC/TimerManager.h"
|
||||
|
||||
|
||||
static uint16_t timerMap[24*60];
|
||||
static uint16_t timerIDs[24*60];
|
||||
|
||||
CTimerChartScreen::CTimerChartScreen(C128x64_OLED& display, CScreenManager& mgr, int instance) : CScreenHeader(display, mgr)
|
||||
{
|
||||
_rowSel = 0;
|
||||
_colSel = 0;
|
||||
_SaveTime = 0;
|
||||
_instance = instance;
|
||||
}
|
||||
|
||||
void
|
||||
CTimerChartScreen::onSelect()
|
||||
{
|
||||
CTimerManager::createMap(0x3fff, timerMap, timerIDs);
|
||||
CTimerManager::condenseMap(timerMap, 12);
|
||||
}
|
||||
|
||||
bool
|
||||
CTimerChartScreen::show()
|
||||
{
|
||||
_display.clearDisplay();
|
||||
|
||||
CTransientFont AF(_display, &miniFontInfo); // temporarily use a mini font
|
||||
|
||||
_printMenuText(0, 7, "S");
|
||||
_printMenuText(0, 14, "M");
|
||||
_printMenuText(0, 21, "T");
|
||||
_printMenuText(0, 28, "W");
|
||||
_printMenuText(0, 35, "T");
|
||||
_printMenuText(0, 42, "F");
|
||||
_printMenuText(0, 49, "S");
|
||||
|
||||
int hour0 = 8;
|
||||
int linespacing = 7;
|
||||
|
||||
for(int tick = 0; tick < 24; tick += 3) {
|
||||
int xpos = tick * 5 + hour0;
|
||||
_display.setCursor(xpos, 0);
|
||||
_display.print(tick);
|
||||
for(int dow = 0; dow < 7; dow++) {
|
||||
int ypos = dow*linespacing + 8;
|
||||
_display.drawFastVLine(xpos, ypos, 3, WHITE); // solid bar
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
for(int dow = 0; dow < 7; dow++) {
|
||||
int day = 0x01 << dow;
|
||||
int ypos = dow*linespacing + 7; // top of first line
|
||||
int pixel = 0;
|
||||
int subpixel = 0;
|
||||
for(int interval = 0; interval < 120; interval++) {
|
||||
// if(Chart[interval] & day) {
|
||||
if(timerMap[interval] & day) {
|
||||
// if(Chart[interval] & (day << 8)) {
|
||||
if(timerMap[interval] & (day << 8)) {
|
||||
// one shot timer - draw peppered
|
||||
for(int yscan = interval & 1; yscan < 6; yscan+=2)
|
||||
_display.drawPixel(interval+hour0, ypos+yscan, WHITE); // peppered vertical bar
|
||||
}
|
||||
else {
|
||||
// repeating timer =- draw solid
|
||||
_display.drawFastVLine(interval+hour0, ypos, 6, WHITE); // solid bar
|
||||
}
|
||||
}
|
||||
else {
|
||||
if(pixel == 0) // every 5th pixel draw a base line
|
||||
_display.drawPixel(interval+hour0, ypos+2, subpixel ? WHITE : BLACK); // base line
|
||||
}
|
||||
pixel++;
|
||||
if(pixel > 4) {
|
||||
pixel = 0;
|
||||
subpixel++;
|
||||
ROLLUPPERLIMIT(subpixel, 2, 0);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
bool
|
||||
CTimerChartScreen::keyHandler(uint8_t event)
|
||||
{
|
||||
static bool bHeld = false;
|
||||
// handle initial key press
|
||||
if(event & keyPressed) {
|
||||
bHeld = false;
|
||||
// press CENTRE
|
||||
if(event & key_Centre) {
|
||||
_ScreenManager.selectTimerScreen(false); // exit: return to clock screen
|
||||
}
|
||||
// press LEFT - navigate fields, or screens
|
||||
if(event & key_Left) {
|
||||
_ScreenManager.prevScreen();
|
||||
}
|
||||
// press RIGHT - navigate fields, or screens
|
||||
if(event & key_Right) {
|
||||
_ScreenManager.nextScreen();
|
||||
}
|
||||
// press UP
|
||||
if(event & key_Up) {
|
||||
}
|
||||
// press DOWN
|
||||
if(event & key_Down) {
|
||||
}
|
||||
}
|
||||
|
||||
// handle held down keys
|
||||
if(event & keyRepeat) {
|
||||
bHeld = true;
|
||||
}
|
||||
|
||||
if(event & keyReleased) {
|
||||
if(!bHeld) {
|
||||
if(event & key_Left) {
|
||||
}
|
||||
// released DOWN - can only leave adjustment by using OK (centre button)
|
||||
if(event & key_Down) {
|
||||
// adjust selected item
|
||||
}
|
||||
if(event & key_Right) {
|
||||
}
|
||||
// released UP
|
||||
if(event & key_Up) {
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
_ScreenManager.reqUpdate();
|
||||
return true;
|
||||
}
|
||||
|
||||
|
46
Arduino/BTCDieselHeater/src/OLED/TimerChartScreen.h
Normal file
46
Arduino/BTCDieselHeater/src/OLED/TimerChartScreen.h
Normal file
|
@ -0,0 +1,46 @@
|
|||
/*
|
||||
* This file is part of the "bluetoothheater" distribution
|
||||
* (https://gitlab.com/mrjones.id.au/bluetoothheater)
|
||||
*
|
||||
* Copyright (C) 2018 Ray Jones <ray@mrjones.id.au>
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
*
|
||||
*/
|
||||
|
||||
#ifndef __TIMERCHARTSCREEN_H__
|
||||
#define __TIMERCHARTSCREEN_H__
|
||||
|
||||
#include <stdint.h>
|
||||
#include "ScreenHeader.h"
|
||||
#include "../Utility/NVStorage.h"
|
||||
|
||||
class C128x64_OLED;
|
||||
class CScreenManager;
|
||||
class CProtocol;
|
||||
|
||||
class CTimerChartScreen : public CScreenHeader {
|
||||
int _rowSel;
|
||||
int _colSel;
|
||||
int _instance;
|
||||
unsigned long _SaveTime;
|
||||
|
||||
public:
|
||||
CTimerChartScreen(C128x64_OLED& display, CScreenManager& mgr, int instance);
|
||||
void onSelect();
|
||||
bool show();
|
||||
bool keyHandler(uint8_t event);
|
||||
};
|
||||
|
||||
#endif
|
|
@ -155,6 +155,11 @@ const uint8_t miniFontBitmaps[] PROGMEM =
|
|||
0xa8, // # # #
|
||||
0x88, // # #
|
||||
|
||||
// @75 'W' (3 pixels wide)
|
||||
0xf8, // #####
|
||||
0x40, // #
|
||||
0xf8, // #####
|
||||
|
||||
};
|
||||
|
||||
// Character descriptors for a 3x5 font
|
||||
|
@ -192,7 +197,7 @@ const FONT_CHAR_INFO miniFontDescriptors[] PROGMEM =
|
|||
{0, 0, 0}, // 'J'
|
||||
{0, 0, 0}, // 'K'
|
||||
{3, 5, 48}, // 'L'
|
||||
{0, 0, 0}, // 'M'
|
||||
{3, 5, 75}, // 'M'
|
||||
{0, 0, 0}, // 'N'
|
||||
{0, 0, 0}, // 'O'
|
||||
{3, 5, 51}, // 'P'
|
||||
|
|
138
Arduino/BTCDieselHeater/src/RTC/TimerManager.cpp
Normal file
138
Arduino/BTCDieselHeater/src/RTC/TimerManager.cpp
Normal file
|
@ -0,0 +1,138 @@
|
|||
/*
|
||||
* This file is part of the "bluetoothheater" distribution
|
||||
* (https://gitlab.com/mrjones.id.au/bluetoothheater)
|
||||
*
|
||||
* Copyright (C) 2018 Ray Jones <ray@mrjones.id.au>
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
*
|
||||
*/
|
||||
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// CTimerManager
|
||||
//
|
||||
// This provides management of the timers
|
||||
//
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
|
||||
#include <Arduino.h>
|
||||
#include "TimerManager.h"
|
||||
#include "../Utility/NVStorage.h"
|
||||
|
||||
// create a bitmap that describes the pattern of on/off times
|
||||
void
|
||||
CTimerManager::createMap(int timerMask, uint16_t timerMap[24*60], uint16_t timerIDs[24*60])
|
||||
{
|
||||
int maxPoints = 24*60;
|
||||
memset(timerMap, 0, 24*60*sizeof(uint16_t));
|
||||
memset(timerIDs, 0, 24*60*sizeof(uint16_t));
|
||||
|
||||
for(int timerID=0; timerID < 14; timerID++) {
|
||||
// only process timer if it is nominated in timerMask (bitfield), timer0 = bit0 .. timerN = bitN
|
||||
uint16_t timerBit = 0x0001 << timerID;
|
||||
if(timerMask & timerBit) {
|
||||
sTimer timer;
|
||||
// get timer settings
|
||||
NVstore.getTimerInfo(timerID, timer);
|
||||
// and add info to map if enabled
|
||||
if(timer.enabled) {
|
||||
// create linear minute of day values for start & stop
|
||||
// note that if stop < start, that is treated as a timer that rolls over midnight
|
||||
int timestart = timer.start.hour * 60 + timer.start.min; // linear minute of day
|
||||
int timestop = timer.stop.hour * 60 + timer.stop.min;
|
||||
for(int dayMinute = 0; dayMinute < maxPoints; dayMinute++) {
|
||||
for(int day = 0x01; day != 0x80; day <<= 1) {
|
||||
if(timer.enabled & day || timer.enabled & 0x80) { // specific or everyday
|
||||
uint16_t activeday = day; // may also hold non repeat flag later
|
||||
if(!timer.repeat) {
|
||||
// flag timers that should get cancelled
|
||||
activeday |= (activeday << 8); // combine one shot status in MS byte
|
||||
}
|
||||
if(timestop > timestart) {
|
||||
// treat normal start < stop times (within same day)
|
||||
if((dayMinute >= timestart) && (dayMinute < timestop)) {
|
||||
timerMap[dayMinute] |= activeday;
|
||||
timerIDs[dayMinute] |= timerBit;
|
||||
}
|
||||
}
|
||||
else {
|
||||
// time straddles a day, start > stop, special treatment required
|
||||
if(dayMinute >= timestart) {
|
||||
// true from start until midnight
|
||||
timerMap[dayMinute] |= activeday;
|
||||
timerIDs[dayMinute] |= timerBit;
|
||||
}
|
||||
if(dayMinute < timestop) {
|
||||
// after midnight, before stop time, i.e. next day
|
||||
// adjust for next day, taking care to wrap week
|
||||
if(day & 0x40) // last day of week?
|
||||
activeday >>= 6; // roll back to start of week
|
||||
else
|
||||
activeday <<= 1; // next day
|
||||
timerMap[dayMinute] |= activeday;
|
||||
timerIDs[dayMinute] |= timerBit;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
void
|
||||
CTimerManager::condenseMap(uint16_t timerMap[24*60], int factor)
|
||||
{
|
||||
int maxPoints = 24*60;
|
||||
|
||||
int opIndex = 0;
|
||||
for(int dayMinute = 0; dayMinute < maxPoints; ) {
|
||||
uint16_t condense = 0;
|
||||
for(int subInterval = 0; subInterval < factor; subInterval++) {
|
||||
condense |= timerMap[dayMinute++];
|
||||
if(dayMinute == maxPoints) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
timerMap[opIndex++] = condense;
|
||||
}
|
||||
}
|
||||
|
||||
uint16_t otherTimers[24*60];
|
||||
uint16_t selectedTimer[24*60];
|
||||
uint16_t timerIDs[24*60];
|
||||
|
||||
int
|
||||
CTimerManager::conflictTest(int timerID)
|
||||
{
|
||||
int selectedMask = 0x0001 << timerID;
|
||||
|
||||
createMap(selectedMask, selectedTimer, timerIDs); // create a map for the nominated timer (under test)
|
||||
createMap(0x3fff & ~selectedMask, otherTimers, timerIDs); // create a map for all other timers, and get their unique IDs
|
||||
for(int i=0; i< 24*60; i++) {
|
||||
if(otherTimers[i] & selectedTimer[i]) { // both have the same day bit set - CONFLICT!
|
||||
uint16_t timerBit = timerIDs[i];
|
||||
int ID = 0;
|
||||
while(timerBit) {
|
||||
timerBit >>= 1;
|
||||
ID++;
|
||||
}
|
||||
return ID;
|
||||
}
|
||||
}
|
||||
return 0; // no conflicts :-)
|
||||
}
|
43
Arduino/BTCDieselHeater/src/RTC/TimerManager.h
Normal file
43
Arduino/BTCDieselHeater/src/RTC/TimerManager.h
Normal file
|
@ -0,0 +1,43 @@
|
|||
/*
|
||||
* This file is part of the "bluetoothheater" distribution
|
||||
* (https://gitlab.com/mrjones.id.au/bluetoothheater)
|
||||
*
|
||||
* Copyright (C) 2018 Ray Jones <ray@mrjones.id.au>
|
||||
*
|
||||
* This program is free software: you can redistribute it and/or modify
|
||||
* it under the terms of the GNU General Public License as published by
|
||||
* the Free Software Foundation, either version 3 of the License, or
|
||||
* (at your option) any later version.
|
||||
*
|
||||
* This program is distributed in the hope that it will be useful,
|
||||
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
* GNU General Public License for more details.
|
||||
*
|
||||
* You should have received a copy of the GNU General Public License
|
||||
* along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
*
|
||||
*/
|
||||
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
//
|
||||
// CTimerManager
|
||||
//
|
||||
// This provides management of the timers
|
||||
//
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
|
||||
#ifndef __TIMERMANAGER_H__
|
||||
#define __TIMERMANAGER_H__
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
class CTimerManager {
|
||||
public:
|
||||
static void createMap(int timermask, uint16_t map[24*60], uint16_t timerIDs[24*60]);
|
||||
static void condenseMap(uint16_t timerMap[24*60], int factor);
|
||||
static int conflictTest(int timerID);
|
||||
};
|
||||
|
||||
#endif //__TIMERMANAGER_H__
|
|
@ -178,7 +178,7 @@ CHeaterStorage::getGlowDrive()
|
|||
void
|
||||
CHeaterStorage::getTimerInfo(int idx, sTimer& timerInfo)
|
||||
{
|
||||
if(idx >= 0 && idx <=1) {
|
||||
if(idx >= 0 && idx < 14) {
|
||||
timerInfo = _calValues.timer[idx];
|
||||
}
|
||||
}
|
||||
|
@ -186,7 +186,7 @@ CHeaterStorage::getTimerInfo(int idx, sTimer& timerInfo)
|
|||
void
|
||||
CHeaterStorage::setTimerInfo(int idx, const sTimer& timerInfo)
|
||||
{
|
||||
if(idx >= 0 && idx <=1) {
|
||||
if(idx >= 0 && idx < 14) {
|
||||
_calValues.timer[idx] = timerInfo;
|
||||
}
|
||||
}
|
||||
|
@ -239,7 +239,7 @@ CESP32HeaterStorage::load()
|
|||
{
|
||||
DebugPort.println("Reading from NV storage");
|
||||
loadHeater();
|
||||
for(int i=0; i<2; i++) {
|
||||
for(int i=0; i<14; i++) {
|
||||
loadTimer(i);
|
||||
}
|
||||
loadUI();
|
||||
|
@ -250,7 +250,7 @@ CESP32HeaterStorage::save()
|
|||
{
|
||||
DebugPort.println("Saving to NV storage");
|
||||
saveHeater();
|
||||
for(int i=0; i<2; i++) {
|
||||
for(int i=0; i<14; i++) {
|
||||
saveTimer(i);
|
||||
}
|
||||
saveUI();
|
||||
|
@ -302,7 +302,8 @@ CESP32HeaterStorage::loadTimer(int idx)
|
|||
validatedLoad("stopHour", timer.stop.hour, 0, s8inBounds, 0, 23);
|
||||
validatedLoad("stopMin", timer.stop.min, 0, s8inBounds, 0, 59);
|
||||
validatedLoad("enabled", timer.enabled, 0, u8inBounds, 0, 255); // all 8 bits used!
|
||||
validatedLoad("repea*t", timer.repeat, 0, u8inBounds, 0, 1);
|
||||
validatedLoad("repeat", timer.repeat, 0, u8inBounds, 0, 1);
|
||||
validatedLoad("temperature", timer.temperature, 22, u8inBounds, 8, 35);
|
||||
preferences.end();
|
||||
}
|
||||
|
||||
|
@ -319,6 +320,7 @@ CESP32HeaterStorage::saveTimer(int idx)
|
|||
preferences.putChar("stopMin", timer.stop.min);
|
||||
preferences.putUChar("enabled", timer.enabled);
|
||||
preferences.putUChar("repeat", timer.repeat);
|
||||
preferences.putUChar("temperature", timer.temperature);
|
||||
preferences.end();
|
||||
}
|
||||
|
||||
|
|
|
@ -77,15 +77,18 @@ struct sTimer {
|
|||
sHourMin stop; // stop time
|
||||
uint8_t enabled; // timer enabled - each bit is a day of week flag
|
||||
uint8_t repeat; // repeating timer
|
||||
uint8_t temperature;
|
||||
sTimer() {
|
||||
enabled = 0;
|
||||
repeat = false;
|
||||
temperature = 22;
|
||||
};
|
||||
sTimer& operator=(const sTimer& rhs) {
|
||||
start = rhs.start;
|
||||
stop = rhs.stop;
|
||||
enabled = rhs.enabled;
|
||||
repeat = rhs.repeat;
|
||||
temperature = rhs.temperature;
|
||||
};
|
||||
void init() {
|
||||
start.hour = 0;
|
||||
|
@ -94,6 +97,7 @@ struct sTimer {
|
|||
stop.min = 0;
|
||||
enabled = 0;
|
||||
repeat = 0;
|
||||
temperature = 22;
|
||||
};
|
||||
bool valid() {
|
||||
bool retval = true;
|
||||
|
@ -102,6 +106,7 @@ struct sTimer {
|
|||
retval &= (stop.hour >= 0 && stop.hour < 24);
|
||||
retval &= (stop.min >= 0 && stop.min < 60);
|
||||
retval &= repeat <= 2;
|
||||
retval &= (temperature >= 8 && temperature <= 35);
|
||||
return retval;
|
||||
};
|
||||
};
|
||||
|
@ -111,7 +116,7 @@ struct sNVStore {
|
|||
sHeater Heater;
|
||||
long DimTime;
|
||||
uint8_t degF;
|
||||
sTimer timer[2];
|
||||
sTimer timer[14];
|
||||
bool valid();
|
||||
void init();
|
||||
};
|
||||
|
|
Loading…
Reference in a new issue