1
0
Fork 0

wasp: Add button presses to the event system.

Here the biggest changes are in the test application because we
refactor a number of the tests to make better use of the button.

Although applications may consume button events it does have a
default behavior which is to switch to the default application
(usually the clock).
This commit is contained in:
Daniel Thompson 2020-03-28 17:27:09 +00:00
parent 0cc7987689
commit b2622b579d
2 changed files with 109 additions and 31 deletions

View file

@ -9,7 +9,7 @@ class TestApp():
""" """
def __init__(self): def __init__(self):
self.tests = ('Touch', 'String') self.tests = ('Touch', 'String', 'Button')
self.test = self.tests[0] self.test = self.tests[0]
def foreground(self): def foreground(self):
@ -17,7 +17,8 @@ class TestApp():
self.on_screen = ( -1, -1, -1, -1, -1, -1 ) self.on_screen = ( -1, -1, -1, -1, -1, -1 )
self.draw() self.draw()
wasp.system.request_event(wasp.EventMask.TOUCH | wasp.system.request_event(wasp.EventMask.TOUCH |
wasp.EventMask.SWIPE_UPDOWN) wasp.EventMask.SWIPE_UPDOWN |
wasp.EventMask.BUTTON)
def background(self): def background(self):
"""De-activate the application (without losing state).""" """De-activate the application (without losing state)."""
@ -26,20 +27,39 @@ class TestApp():
def sleep(self): def sleep(self):
return False return False
def press(self, button, state):
draw = wasp.watch.drawable
if self.test == 'Touch':
draw.string('Button', 0, 108, width=240)
if self.test == 'String':
self.benchmark_string()
elif self.test == 'Button':
draw.string('{}: {}'.format(button, state), 0, 108, width=240)
def swipe(self, event): def swipe(self, event):
tests = self.tests tests = self.tests
i = tests.index(self.test) + 1 i = tests.index(self.test)
if event[0] == wasp.EventType.UP:
i += 1
if i >= len(tests): if i >= len(tests):
i = 0 i = 0
else:
i -= 1
if i < 0:
i = len(tests) - 1
self.test = tests[i] self.test = tests[i]
self.draw() self.draw()
def touch(self, event): def touch(self, event):
draw = wasp.watch.drawable
if self.test == 'Touch': if self.test == 'Touch':
draw.string('({}, {})'.format(event[1], event[2]), wasp.watch.drawable.string('({}, {})'.format(
0, 108, width=240) event[1], event[2]), 0, 108, width=240)
elif self.test == 'String': elif self.test == 'String':
self.benchmark_string()
def benchmark_string(self):
draw = wasp.watch.drawable
draw.fill(0, 0, 30, 240, 240-30) draw.fill(0, 0, 30, 240, 240-30)
t = machine.Timer(id=1, period=8000000) t = machine.Timer(id=1, period=8000000)
t.start() t.start()
@ -53,8 +73,6 @@ class TestApp():
del t del t
draw.string('{}s'.format(elapsed / 1000000), 12, 24+192) draw.string('{}s'.format(elapsed / 1000000), 12, 24+192)
return True
def draw(self): def draw(self):
"""Redraw the display from scratch.""" """Redraw the display from scratch."""
wasp.watch.display.mute(True) wasp.watch.display.mute(True)

View file

@ -39,6 +39,36 @@ class EventMask():
SWIPE_UPDOWN = 0x0004 SWIPE_UPDOWN = 0x0004
BUTTON = 0x0008 BUTTON = 0x0008
class Debounce(object):
"""Pin (and Signal) debounce wrapper.
TODO: Currently this driver doesn't actually implement any
debounce but it will!
"""
def __init__(self, pin):
"""
:param Pin pin: The pin to generate events from
"""
self._pin = pin
self._value = pin.value()
def get_event(self):
"""Receive a pin change event.
Check for a pending pin change event and, if an event is pending,
return it.
:return: boolean of the pin state if an event is received, None
otherwise.
"""
new_value = self._pin.value()
if self._value == new_value:
return None
self._value = new_value
return new_value
class Manager(): class Manager():
"""WASP system manager """WASP system manager
@ -53,14 +83,23 @@ class Manager():
def __init__(self): def __init__(self):
self.app = None self.app = None
self.applications = [ self.applications = []
ClockApp(),
FlashlightApp(),
TestApp()
]
self.charging = True
self.charging = True
self._brightness = 2 self._brightness = 2
self._button = Debounce(watch.button)
# TODO: Eventually these should move to main.py
self.register(ClockApp(), True)
self.register(FlashlightApp(), True)
self.register(TestApp(), True)
def register(self, app, quick_ring=True):
"""Register an application with the system.
:param object app: The application to regsister
"""
self.applications.append(app)
@property @property
def brightness(self): def brightness(self):
@ -115,6 +154,12 @@ class Manager():
if i < 0: if i < 0:
i = len(app_list)-1 i = len(app_list)-1
self.switch(app_list[i]) self.switch(app_list[i])
elif direction == EventType.HOME:
i = app_list.index(self.app)
if i != 0:
self.switch(app_list[0])
else:
self.sleep()
def request_event(self, event_mask): def request_event(self, event_mask):
"""Subscribe to events. """Subscribe to events.
@ -159,6 +204,20 @@ class Manager():
self.keep_awake() self.keep_awake()
def _handle_button(self, state):
"""Process a button-press (or unpress) event.
"""
self.keep_awake()
if bool(self.event_mask & EventMask.BUTTON):
# Currently we only support one button
if not self.app.press(EventType.HOME, state):
# If app reported None or False then we are done
return
if state:
self.navigate(EventType.HOME)
def _handle_touch(self, event): def _handle_touch(self, event):
"""Process a touch event. """Process a touch event.
""" """
@ -196,14 +255,15 @@ class Manager():
ticks += 1 ticks += 1
self.app.tick(ticks) self.app.tick(ticks)
if watch.button.value(): state = self._button.get_event()
self.keep_awake() if None != state:
self._handle_button(state)
event = watch.touch.get_event() event = watch.touch.get_event()
if event: if event:
self._handle_touch(event) self._handle_touch(event)
if watch.rtc.uptime > self.sleep_at: if self.sleep_at and watch.rtc.uptime > self.sleep_at:
self.sleep() self.sleep()
gc.collect() gc.collect()
@ -211,7 +271,7 @@ class Manager():
watch.rtc.update() watch.rtc.update()
charging = watch.battery.charging() charging = watch.battery.charging()
if watch.button.value() or self.charging != charging: if 1 == self._button.get_event() or self.charging != charging:
self.wake() self.wake()
def run(self): def run(self):