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.
MicroDexed/third-party/LCDMenuLib2/src/LCDMenuLib2.h

361 lines
23 KiB

/* ******************************************************************************
* LCDMenuLib2 (LCDML)
* ******************************************************************************
*
* MIT License
*
* Copyright (c) [2021] [Nils Feldkämper]
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in all
* copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*
* ******************************************************************************
*
* BUG / ISSUES REPORTING
* https://github.com/Jomelo/LCDMenuLib2/issues
*
* ARDUINO FORUM
* http://forum.arduino.cc/index.php?topic=73816.0
*
* ******************************************************************************
*/
// File header
#ifndef LCDMenuLib2_h
#define LCDMenuLib2_h
// ####################### //
// This following defines can be changed
// ####################### //
// you can change this parameters
//#define _LCDML_cfg_use_ram // enable this line when you want to use the ram mode
// set the number of custom events (this could be a button ore something else)
#define _LCDML_CE_cb_function_cnt 4 // this is the number of custom event callback functions which are supported
// 4 = default
// enable debug strings (remove comments from this line)
//#define LCDML_DBG 1
// debug special method groups // enable a flag to control the function call order
#define LCDML_DBG_function_name_LOOP 0
#define LCDML_DBG_function_name_MENU 0
#define LCDML_DBG_function_name_FUNC 0
#define LCDML_DBG_function_name_BT 0
#define LCDML_DBG_function_name_CE 0
#define LCDML_DBG_function_name_OTHER 0
#define LCDML_DBG_function_name_DISP 0
#define LCDML_DBG_function_name_TIMER 0
#define LCDML_DBG_function_name_SCREEN 0
// ####################### //
// Do nothing change here
// ####################### //
// create DBG_print makro when debugging is enabled
#ifdef LCDML_DBG
#define DBG_print(enable, str) if(enable == 1) { Serial.print(str); }
#define DBG_println(enable, str) if(enable == 1) { Serial.println(str); }
#define DBG_printstrln(enable, str, val, form) if(enable == 1) { Serial.print(str); Serial.println(val, form);}
#else
#define DBG_print(enable, str)
#define DBG_println(enable, str)
#define DBG_printstrln(enable, str, val, form)
#endif
// Version
#define _LCDML_VERSION "LCDML2 v2.2.7"
// this makro is for unused variables which exists for compatibility things ...
#define LCDML_UNUSED(expr) do { (void)(expr); } while (0)
// Include Arduino ios
#include "Arduino.h"
// Arduino specific settings
#if ARDUINO >= 160
#define _LCDMenuLib2_arduino_version 1 // for new Arduino version like 1.6.0 and higher
#else
#define _LCDMenuLib2_arduino_version 0 // for old Arduino version like 1.0.6 or 1.0.5
#endif
// ESP specific settings
#if defined( ESP8266 ) || defined( ESP32 )
#ifndef _LCDML_cfg_use_ram
#define _LCDML_cfg_use_ram
#endif
#endif
// 64 this is the number of events which are supported
#define _LCDML_CE_events_cnt 64
// No function constant
#define _LCDML_NO_FUNC 255
#define _LCDML_FIRST_ELEMENT_ID 255
// Bit pos control flags
#define _LCDML_REG_control_free_7 7
#define _LCDML_REG_control_ce_init_setup 6
#define _LCDML_REG_control_content_ref_is_displayed 5
#define _LCDML_REG_control_bt_init_setup 4
#define _LCDML_REG_control_update_direct 3
#define _LCDML_REG_control_refresh_all_on_button_action 2
#define _LCDML_REG_control_rollover 1
#define _LCDML_REG_control_en_use_dyn_elements_as_menu 0
// screensaver, jump to function, go Root, ...
#define _LCDML_REG_special_setCursorTo 7
#define _LCDML_REG_special_OTHER_function_active 6
#define _LCDML_REG_special_free_5 5
#define _LCDML_REG_special_free_4 4
#define _LCDML_REG_special_free_3 3
#define _LCDML_REG_special_goRoot 2
#define _LCDML_REG_special_disable_scroll 1
#define _LCDML_REG_special_disable_screensaver 0
// menu function flags
#define _LCDML_REG_MenuFunction_called_from_menu 7
#define _LCDML_REG_MenuFunction_free_6 6
#define _LCDML_REG_MenuFunction_free_5 5
#define _LCDML_REG_MenuFunction_goBackToFuncID 4
#define _LCDML_REG_MenuFunction_setCursorToID 3
#define _LCDML_REG_MenuFunction_close_active 2
#define _LCDML_REG_MenuFunction_setup 1
#define _LCDML_REG_MenuFunction_end 0
// button flags
#define _LCDML_REG_button_custom_2 7
#define _LCDML_REG_button_custom_1 6
#define _LCDML_REG_button_quit 5
#define _LCDML_REG_button_right 4
#define _LCDML_REG_button_left 3
#define _LCDML_REG_button_down 2
#define _LCDML_REG_button_up 1
#define _LCDML_REG_button_enter 0
// display update handling
#define _LCDML_REG_update_content 7
#define _LCDML_REG_update_cursor 6
#define _LCDML_REG_update_menu 5
#define _LCDML_REG_update_menu_function_content 4
#define _LCDML_REG_update_update_dyn_content 3
#define _LCDML_REG_update_free_2 2
#define _LCDML_REG_update_free_1 1
#define _LCDML_REG_update_free_0 0
// reg control
#define _LCDML_REG_CTRL__button 0
#define _LCDML_REG_CTRL__events 1
// Configure Arduino flash lib and load it*/
#ifndef __PROG_TYPES_COMPAT__
#define __PROG_TYPES_COMPAT__
#endif
// Include PGMSPACE
#ifndef _LCDML_cfg_use_ram
#include <avr/pgmspace.h>
#endif
// Include menu class
#include "LCDMenuLib2_typedef.h"
#include "LCDMenuLib2_menu.h"
// Include macros for this lib
#include "LCDMenuLib2_macros.h"
//# =======================
//# LCD Menu Lib
//# =======================
class LCDMenuLib2
{
private:
// object pointer
LCDMenuLib2_menu *curMenu; // currrent menu structure
// callback functions
LCDML_FuncPtr callback_menuControl; // a callback function which checks the input buttons
LCDML_FuncPtr callback_contentUpdate; // a callback function which contains the menu function content
LCDML_FuncPtr callback_contentClear; // a callback function which clears the display
LCDML_FuncPtr_pu8 cb_screensaver; // a callback function as screensaver (a normal menu function, but a defined name)
LCDML_FuncPtr_pu8 ce_cb[_LCDML_CE_cb_function_cnt]; // callback functions for custom events
// activ menu values
LCDML_FuncPtr_pu8 actMenu_cb_function; // Menu Function callback
uint8_t actMenu_id; // Name of this menu
uint8_t actMenu_param; // Parameter this menu
uint8_t actMenu_lastFuncID; // History of the last three active menu functions
uint8_t actMenu_cursorPositionID; // current cursor position id
uint8_t actMenu_lastCursorPositionID; // Save the last Cursor position before a new function was called
unsigned long actMenu_default_time; // default loop time
uint64_t actMenu_ce_mask; // ce mask
// jump To variables
uint8_t jT_id; // contains the jumpTo id
uint8_t jT_param; // contains the jumpTo param
uint8_t jT_paramOld; // contains the jumpTo param
LCDML_FuncPtr_pu8 jT_function; // contains the jumpTo function
// menu intern values
uint8_t last_element_id; // the id of the last element
uint8_t window_rows; // the maximum rows of the current windows (1 is the minium)
uint8_t window_start; // the window start
uint8_t cursor_obj_pos;
uint8_t cursor_pos; // the current cursor position on the lcd
uint8_t cursor_pos_abs;
uint8_t layer; // contains the current layer
// variables with bitfields => bit register
uint8_t REG_control; // control flags
uint8_t REG_button;
uint8_t REG_MenuFunction; // control flags for menu functions
uint8_t REG_special; // control flags for special function like screensaver, jumpTo, setCursorTo, goRoot, ..
uint8_t REG_update; // control flags to update the content
uint64_t REG_custom_event; // control flags for custom event actions
// variables for handling with menu function
uint8_t goBackCnt; // save the layer to go back
// timer variable for the loop time of a menu function
unsigned long menuFunction_loopTimer;
// timer variable for the screensaver
unsigned long screensaver_timer;
unsigned long screensaver_default_time;
// private methods
void MENU_goInto(void); // activate the menu under the cursor
void MENU_goBack(void); // move to the parent menu
void MENU_doScroll(uint8_t state); // scroll the menu
uint8_t MENU_countChilds(LCDMenuLib2_menu *menu, uint8_t all=false); // how many children exists on next layer
void MENU_initFirstElement(void); // set counter to the first object child (when first element is hidden, go to the first visible element)
void MENU_resetActiveMenu(void); // clear all neccessary variables
// callback function
void FUNC_call(void);
public:
// constructor
LCDMenuLib2(LCDMenuLib2_menu &p_r ,const uint8_t p_rows, const uint8_t p_cols, LCDML_FuncPtr contentUpdate, LCDML_FuncPtr contentClear, LCDML_FuncPtr menuControl);
// init method
void init(uint8_t); // initialisation of the menu / reset the complete menu
// loop methods
void loop(void); // call the loop_menu and the loop_control function
void loop_control(void); // call callback function to control the menu
void loop_menu(void); // checks if a button or a menu function are needed and call them
// menu methods
void MENU_display(uint8_t update=0); // update the current menu structure but do not display it
void MENU_goRoot(void); // set the cursor to the root element
void MENU_enRollover(void); // enable menu rollover (first to last or last to first element on a layer)
void MENU_disRollover(void); // disable the menu rollover function
void MENU_enScroll(void); // enable the scroll function (this function can be disabled on dyn content elements and when a encoder is used)
void MENU_disScroll(void); // disable the scroll function to catch the cursor to an dyn content element when a encoder is used
bool MENU_getScrollDisableStatus(void); // check if the scroll function is disabled
uint8_t MENU_getLayer(void); // get the current layer of the menu
uint8_t MENU_getCursorPos(void); // get the current cursor position (row 0, row 1, row 2, ...)
uint8_t MENU_getCursorPosAbs(void); // get the current cursor postion absolute (..., row 10, row 11, ..) based on the menu structure
uint8_t MENU_getCursorObjPos(void); // returns the cursor object position
uint8_t MENU_getChilds(void); // get the current number of childs on this layer
uint8_t MENU_getParentID(uint8_t p_layer=0); // get the parent id of a selected higher layer
uint8_t MENU_getScroll(void); // get the current scroll value
uint8_t MENU_getLastActiveFunctionID(void); // returns the id of the last active function
uint8_t MENU_getLastCursorPositionID(void); // returns the last cursor position function id
uint8_t MENU_getElementIDFromCursorPos(void); // returns the menu element id from current cursor position
void MENU_updateAllCondetions(void); // update all menu element condetions
void MENU_setDynFunctionContentUpdate(void); // this is a special function for dynamic content to update the dynamic content again
void MENU_clearDynFunctionContentUpdate(void); // this is a special function to clear the dynamic content update. this function is called internally
void MENU_allCondetionRefresh(void); // update all condetions
void MENU_enUseDynElementsWithSubElements(void); // enable subelements for dynamic menu elements
void MENU_disUseDynElementsWithSubElements(void); // disable subelements for dynamic menu elements
void MENU_enRefreshAllOnButtonAction(void); // enable update on every button action the complete menu
void MENU_disRefreshAllOnButtonAction(void); // disable update on every button action the complete menu
void MENU_setDynRContent(void); // enable dynamic refreshed content is shown
void MENU_clearDynRContent(void); // disable dynamic refreshed content is shown
bool MENU_checkDynRContent(void); // get dynamic refreshed content is shown
LCDMenuLib2_menu * MENU_getDisplayedObj(void); // get the objection with the current content to display
LCDMenuLib2_menu * MENU_getCurrentObj(void); // get the current menu child object
LCDMenuLib2_menu * MENU_getRootObj(void); // get the root menu object
// button / event / reg methods
bool BT_setup(void); // check if the button initialisation was done
bool CE_setup(void); // check if the button initialisation was done
//
void CE_setOnChangeCbFunction(uint8_t p_event, LCDML_FuncPtr_pu8 p_function); // add callback function
void CE_clearOnChangeCbFunction(uint8_t p_event); // remove callback function
//
void REG_set(uint8_t p_reg, uint8_t p_val);
void REG_reset(uint8_t p_reg, uint8_t p_val);
void REG_resetAll(uint8_t p_reg);
bool REG_check(uint8_t p_reg, uint8_t p_val);
bool REG_checkAny(uint8_t p_reg);
// display methods
void DISP_update(void); // display the content and update the menu structure
void DISP_menuUpdate(void); // display the content but do not update the menu structure
void DISP_clear(void); // calls the callback function to clear the display
bool DISP_checkMenuUpdate(void); // check if there is new content to display
bool DISP_checkMenuCursorUpdate(void); // check if the cursor was moved
uint8_t DISP_getMenuContentId(uint8_t n); // get the id`s from a contentelement for a special line
// menu function methods
bool FUNC_setup(void); // check if a menu function is called the first time to init some things
bool FUNC_loop(void); // check if a menu function is running in a loop
bool FUNC_close(void); // check if a menu function is closed to reach a stable state
void FUNC_goBackToMenu(uint8_t p_goBackCnt=0); // close the current menu function (the FUNC_close check is true when this is set)
uint8_t FUNC_getID(void); // get the ID of the current menu function
void FUNC_setLoopInterval(unsigned long p_t); // set a loop intervall for the current menu function the default loop intervall is 100000000 ms
void FUNC_disableScreensaver(void); // disable the screensaver for the current menu function
void FUNC_setGBAToLastCursorPos(void); // set a special "go back handling" - only to last cursor position
void FUNC_setGBAToLastFunc(void); // set a special "go back handling" - only to last function
void FUNC_setGBA(void); // set a special "go back handling" which decide between the two function above
void FUNC_setCEMask(unsigned long p_mask); // set a mask to enable only special events for a menu function
// timer methods
bool TIMER_ms(unsigned long &p_var, unsigned long p_t); // a small timer based on the millis() function
void TIMER_msReset(unsigned long &p_var); // reset the millis timer
bool TIMER_us(unsigned long &p_var, unsigned long p_t); // a small timer based on the micros() function
void TIMER_usReset(unsigned long &p_var); // reset the micros timer
// other methods
void OTHER_jumpToFunc(LCDML_FuncPtr_pu8 p_search, uint8_t p_para = 0); // jumpTo a defined function based on the function name
void OTHER_jumpToID(uint8_t p_search, uint8_t p_para = 0); // jumpTo a defined function based on the function id
void OTHER_setCursorToID(uint8_t p_search); // set the cursor to a defined function based on the id
void OTHER_setCursorToFunc(LCDML_FuncPtr_pu8 p_search); // set the cursor to a defined function based on the function name
uint8_t OTHER_getIDFromFunction(LCDML_FuncPtr_pu8 p_search); // get the menu element id from a menu element function name
void OTHER_updateFunc(void); // update a menu function directly when it runs
// screensaver methods
void SCREEN_enable(LCDML_FuncPtr_pu8 p_function, unsigned long p_t); // enable the screensaver feature
void SCREEN_disable(void); // disable the screensaver feature
void SCREEN_resetTimer(void); // reset the screensaver timer
void SCREEN_start(void); // start the screensaver directly
};
#endif