vibrating_alarm_m5stickc.py 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268
  1. """
  2. tested on uiflow for stickc v1.8.1
  3. """
  4. import json
  5. # pylint: disable=import-error
  6. import esp32
  7. import m5ui
  8. import machine
  9. import micropython
  10. import utils
  11. import utime
  12. from m5stack import axp, btnA, btnB, lcd, rtc
  13. _LARGE_FONT = lcd.FONT_DejaVu40
  14. _SMALL_FONT = lcd.FONT_DejaVu18
  15. _DEFAULT_FONT_COLOR = lcd.WHITE
  16. _LEFT_PADDING = 8
  17. _ALARM_TIME_PATH = "alarm.json"
  18. _SCREEN_WIDTH, _SCREEN_HEIGHT = lcd.winsize()
  19. _AWAKE_SECONDS = 8
  20. _VIBRATION_MOTOR_PIN = 26
  21. def _handle_pending_events():
  22. # > [...] a millisecond sleep larger than 10ms will check for pending (soft)
  23. # > interrupts during the sleep. [...] The reason for the 10ms value is
  24. # > because the FreeRTOS tick is 10ms, [...]
  25. # https://github.com/micropython/micropython/issues/3493#issuecomment-352617624
  26. # https://github.com/micropython/micropython/commit/4ed586528047d3eced28a9f4af11dbbe64fa99bb
  27. utime.sleep_ms(11)
  28. def _get_current_daytime() -> int:
  29. # > [contradictory to] official micropython documentation, to set RTC,
  30. # > use particular tuple (year , month, day, week=0, hour, minute, second, timestamp=0)
  31. # https://community.m5stack.com/topic/3108/m5stack-core2-micropython-rtc-example
  32. hour, minute, seconds = rtc.now()[3:]
  33. return (hour * 60 + minute) * 60 + seconds
  34. def _sleep(duration_seconds: int) -> None:
  35. # TODO turn off display
  36. axp.setLcdBrightness(30)
  37. machine.lightsleep(duration_seconds * 1000)
  38. # TODO turn on display
  39. axp.setLcdBrightness(100)
  40. print("wake reason:", machine.wake_reason())
  41. class _AlarmConfig:
  42. def __init__(self):
  43. self._hour = 0
  44. self._minute = 0
  45. @property
  46. def hour(self) -> int:
  47. return self._hour
  48. @property
  49. def minute(self) -> int:
  50. return self._minute
  51. def shift_hour(self, delta: int) -> None:
  52. self._hour = (self._hour + delta) % 24
  53. def shift_minute(self, delta: int) -> None:
  54. self._minute = (self._minute + delta) % 60
  55. @property
  56. def _daytime(self) -> int:
  57. return (self._hour * 60 + self._minute) * 60
  58. @property
  59. def _seconds_until_next(self) -> int:
  60. return (self._daytime - _get_current_daytime() - 1) % (24 * 60 * 60) + 1
  61. @property
  62. def next_time(self) -> int:
  63. return utime.time() + self._seconds_until_next
  64. def load(self, path: str) -> None:
  65. with open(path, "r") as alarm_time_file:
  66. alarm_time = json.load(alarm_time_file)
  67. self._hour = alarm_time["hour"]
  68. self._minute = alarm_time["minute"]
  69. def save(self, path: str) -> None:
  70. with open(path, "w") as alarm_time_file:
  71. json.dump(
  72. {"hour": self._hour, "minute": self._minute},
  73. alarm_time_file,
  74. )
  75. class App:
  76. # pylint: disable=too-few-public-methods,too-many-instance-attributes
  77. def __init__(self) -> None:
  78. self._clock_text_box = None
  79. self._menu_position = 0
  80. self._wait_for_sleep_start_time_update = False
  81. self._sleep_start_time = 0.0
  82. self._alarm_config = _AlarmConfig()
  83. self._next_alarm_time = self._alarm_config.next_time
  84. self._alarm_hour_text_box = None
  85. self._alarm_minute_text_box = None
  86. self._battery_status_text_box = None
  87. self._alarm_active = False
  88. self._vibration_motor_pin = None
  89. def _reschedule_sleep(self, interrupt: bool) -> None:
  90. if interrupt:
  91. self._wait_for_sleep_start_time_update = True
  92. micropython.schedule(self._reschedule_sleep, False)
  93. else:
  94. self._sleep_start_time = utime.time() + _AWAKE_SECONDS
  95. self._wait_for_sleep_start_time_update = False
  96. def _update_menu(self, event_arg: None) -> None:
  97. # pylint: disable=unused-argument; callback
  98. self._alarm_hour_text_box.setColor( # type: ignore
  99. lcd.GREEN
  100. if self._menu_position == 1
  101. else (lcd.RED if self._menu_position == 2 else _DEFAULT_FONT_COLOR)
  102. )
  103. self._alarm_minute_text_box.setColor( # type: ignore
  104. lcd.GREEN
  105. if self._menu_position == 3
  106. else (lcd.RED if self._menu_position == 4 else _DEFAULT_FONT_COLOR)
  107. )
  108. def _button_a_pressed(self) -> None:
  109. self._reschedule_sleep(interrupt=True)
  110. if self._alarm_active:
  111. print("alarm stopped")
  112. self._alarm_active = False
  113. else:
  114. self._menu_position = (self._menu_position + 1) % 5
  115. # https://docs.micropython.org/en/latest/library/micropython.html#micropython.schedule
  116. micropython.schedule(self._update_menu, None)
  117. def _alarm(self) -> None:
  118. print("ALARM")
  119. self._alarm_active = True
  120. while self._alarm_active:
  121. self._vibration_motor_pin.on() # type: ignore
  122. utime.sleep(0.5)
  123. self._vibration_motor_pin.off() # type: ignore
  124. utime.sleep(0.5)
  125. self._next_alarm_time = self._alarm_config.next_time
  126. def _update_alarm_time(self, event_arg: None) -> None:
  127. # pylint: disable=unused-argument; callback
  128. self._next_alarm_time = self._alarm_config.next_time
  129. self._alarm_config.save(_ALARM_TIME_PATH)
  130. self._alarm_hour_text_box.setText( # type: ignore
  131. "{:02d}".format(self._alarm_config.hour)
  132. )
  133. self._alarm_minute_text_box.setText( # type: ignore
  134. "{:02d}".format(self._alarm_config.minute)
  135. )
  136. def _button_b_pressed(self) -> None:
  137. self._reschedule_sleep(interrupt=True)
  138. if self._menu_position == 1:
  139. self._alarm_config.shift_hour(1)
  140. elif self._menu_position == 2:
  141. self._alarm_config.shift_hour(-1)
  142. elif self._menu_position == 3:
  143. self._alarm_config.shift_minute(1)
  144. elif self._menu_position == 4:
  145. self._alarm_config.shift_minute(-1)
  146. micropython.schedule(self._update_alarm_time, None)
  147. @staticmethod
  148. def _format_time() -> str:
  149. return "{:02d}:{:02d}".format(*rtc.now()[3:5])
  150. def _setup_clock(self) -> None:
  151. # https://github.com/m5stack/UIFlow-Code/wiki/M5UI#textbox
  152. self._clock_text_box = m5ui.M5TextBox(
  153. _SCREEN_WIDTH - 1,
  154. _LEFT_PADDING,
  155. self._format_time(),
  156. _LARGE_FONT,
  157. _DEFAULT_FONT_COLOR,
  158. rotate=90,
  159. )
  160. machine.Timer(0).init( # TODO update in main loop
  161. period=4000, # ms
  162. mode=machine.Timer.PERIODIC,
  163. callback=lambda t: self._clock_text_box.setText(self._format_time()), # type: ignore
  164. )
  165. def _setup_alarm(self) -> None:
  166. if utils.exists(_ALARM_TIME_PATH):
  167. self._alarm_config.load(_ALARM_TIME_PATH)
  168. self._alarm_hour_text_box = m5ui.M5TextBox(
  169. _SCREEN_WIDTH // 2,
  170. _LEFT_PADDING,
  171. "{:02d}".format(self._alarm_config.hour),
  172. _LARGE_FONT,
  173. _DEFAULT_FONT_COLOR,
  174. rotate=90,
  175. )
  176. m5ui.M5TextBox(
  177. _SCREEN_WIDTH // 2,
  178. _LEFT_PADDING + 53,
  179. ":",
  180. _LARGE_FONT,
  181. _DEFAULT_FONT_COLOR,
  182. rotate=90,
  183. )
  184. self._alarm_minute_text_box = m5ui.M5TextBox(
  185. _SCREEN_WIDTH // 2,
  186. _LEFT_PADDING + 66,
  187. "{:02d}".format(self._alarm_config.minute),
  188. _LARGE_FONT,
  189. _DEFAULT_FONT_COLOR,
  190. rotate=90,
  191. )
  192. def _update_battery_status_info(self) -> None:
  193. self._battery_status_text_box.setText( # type: ignore
  194. "{:.02f}V".format(axp.getBatVoltage())
  195. )
  196. def _setup_battery_status_info(self) -> None:
  197. self._battery_status_text_box = m5ui.M5TextBox(
  198. 12,
  199. _SCREEN_HEIGHT - 20,
  200. "",
  201. _SMALL_FONT,
  202. _DEFAULT_FONT_COLOR,
  203. rotate=0,
  204. )
  205. def run(self) -> None:
  206. m5ui.setScreenColor(0x000000) # clear screen
  207. self._setup_clock()
  208. self._setup_alarm()
  209. self._setup_battery_status_info()
  210. self._vibration_motor_pin = machine.Pin(_VIBRATION_MOTOR_PIN, machine.Pin.OUT)
  211. btnA.wasPressed(self._button_a_pressed)
  212. btnB.wasPressed(self._button_b_pressed)
  213. # not sure whether ext0 would be better
  214. esp32.wake_on_ext1([btnA.pin], esp32.WAKEUP_ALL_LOW)
  215. self._reschedule_sleep(interrupt=False)
  216. self._update_battery_status_info()
  217. while True:
  218. seconds_until_alarm = self._next_alarm_time - utime.time()
  219. print("seconds_until_alarm =", seconds_until_alarm) # TODO remove
  220. if seconds_until_alarm <= 0:
  221. self._alarm()
  222. elif seconds_until_alarm < _AWAKE_SECONDS:
  223. utime.sleep(seconds_until_alarm)
  224. elif utime.time() < self._sleep_start_time:
  225. utime.sleep(1) # second
  226. else:
  227. _sleep(duration_seconds=seconds_until_alarm)
  228. _handle_pending_events()
  229. self._update_battery_status_info()
  230. while self._wait_for_sleep_start_time_update:
  231. _handle_pending_events()
  232. App().run()