123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240 |
- """
- tested on uiflow for stickc v1.8.1
- """
- import json
- # pylint: disable=import-error
- import esp32
- import m5ui
- import machine
- import micropython
- import utils
- import utime
- from m5stack import axp, btnA, btnB, lcd, rtc
- _LARGE_FONT = lcd.FONT_DejaVu40
- _SMALL_FONT = lcd.FONT_DejaVu18
- _DEFAULT_FONT_COLOR = lcd.WHITE
- _LEFT_PADDING = 8
- _ALARM_TIME_PATH = "alarm.json"
- _SCREEN_WIDTH, _SCREEN_HEIGHT = lcd.winsize()
- _AWAKE_SECONDS = 8
- def _handle_pending_events():
- # > [...] a millisecond sleep larger than 10ms will check for pending (soft)
- # > interrupts during the sleep. [...] The reason for the 10ms value is
- # > because the FreeRTOS tick is 10ms, [...]
- # https://github.com/micropython/micropython/issues/3493#issuecomment-352617624
- # https://github.com/micropython/micropython/commit/4ed586528047d3eced28a9f4af11dbbe64fa99bb
- utime.sleep_ms(11)
- def _get_current_daytime() -> int:
- # > [contradictory to] official micropython documentation, to set RTC,
- # > use particular tuple (year , month, day, week=0, hour, minute, second, timestamp=0)
- # https://community.m5stack.com/topic/3108/m5stack-core2-micropython-rtc-example
- hour, minute, seconds = rtc.now()[3:]
- return (hour * 60 + minute) * 60 + seconds
- class App:
- # pylint: disable=too-few-public-methods,too-many-instance-attributes
- def __init__(self) -> None:
- self._clock_text_box = None
- self._menu_position = 0
- self._wait_for_sleep_start_time_update = False
- self._sleep_start_time_seconds = 0.0
- self._alarm_hour = None
- self._alarm_minute = None
- self._alarm_hour_text_box = None
- self._alarm_minute_text_box = None
- self._alarm_timer = None
- self._battery_status_text_box = None
- @property
- def _alarm_daytime(self) -> int:
- return (self._alarm_hour * 60 + self._alarm_minute) * 60 # type: ignore
- def _load_alarm_time(self) -> None:
- with open(_ALARM_TIME_PATH, "r") as alarm_time_file:
- alarm_time = json.load(alarm_time_file)
- self._alarm_hour = alarm_time["hour"]
- self._alarm_minute = alarm_time["minute"]
- def _save_alarm_time(self) -> None:
- with open(_ALARM_TIME_PATH, "w") as alarm_time_file:
- json.dump(
- {"hour": self._alarm_hour, "minute": self._alarm_minute},
- alarm_time_file,
- )
- def _reschedule_sleep(self, interrupt: bool) -> None:
- if interrupt:
- self._wait_for_sleep_start_time_update = True
- micropython.schedule(self._reschedule_sleep, False)
- else:
- self._sleep_start_time_seconds = utime.time() + _AWAKE_SECONDS
- self._wait_for_sleep_start_time_update = False
- def _update_menu(self, event_arg: None) -> None:
- # pylint: disable=unused-argument; callback
- self._alarm_hour_text_box.setColor( # type: ignore
- lcd.GREEN
- if self._menu_position == 1
- else (lcd.RED if self._menu_position == 2 else _DEFAULT_FONT_COLOR)
- )
- self._alarm_minute_text_box.setColor( # type: ignore
- lcd.GREEN
- if self._menu_position == 3
- else (lcd.RED if self._menu_position == 4 else _DEFAULT_FONT_COLOR)
- )
- def _button_a_pressed(self) -> None:
- self._reschedule_sleep(interrupt=True)
- self._menu_position = (self._menu_position + 1) % 5
- # https://docs.micropython.org/en/latest/library/micropython.html#micropython.schedule
- micropython.schedule(self._update_menu, None)
- @staticmethod
- def _alert() -> None:
- print("ALARM")
- def _alarm(self, timer: machine.Timer) -> None:
- # pylint: disable=unused-argument; callback
- micropython.schedule(lambda n: self._alert(), None)
- micropython.schedule(lambda n: self._configure_alarm_timer(), None)
- def _configure_alarm_timer(self) -> None:
- seconds_until_alarm = (self._alarm_daytime - _get_current_daytime() - 1) % (
- 24 * 60 * 60
- ) + 1
- print("alarm in ", seconds_until_alarm / 60, " min")
- # TODO configure wake from sleep
- self._alarm_timer.init( # type: ignore
- period=seconds_until_alarm * 1000, # ms
- mode=machine.Timer.ONE_SHOT,
- callback=self._alarm,
- )
- def _update_alarm_time(self, event_arg: None) -> None:
- # pylint: disable=unused-argument; callback
- self._alarm_hour_text_box.setText( # type: ignore
- "{:02d}".format(self._alarm_hour) # type: ignore
- )
- self._alarm_minute_text_box.setText( # type: ignore
- "{:02d}".format(self._alarm_minute) # type: ignore
- )
- self._save_alarm_time()
- self._configure_alarm_timer()
- def _button_b_pressed(self) -> None:
- self._reschedule_sleep(interrupt=True)
- if self._menu_position == 1:
- self._alarm_hour += 1 # type: ignore
- elif self._menu_position == 2:
- self._alarm_hour -= 1 # type: ignore
- elif self._menu_position == 3:
- self._alarm_minute += 1 # type: ignore
- elif self._menu_position == 4:
- self._alarm_minute -= 1 # type: ignore
- self._alarm_hour %= 24 # type: ignore
- self._alarm_minute %= 60 # type: ignore
- micropython.schedule(self._update_alarm_time, None)
- @staticmethod
- def _format_time() -> str:
- return "{:02d}:{:02d}".format(*rtc.now()[3:5])
- def _setup_clock(self) -> None:
- # https://github.com/m5stack/UIFlow-Code/wiki/M5UI#textbox
- self._clock_text_box = m5ui.M5TextBox(
- _SCREEN_WIDTH - 1,
- _LEFT_PADDING,
- self._format_time(),
- _LARGE_FONT,
- _DEFAULT_FONT_COLOR,
- rotate=90,
- )
- machine.Timer(0).init(
- period=4000, # ms
- mode=machine.Timer.PERIODIC,
- callback=lambda t: self._clock_text_box.setText(self._format_time()), # type: ignore
- )
- def _setup_alarm(self) -> None:
- if not utils.exists(_ALARM_TIME_PATH):
- self._alarm_hour = self._alarm_minute = 0 # type: ignore
- self._save_alarm_time()
- else:
- self._load_alarm_time()
- self._alarm_hour_text_box = m5ui.M5TextBox(
- _SCREEN_WIDTH // 2,
- _LEFT_PADDING,
- "{:02d}".format(self._alarm_hour), # type: ignore
- _LARGE_FONT,
- _DEFAULT_FONT_COLOR,
- rotate=90,
- )
- m5ui.M5TextBox(
- _SCREEN_WIDTH // 2,
- _LEFT_PADDING + 53,
- ":",
- _LARGE_FONT,
- _DEFAULT_FONT_COLOR,
- rotate=90,
- )
- self._alarm_minute_text_box = m5ui.M5TextBox(
- _SCREEN_WIDTH // 2,
- _LEFT_PADDING + 66,
- "{:02d}".format(self._alarm_minute), # type: ignore
- _LARGE_FONT,
- _DEFAULT_FONT_COLOR,
- rotate=90,
- )
- # machine.Timer(1).init(...) breaks button handling
- self._alarm_timer = machine.Timer(2)
- self._configure_alarm_timer()
- def _update_battery_status_info(self) -> None:
- self._battery_status_text_box.setText( # type: ignore
- "{:.02f}V".format(axp.getBatVoltage())
- )
- def _setup_battery_status_info(self) -> None:
- self._battery_status_text_box = m5ui.M5TextBox(
- 12,
- _SCREEN_HEIGHT - 20,
- "",
- _SMALL_FONT,
- _DEFAULT_FONT_COLOR,
- rotate=0,
- )
- def run(self) -> None:
- m5ui.setScreenColor(0x000000) # clear screen
- self._setup_clock()
- self._setup_alarm()
- self._setup_battery_status_info()
- btnA.wasPressed(self._button_a_pressed)
- btnB.wasPressed(self._button_b_pressed)
- # not sure whether ext0 would be better
- esp32.wake_on_ext1([btnA.pin], esp32.WAKEUP_ALL_LOW)
- self._reschedule_sleep(interrupt=False)
- while True:
- self._update_battery_status_info()
- while utime.time() < self._sleep_start_time_seconds:
- utime.sleep(1) # seconds
- # TODO turn off display
- axp.setLcdBrightness(30)
- machine.lightsleep()
- # TODO turn on display
- axp.setLcdBrightness(100)
- print("wake reason:", machine.wake_reason())
- _handle_pending_events()
- while self._wait_for_sleep_start_time_update:
- _handle_pending_events()
- App().run()
|