2020-03-22 15:40:18 +00:00
|
|
|
# SPDX-License-Identifier: LGPL-3.0-or-later
|
|
|
|
# Copyright (C) 2020 Daniel Thompson
|
|
|
|
|
2020-01-31 19:36:55 +00:00
|
|
|
import time
|
|
|
|
def sleep_ms(ms):
|
|
|
|
time.sleep(ms / 1000)
|
|
|
|
time.sleep_ms = sleep_ms
|
|
|
|
|
2020-04-11 20:14:30 +01:00
|
|
|
import sys, traceback
|
|
|
|
def print_exception(exc, file=sys.stdout):
|
|
|
|
exc_type, exc_value, exc_traceback = sys.exc_info()
|
|
|
|
traceback.print_exception(exc_type, exc_value, exc_traceback, file=file)
|
|
|
|
sys.print_exception = print_exception
|
|
|
|
|
2020-06-09 17:18:27 +07:00
|
|
|
import array
|
2020-10-27 17:10:56 +00:00
|
|
|
import draw565
|
|
|
|
import os
|
2020-03-08 23:16:30 +00:00
|
|
|
|
2020-03-07 11:50:26 +00:00
|
|
|
from machine import I2C
|
2020-01-31 19:36:55 +00:00
|
|
|
from machine import Pin
|
|
|
|
from machine import SPI
|
|
|
|
|
2020-03-07 11:50:26 +00:00
|
|
|
from drivers.cst816s import CST816S
|
2020-01-31 19:36:55 +00:00
|
|
|
from drivers.st7789 import ST7789_SPI
|
|
|
|
from drivers.vibrator import Vibrator
|
|
|
|
|
2020-06-09 21:29:00 +01:00
|
|
|
class Accelerometer:
|
|
|
|
"""Simulated accelerometer.
|
|
|
|
|
|
|
|
Accelerometers such as BMA421 are complex and most of the driver
|
|
|
|
is written in C. For that reason we simulate the accelerometer
|
|
|
|
rather than emulate (by comparison we emulate the ST7789).
|
|
|
|
"""
|
|
|
|
def reset(self):
|
|
|
|
self._steps = 3
|
|
|
|
|
|
|
|
@property
|
|
|
|
def steps(self):
|
|
|
|
"""Report the number of steps counted."""
|
|
|
|
if self._steps < 10000:
|
|
|
|
self._steps = int(self._steps * 1.34)
|
|
|
|
else:
|
|
|
|
self._steps += 1
|
|
|
|
return self._steps
|
|
|
|
|
|
|
|
@steps.setter
|
|
|
|
def steps(self, value):
|
|
|
|
self.reset()
|
2020-02-23 20:19:37 +00:00
|
|
|
|
2020-02-01 20:20:53 +00:00
|
|
|
class Backlight(object):
|
|
|
|
def __init__(self, level=1):
|
2020-03-08 23:16:30 +00:00
|
|
|
pass
|
2020-02-01 20:20:53 +00:00
|
|
|
|
|
|
|
def set(self, level):
|
2020-03-08 23:16:30 +00:00
|
|
|
"""Set the simulated backlight level.
|
|
|
|
|
|
|
|
This function contains a subtle trick. As soon as the backlight is
|
|
|
|
turned off (e.g. the watch goes to sleep) then we will simulate
|
|
|
|
a button press in order to turn the watch back on again.
|
|
|
|
"""
|
2020-02-01 20:20:53 +00:00
|
|
|
print(f'BACKLIGHT: {level}')
|
|
|
|
|
2020-02-23 20:19:37 +00:00
|
|
|
button.value(bool(level))
|
|
|
|
|
2020-02-03 22:29:57 +00:00
|
|
|
class Battery(object):
|
|
|
|
def __init__(self):
|
2020-12-28 12:01:51 +00:00
|
|
|
self.voltage = 4.1
|
2020-02-03 22:29:57 +00:00
|
|
|
self.step = -0.01
|
|
|
|
self.powered = False
|
|
|
|
|
|
|
|
def charging(self):
|
|
|
|
self.voltage_mv()
|
|
|
|
return self.powered
|
|
|
|
|
|
|
|
def power(self):
|
|
|
|
self.voltage_mv()
|
|
|
|
return self.powered
|
|
|
|
|
|
|
|
def voltage_mv(self):
|
|
|
|
if self.voltage > 4:
|
2020-02-23 20:19:37 +00:00
|
|
|
self.step = -0.01
|
2020-02-03 22:29:57 +00:00
|
|
|
self.powered = False
|
|
|
|
elif self.voltage < 3.4:
|
2020-02-23 20:19:37 +00:00
|
|
|
self.step = 0.04
|
2020-02-03 22:29:57 +00:00
|
|
|
self.powered = True
|
|
|
|
self.voltage += self.step
|
|
|
|
|
|
|
|
return int(self.voltage * 1000)
|
|
|
|
|
|
|
|
def level(self):
|
|
|
|
mv = self.voltage_mv()
|
|
|
|
level = ((19 * mv) // 100) - 660
|
|
|
|
if level > 100:
|
|
|
|
return 100
|
|
|
|
if level < 0:
|
|
|
|
return 0
|
|
|
|
return level
|
|
|
|
|
2020-02-03 19:23:10 +00:00
|
|
|
class RTC(object):
|
|
|
|
def __init__(self):
|
2020-06-12 08:49:54 +01:00
|
|
|
self._epoch = time.time()
|
|
|
|
self._lasttime = 0
|
2020-02-03 19:23:10 +00:00
|
|
|
|
|
|
|
def update(self):
|
|
|
|
now = time.time()
|
2020-06-12 08:49:54 +01:00
|
|
|
if now == self._lasttime:
|
2020-02-03 19:23:10 +00:00
|
|
|
return False
|
2020-06-12 08:49:54 +01:00
|
|
|
self._lasttime = now
|
2020-02-03 19:23:10 +00:00
|
|
|
return True
|
|
|
|
|
2020-02-19 19:32:06 +00:00
|
|
|
def get_localtime(self):
|
2020-06-12 08:49:54 +01:00
|
|
|
#if self.uptime < 60:
|
|
|
|
# # Jump back a little over a day
|
|
|
|
# return time.localtime(time.time() - 100000)
|
2020-02-19 19:32:06 +00:00
|
|
|
return time.localtime()
|
|
|
|
|
2020-02-03 19:23:10 +00:00
|
|
|
def get_time(self):
|
2020-06-12 08:49:54 +01:00
|
|
|
now = self.get_localtime()
|
2020-02-03 19:23:10 +00:00
|
|
|
return (now[3], now[4], now[5])
|
|
|
|
|
2020-11-14 12:24:28 +00:00
|
|
|
def time(self):
|
|
|
|
return time.time()
|
|
|
|
|
2020-06-12 08:49:54 +01:00
|
|
|
@property
|
2020-02-03 19:23:10 +00:00
|
|
|
def uptime(self):
|
2020-06-12 08:49:54 +01:00
|
|
|
return time.time() - self._epoch
|
2020-03-07 11:50:26 +00:00
|
|
|
|
|
|
|
def get_uptime_ms(self):
|
2020-06-12 08:49:54 +01:00
|
|
|
return int(self.uptime * 1000)
|
2020-02-03 19:23:10 +00:00
|
|
|
|
2020-06-22 22:51:06 +01:00
|
|
|
class HRS():
|
|
|
|
DATA = (
|
|
|
|
9084,9084,9025,9025,9009,9009,9009,9015,9015,9024,9024,9024,9073,9073,9074,9074,
|
|
|
|
9074,9100,9100,9097,9097,9097,9045,9045,9023,9023,9023,9035,9035,9039,9039,9039,
|
|
|
|
9049,9049,9052,9052,9052,9066,9066,9070,9070,9070,9078,9078,9081,9081,9081,9092,
|
|
|
|
9092,9093,9093,9093,9094,9094,9108,9108,9108,9124,9124,9122,9122,9122,9100,9100,
|
|
|
|
9110,9110,9110,9112,9112,9118,9118,9118,9127,9127,9136,9136,9136,9147,9147,9154,
|
|
|
|
9154,9154,9156,9156,9153,9153,9153,9152,9152,9156,9156,9156,9161,9161,9161,9177,
|
|
|
|
9177,9186,9186,9196,9196,9196,9201,9201,9201,9189,9189,9176,9176,9176,9176,9176,
|
|
|
|
9175,9175,9175,9175,9175,9180,9180,9180,9189,9189,9202,9202,9202,9207,9207,9181,
|
|
|
|
9181,9181,9167,9167,9169,9169,9169,9163,9163,9164,9164,9164,9165,9165,9172,9172,
|
|
|
|
9172,9180,9180,9192,9192,9192,9178,9178,9161,9161,9161,9163,9163,9173,9173,9173,
|
|
|
|
9170,9170,9179,9179,9183,9183,9183,9196,9196,9207,9207,9207,9208,9208,9186,9186,
|
|
|
|
9186,9182,9182,9193,9193,9193,9197,9197,9188,9204,9204,9212,9212,9212,9223,9223,
|
|
|
|
9228,9228,9228,9235,9235,9215,9215,9215,9217,9217,9225,9225,9225,9230,9230,9237,
|
|
|
|
9237,9237,9246,9246,9260,9260,9260,9270,9270,9269,9269,9269,9256,9256,9256,9256,
|
2020-06-24 21:16:46 +01:00
|
|
|
9256,9263,9263,9274,9274,9274,9288,9288,9292,9292,9292,9307,9307,9310
|
2020-06-22 22:51:06 +01:00
|
|
|
)
|
|
|
|
def __init__(self):
|
|
|
|
self._i = 0
|
2020-06-24 21:16:46 +01:00
|
|
|
self._step = 1
|
2020-06-22 22:51:06 +01:00
|
|
|
|
|
|
|
def enable(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def disable(self):
|
|
|
|
pass
|
|
|
|
|
|
|
|
def read_hrs(self):
|
|
|
|
d = self.DATA[self._i]
|
2020-06-24 21:16:46 +01:00
|
|
|
|
|
|
|
self._i += self._step
|
2020-06-22 22:51:06 +01:00
|
|
|
if self._i >= len(self.DATA):
|
2020-06-24 21:16:46 +01:00
|
|
|
self._i -= 1
|
|
|
|
self._step = -1
|
|
|
|
elif self._i < 0:
|
|
|
|
self._i += 1
|
|
|
|
self._step = 1
|
|
|
|
|
2020-06-22 22:51:06 +01:00
|
|
|
return d
|
|
|
|
|
2020-02-01 20:20:53 +00:00
|
|
|
backlight = Backlight()
|
2020-03-08 23:16:30 +00:00
|
|
|
spi = SPI(0)
|
|
|
|
spi.init(polarity=1, phase=1, baudrate=8000000)
|
|
|
|
display = ST7789_SPI(240, 240, spi,
|
|
|
|
cs=Pin("DISP_CS", Pin.OUT, quiet=True),
|
|
|
|
dc=Pin("DISP_DC", Pin.OUT, quiet=True),
|
|
|
|
res=Pin("DISP_RST", Pin.OUT, quiet=True))
|
|
|
|
drawable = draw565.Draw565(display)
|
|
|
|
|
2020-06-09 21:29:00 +01:00
|
|
|
accel = Accelerometer()
|
2020-02-03 22:29:57 +00:00
|
|
|
battery = Battery()
|
2020-03-08 23:16:30 +00:00
|
|
|
button = Pin('BUTTON', Pin.IN, quiet=True)
|
2020-06-22 22:51:06 +01:00
|
|
|
hrs = HRS()
|
2020-02-03 19:23:10 +00:00
|
|
|
rtc = RTC()
|
2020-06-11 21:13:58 +01:00
|
|
|
touch = CST816S(I2C(0), Pin('TP_INT', Pin.IN, quiet=True), Pin('TP_RST', Pin.OUT, quiet=True))
|
2020-01-31 19:36:55 +00:00
|
|
|
vibrator = Vibrator(Pin('MOTOR', Pin.OUT, value=0), active_low=True)
|
2020-02-03 19:23:10 +00:00
|
|
|
|
2020-07-24 17:04:46 +01:00
|
|
|
def connected():
|
|
|
|
return not (int(rtc.uptime / 30) & 1)
|
2020-12-12 21:35:45 +00:00
|
|
|
|
|
|
|
# Free memory cannot be measured on the simulator
|
|
|
|
free = 0
|