5 Commits

4 changed files with 181 additions and 42 deletions

View File

@@ -104,6 +104,17 @@ So, as the `Config` class uses python's magic function for getting and setting a
- [lcdMenu](https://git.privacynerd.de/BlueFox/lcdMenu) - used for the menus - [lcdMenu](https://git.privacynerd.de/BlueFox/lcdMenu) - used for the menus
## Learning curve
Here are some of the websites I learnt a lot from while programming this project - mainly here for documentation reasons.
- [A handy guide into configuration files in json](https://bhave.sh/micropython-json-config/)
- [The official micropython wiki page about the same topic](https://docs.micropython.org/en/latest/library/json.html)
- [StackOverflow thread about how to format a json file (used by this project to make it a bit more readable)](https://stackoverflow.com/questions/16311562/python-json-without-whitespaces)
- [A guide about magic functions for settings and getting attributes](https://staskoltsov.medium.com/python-magic-methods-to-get-set-attributes-716a12d0b106)
- [The official guide into git submodules](https://git-scm.com/book/en/v2/Git-Tools-Submodules)
## License ## License
This project is licensed under GPL-3.0-or-later. See [LICENSE](LICENSE). This project is licensed under GPL-3.0-or-later. See [LICENSE](LICENSE).

202
main.py
View File

@@ -32,22 +32,54 @@ def manual():
if config.PIN_IN_BTN_1.value() == 1 or config.PIN_IN_BTN_2.value() == 1: if config.PIN_IN_BTN_1.value() == 1 or config.PIN_IN_BTN_2.value() == 1:
return True # exit on press of these buttons; True to disable the Quitting message from lcdMenu return True # exit on press of these buttons; True to disable the Quitting message from lcdMenu
def timer(): def timer():
# display WIP timer_menu = lcdMenu(config.LCD, btn_mapping, scroll_direction=True, cycle=True, hide_menu_name=False, name="TIMERS")
config.LCD.clear() timer_values_original = [config.TIMER_1_DURATION, config.TIMER_2_DURATION, config.TIMER_3_DURATION]
config.LCD.putstr(" Still work-in-progress") timer_values = timer_values_original.copy() # here, the current timers time is stored when interrupting via the interrupt_pin (see below)
sleep(3) timer_splits = [lambda: divmod(round(timer_values[0]), 60), lambda: divmod(round(timer_values[1]), 60), lambda: divmod(round(timer_values[2]), 60)]
return True # disable the "Quitting" message from lcdMenu interrupt_pin = config.PIN_IN_BTN_1 # the interrupt btn stops the timer, saves the current time and goes back to the menu
def uv_on(): reset_pin = config.PIN_IN_BTN_2 # the reset btn restores the default value
config.PIN_OUT_RELAIS.value(1) start_stop_pin = config.PIN_IN_SWITCH # the start_stop switch starts/stops the timer
config.LCD.clear()
config.LCD.putstr("------ UV ------ turned on ") # timer_number is the number of the timer that will be run, starting at 1
sleep(1) # the _timer variable is needed because otherwise python will throw crazy errors regarding "variable accessed before assignment"...
return True # disable the "Quitting" message from lcdMenu # ...just see it as a copy of the timer_number-1'th elemnt of the timer_values list (see above)
def uv_off(): def run_timer(timer_number: int, _timer: int):
config.PIN_OUT_RELAIS.value(0) config.LCD.clear()
config.LCD.clear() config.LCD.putstr(f"Timer {timer_number}".center(16))
config.LCD.putstr("------ UV ------ turned off ") last_start_stop_value = start_stop_pin.value()
sleep(1) config.PIN_OUT_RELAIS.value(last_start_stop_value)
last_time_ns = time_ns()
while True: # now run the timer (if the switch is high)
config.LCD.move_to(0,1)
_timer_div = divmod(round(_timer), 60)
config.LCD.putstr(f"{_timer_div[0]:02d}:{_timer_div[1]:02d}".center(16))
if interrupt_pin.value() == 1:
timer_values[timer_number-1] = _timer # save the current state
last_start_stop_value = 0 # turn the LEDs off!
config.PIN_OUT_RELAIS.value(last_start_stop_value)
return None
if reset_pin.value() == 1:
_timer = timer_values_original[timer_number-1] # reset the timers
if _timer <= 0:
config.PIN_OUT_RELAIS.off()
return True
sleep(0.05)
if last_start_stop_value != (new_value := start_stop_pin.value()):
last_start_stop_value = new_value
config.PIN_OUT_RELAIS.value(new_value)
last_time_ns = time_ns()
if start_stop_pin.value() == 1:
_timer -= (time_ns() - last_time_ns) / 1000**3
last_time_ns = time_ns()
timer_programs = [(f"T1 - {timer_splits[0]()[0]:02d}:{timer_splits[0]()[1]:02d}", lambda: run_timer(1, timer_values[0])),
(f"T2 - {timer_splits[1]()[0]:02d}:{timer_splits[1]()[1]:02d}", lambda: run_timer(2, timer_values[1])),
(f"T3 - {timer_splits[2]()[0]:02d}:{timer_splits[2]()[1]:02d}", lambda: run_timer(3, timer_values[2])),
("Exit", timer_menu.stop)]
timer_menu.setup(timer_programs) # give it the callback list
timer_menu.run()
del timer_menu, timer_programs
collect()
return True # disable the "Quitting" message from lcdMenu return True # disable the "Quitting" message from lcdMenu
def lcd_big_hello(): def lcd_big_hello():
import lcd_big_hello as lbh import lcd_big_hello as lbh
@@ -77,35 +109,131 @@ def settings():
option_down = [" ", "v"][current_cycles>1] option_down = [" ", "v"][current_cycles>1]
config.LCD.putstr(f" Cycles \n{option_down} {str(current_cycles).center(12)} ^") config.LCD.putstr(f" Cycles \n{option_down} {str(current_cycles).center(12)} ^")
btn_left = config.PIN_IN_BTN_1
btn_right = config.PIN_IN_BTN_2
while True: while True:
if config.PIN_IN_BTN_1.value() == 1: if btn_left.value() == 1:
time_ns_when_pressed = time_ns() sleep(0.1) # this value is a good compromise between being able to press both buttons and a fast up/down speed
while config.PIN_IN_BTN_1.value() == 1: if btn_right.value() == 1:
if (time_ns() - time_ns_when_pressed) > 1000000000: # if the time passed is longer than a second config.STARTUP_WELCOME_CYCLES = current_cycles
while config.PIN_IN_BTN_1.value() == 1: config.LCD.move_to(0,1) # move to the second row
pass # wait till release config.LCD.putstr(f"Now set to {current_cycles}".center(16)) # show a little info that it is now set
config.STARTUP_WELCOME_CYCLES = current_cycles while btn_right.value() == 1 or btn_left.value() == 1: # wait till both btns are released
return True pass
sleep(0.05) return True
current_cycles -= 1 current_cycles -= 1
if current_cycles < 1: current_cycles = 1 if current_cycles < 1: current_cycles = 1
option_down = [" ", "v"][current_cycles>1] option_down = [" ", "v"][current_cycles>1]
config.LCD.putstr(f" Cycles \n{option_down} {str(current_cycles).center(12)} ^") config.LCD.putstr(f" Cycles \n{option_down} {str(current_cycles).center(12)} ^")
if config.PIN_IN_BTN_2.value() == 1: if btn_right.value() == 1:
time_ns_when_pressed = time_ns() sleep(0.1)
while config.PIN_IN_BTN_2.value() == 1: if btn_left.value() == 1:
if (time_ns() - time_ns_when_pressed) > 1000000000: # if the time passed is longer than a second config.STARTUP_WELCOME_CYCLES = current_cycles
while config.PIN_IN_BTN_2.value() == 1: config.LCD.move_to(0,1) # move to the second row
pass # wait till release config.LCD.putstr(f"Now set to {current_cycles}".center(16)) # show a little info that it is now set
config.STARTUP_WELCOME_CYCLES = current_cycles while btn_right.value() == 1 or btn_left.value() == 1: # wait till both btns are released
return True pass
sleep(0.05) return True
current_cycles += 1 current_cycles += 1
option_down = [" ", "v"][current_cycles>1] option_down = [" ", "v"][current_cycles>1]
config.LCD.putstr(f" Cycles \n{option_down} {str(current_cycles).center(12)} ^") config.LCD.putstr(f" Cycles \n{option_down} {str(current_cycles).center(12)} ^")
# with n being the number of the timer (starting at 0)
def set_n_timer(n: int):
# a small helper function to save the value of the n'th timer...
# as you can't programatically access TIMER_n_DURATION
def set_timer_helper(n, value):
if n == 0:
config.TIMER_1_DURATION = value
elif n == 1:
config.TIMER_2_DURATION = value
elif n == 2:
config.TIMER_3_DURATION = value
else:
utils.log(0, "There are only 3 timers at all. Trying to set the timer number {n} failed.")
config.LCD.clear()
timer_values = [config.TIMER_1_DURATION, config.TIMER_2_DURATION, config.TIMER_3_DURATION]
current_timer = timer_values[n] # get the n'th timer
current_timer_div = lambda: divmod(current_timer, 60)
config.LCD.putstr(f"Timer {n+1}".center(16))
config.LCD.putstr(f"{'v' if current_timer > 1 else ' '} " + f"{current_timer_div()[0]:02d}:{current_timer_div()[1]:02d}".center(12) + " ^")
btn_left = config.PIN_IN_BTN_1
btn_right = config.PIN_IN_BTN_2
left_was_released = True
right_was_released = True
while True:
if btn_left.value() == 1:
if left_was_released:
time_press_start = time_ns()
left_was_released = False
sleep(0.1) # this value is a good compromise between being able to press both buttons and a fast up/down speed
if btn_right.value() == 1: # exit if both btns are pressed
set_timer_helper(n, current_timer)
config.LCD.move_to(0,1) # move to the second row
config.LCD.putstr(f"Saved!".center(16)) # show a little info that it is now set
while btn_right.value() == 1 or btn_left.value() == 1: # wait till both btns are released
pass
return True
# define the step width
time_now = time_ns()
if (time_now - time_press_start) <= 1*(10**9): # max. 1 seconds pressed
current_timer -= 1
elif (time_now - time_press_start) <= 2*(10**9): # max. 2 seconds pressed
current_timer -= 5
elif (time_now - time_press_start) <= 3*(10**9): # max. 3 seconds pressed
current_timer -= 10
elif (time_now - time_press_start) <= 4*(10**9): # max. 4 seconds pressed
current_timer -= 30
elif (time_now - time_press_start) <= 5*(10**9): # max. 5 seconds pressed
current_timer -= 60
else: # longer than 5s pressed
current_timer -= 300
if current_timer < 1: current_timer = 5999
config.LCD.move_to(0,1)
config.LCD.putstr("v " + f"{current_timer_div()[0]:02d}:{current_timer_div()[1]:02d}".center(12) + " ^")
else:
left_was_released = True
if btn_right.value() == 1:
if right_was_released:
time_press_start = time_ns()
right_was_released = False
sleep(0.1)
if btn_left.value() == 1: # exit if both btns are pressed
set_timer_helper(n, current_timer)
config.LCD.move_to(0,1) # move to the second row
config.LCD.putstr(f"Saved!".center(16)) # show a little info that it is now set
while btn_right.value() == 1 or btn_left.value() == 1: # wait till both btns are released
pass
return True
# define the step width
time_now = time_ns()
if (time_now - time_press_start) <= 1*(10**9): # max. 1 seconds pressed
current_timer += 1
elif (time_now - time_press_start) <= 2*(10**9): # max. 2 seconds pressed
current_timer += 5
elif (time_now - time_press_start) <= 3*(10**9): # max. 3 seconds pressed
current_timer += 10
elif (time_now - time_press_start) <= 4*(10**9): # max. 4 seconds pressed
current_timer += 30
elif (time_now - time_press_start) <= 5*(10**9): # max. 5 seconds pressed
current_timer += 60
else: # longer than 5s pressed
current_timer += 300
if current_timer > 5999: current_timer = 1
config.LCD.move_to(0,1)
config.LCD.putstr("v " + f"{current_timer_div()[0]:02d}:{current_timer_div()[1]:02d}".center(12) + " ^")
else:
right_was_released = True
settings_programs = [("Show welcome", swap_welcome), settings_programs = [("Show welcome", swap_welcome),
("Welcome cycles", welcome_cycles), ("Welcome cycles", welcome_cycles),
("Timer 1", lambda: set_n_timer(0)),
("Timer 2", lambda: set_n_timer(1)),
("Timer 3", lambda: set_n_timer(2)),
("Exit", settings_menu.stop)] ("Exit", settings_menu.stop)]
settings_menu.setup(settings_programs) # give it the callback list settings_menu.setup(settings_programs) # give it the callback list
settings_menu.run() # run the menu until it's closed settings_menu.run() # run the menu until it's closed
@@ -118,10 +246,10 @@ def run_demo_menu():
("Input tests", input_tests), ("Input tests", input_tests),
("Exit", demo_menu.stop)] ("Exit", demo_menu.stop)]
demo_menu.setup(demo_programs) # give it the callback list demo_menu.setup(demo_programs) # give it the callback list
ret = demo_menu.run() demo_menu.run()
del demo_menu, demo_programs del demo_menu, demo_programs
collect() collect()
return ret return True
if config.STARTUP_WELCOME_SHOW: if config.STARTUP_WELCOME_SHOW:
ws = WelcomeScreen(config.LCD, ws = WelcomeScreen(config.LCD,
@@ -138,8 +266,6 @@ if config.STARTUP_WELCOME_SHOW:
main_menu = lcdMenu(config.LCD, btn_mapping, scroll_direction=True, cycle=True, hide_menu_name=False, name="PROGRAMS") main_menu = lcdMenu(config.LCD, btn_mapping, scroll_direction=True, cycle=True, hide_menu_name=False, name="PROGRAMS")
main_programs = [("Timer", timer), main_programs = [("Timer", timer),
("Manual", manual), ("Manual", manual),
("UV off", uv_off),
("UV on", uv_on),
("Demos", run_demo_menu), ("Demos", run_demo_menu),
("Settings", settings)] ("Settings", settings)]
main_menu.setup(main_programs) # give it the callback list main_menu.setup(main_programs) # give it the callback list

View File

@@ -39,7 +39,10 @@ class Config:
"LCD_I2C_ADDR", # the i2c adress of the display "LCD_I2C_ADDR", # the i2c adress of the display
"LCD_I2C_NUM_ROWS", # how many rows for character display has the display? "LCD_I2C_NUM_ROWS", # how many rows for character display has the display?
"LCD_I2C_NUM_COLS", # and how many characters can it display per row? "LCD_I2C_NUM_COLS", # and how many characters can it display per row?
"LCD"] # the actual lcd object (of the PCF8574T I2C_LCD class, see libraries) "LCD", # the actual lcd object (of the PCF8574T I2C_LCD class, see libraries)
"TIMER_1_DURATION", # the duration of the first timer in seconds
"TIMER_2_DURATION", # the duration of the second first timer in seconds
"TIMER_3_DURATION"] # the duration of the third timer in seconds
self._config_file = config_file self._config_file = config_file
self.load_config() self.load_config()
@@ -129,5 +132,4 @@ def log(log_level: int, message: str):
print(f"[LOGGER] Got a message of unknown log level ({log_level}). Original message is printed below.") print(f"[LOGGER] Got a message of unknown log level ({log_level}). Original message is printed below.")
print(f"{message}") print(f"{message}")
elif cfg.LOG_LEVEL >= log_level: # if log level is valid elif cfg.LOG_LEVEL >= log_level: # if log level is valid
print(f"[{log_mapping[log_level]}] {message}") print(f"[{log_mapping[log_level]}] {message}")