Edited at

M5GO(M5Stack)で気温・湿度・気圧メーター&時計表示

【2019/06/10更新】M5 UI.Flow v1.3 で変更になった MicroPython に合わせて対応

【2019/02/06更新】M5 UI.Flow (Firmware v1.1.2-en) では、各関数内で追加の import や global 宣言が不要になっていたので、関連するコメントおよび宣言を削除

【2018/12/07更新】動作環境を M5GO Cloud (io.m5go.com Firmware 0.15) から M5 UI.Flow (Firmware v1.0.1-en) への変更に合わせて修正

【2019/04/23更新】CO2濃度メーター表示を追加した版 M5GO(M5Stack)で気温・湿度・気圧・CO2濃度メーター&時計表示


概要

せっかくの M5GO で色々センサーもついているのに、結局、環境センサーだけで遊んでいます。

今回は、MicroPython での画面表示のお勉強を兼ねて以下の表示を行うプログラムを作成しました。


  • M5GO

  • M5 UI.Flow (Firmware v1.3.2-en)

  • 環境センサー(DHT12、BMP280)

  • MicroPythonで記述

  • アナログメーター風の気温・湿度・気圧計

  • アナログ時計


実行結果

IMG_1475.png


プログラム


env_meter.py

# UI.Flow 1.2 以前は buttonA, buttonB, buttonC

# UI.Flow 1.3 以降は btnA, btnB, btnC
from m5stack import lcd, btnA, btnB, btnC
from dht12 import DHT12
from bmp280 import BMP280
import i2c_bus
import machine
import time
import math
import gc
import ambient

# Ambientで取得したチャネルのチャネルId, ライトキーを指定
AMBIENT_CHANNEL_ID = チャネルId
AMBIENT_WRITE_KEY = 'ライトキー'

class Meter:
def __init__(self, x, y, w, h, tick_s, tick_e, color, title, value_format):
self.x = x # メーターの表示位置
self.y = y # メーターの表示位置
self.w = w # メーターの表示幅
self.h = h # メーターの表示高
self.tick_s = tick_s # 目盛の最小値
self.tick_e = tick_e # 目盛の最大値
self.title = title
self.value_format = value_format # 値をテキスト表示する際のフォーマット

self.center_x = x + w // 2 # 針の原点
self.center_y = y + int(h * 0.9) # 針の原点
self.prev_value = tick_s
self.prev_angle = None

lcd.roundrect(x, y, w, h, h // 10, lcd.BLACK, lcd.WHITE)
lcd.arc(self.center_x, self.center_y, int(h * 0.67), int(h * 0.07), -50, 50, color, color)
lcd.arc(self.center_x, self.center_y, int(h * 0.6), 2, -50, 50, lcd.BLACK)

# 目盛の値表示用フォント設定
if self.w == win_w:
lcd.font(lcd.FONT_Default, transparent=False)
else:
lcd.font(lcd.FONT_DefaultSmall, transparent=True)

fw, fh = lcd.fontSize()

tick = tick_s
tick_i = (tick_e - tick_s) // 4
for r in range(-50, 51, 5):
if r % 25 == 0:
# 目盛の最小値から最大値を4分割して目盛値を表示
lcd.lineByAngle(self.center_x - 1, self.center_y, int(h * 0.6), int(h * 0.1), r, lcd.BLACK)
lcd.lineByAngle(self.center_x, self.center_y, int(h * 0.6), int(h * 0.1), r, lcd.BLACK)
tick_text = str(tick)
text_width = lcd.textWidth(tick_text)
lcd.print(tick_text, self.center_x + int(math.sin(math.radians(r)) * h * 0.7) - text_width // 2,
self.center_y - int(math.cos(math.radians(r)) * h * 0.7) - fh,
lcd.BLACK)
tick += tick_i
else:
# 細かい目盛線を表示
lcd.lineByAngle(self.center_x, self.center_y, int(h * 0.6), int(h * 0.05), r, lcd.BLACK)

def update(self, value):
# 取得値をテキストでも表示
if self.w == win_w:
lcd.font(lcd.FONT_DejaVu24, transparent=False)
else:
lcd.font(lcd.FONT_DejaVu18, transparent=False)

fw, fh = lcd.fontSize()

if value is not None:
angle = int((value - self.tick_s) / (self.tick_e - self.tick_s) * 100 - 50)
if angle != self.prev_angle:
# 前回取得値の針を消去
if self.prev_angle is not None:
for i in range(-1, 2):
lcd.lineByAngle(self.center_x + i, self.center_y, int(self.h * 0.15), int(self.h * 0.42),
self.prev_angle, lcd.WHITE)
# 今回取得値の針を表示
for i in range(-1, 2):
lcd.lineByAngle(self.center_x + i, self.center_y, int(self.h * 0.15), int(self.h * 0.42),
angle, lcd.RED)

if self.title != '':
lcd.print(self.title, self.center_x - lcd.textWidth(self.title) // 2, self.y + self.h - int(fh * 2.4), lcd.BLACK)
self.prev_angle = angle

if value != self.prev_value:
text = self.value_format.format(value)
lcd.print(text, self.center_x - lcd.textWidth(text) // 2, self.y + self.h - int(fh * 1.2), lcd.BLACK)
self.prev_value = value
else:
text = self.value_format.format(self.prev_value)
lcd.print(text, self.center_x - lcd.textWidth(text) // 2, self.y + self.h - int(fh * 1.2), lcd.RED)

class Clock_digital:
def __init__(self, x, y):
self.x = x # 時計の表示位置
self.y = y # 時計の表示位置

def update(self):
lcd.font(lcd.FONT_DejaVu18, transparent=False)
lcd.print('{:02d}:{:02d}:{:02d}'.format(*time.localtime()[3:6]), self.x, self.y, lcd.BLACK)

class Clock:
def __init__(self, x, y, w, h, color):
self.x = x # 時計の表示位置
self.y = y # 時計の表示位置
self.w = w # 時計の表示幅
self.h = h # 時計の表示高
self.center_x = x + w // 2 # 針の中心
self.center_y = y + h // 2 # 針の中心
self.hour_deg = 0
self.minute_deg = 0
self.second_deg = 0

lcd.roundrect(x, y, w, h, h // 10, lcd.BLACK, lcd.WHITE)
# 0 から 360 とは書けないので、半分の円弧を合わせる
lcd.arc(self.center_x, self.center_y, int(h * 0.39), int(h * 0.08), 0, 180, color, color)
lcd.arc(self.center_x, self.center_y, int(h * 0.39), int(h * 0.08), 180, 360, color, color)

if self.w == win_w:
lcd.font(lcd.FONT_Default, transparent=False)
else:
lcd.font(lcd.FONT_DefaultSmall, transparent=True)

fw, fh = lcd.fontSize()
hour = 12
for r in range(0, 360, 360 // 60):
if r % (360 // 12) == 0:
# 1〜12の位置に黒点および数字を表示
lcd.circle(self.center_x + int(math.sin(math.radians(r)) * h / 2 * 0.7),
self.center_y - int(math.cos(math.radians(r)) * h / 2 * 0.7), 2, lcd.BLACK, lcd.BLACK)
hour_text = str(hour)
text_width = lcd.textWidth(hour_text)
lcd.print(hour_text, self.center_x + int(math.sin(math.radians(r)) * h / 2 * 0.85) - text_width // 2,
self.center_y - int(math.cos(math.radians(r)) * h / 2 * 0.85) - fh // 2,
lcd.BLACK)
hour = (hour + 1) % 12
else:
lcd.pixel(self.center_x + int(math.sin(math.radians(r)) * h / 2 * 0.7),
self.center_y - int(math.cos(math.radians(r)) * h / 2 * 0.7), lcd.BLACK)

def update(self):
def needle(n, m, deg, l, color):
for i in range(n, n + m):
if deg >= 315 or deg < 45 or deg >= 135 and deg < 225:
x, y = i, 0
else:
x, y = 0, i
lcd.lineByAngle(self.center_x + x, self.center_y + y,
0, l, deg, color)

# 時分秒の各針の角度を計算
(year, month, mday, hour, minute, second, weekday, yearday) = time.localtime()
second_deg = second * 6
minute_deg = minute * 6 + second_deg // 60
hour_deg = hour % 12 * 30 + minute_deg // 12

# 時針の消去(角度が変わっていないときは消さない)
if hour_deg != self.hour_deg:
needle(-2, 4, self.hour_deg, int(self.h / 2 * 0.3), lcd.WHITE)
# 分針の消去(角度が変わっていないときは消さない)
if minute_deg != self.minute_deg:
needle(-1, 2, self.minute_deg, int(self.h / 2 * 0.45), lcd.WHITE)
# 秒針の消去
needle(0, 1, self.second_deg, int(self.h / 2 * 0.6), lcd.WHITE)

self.second_deg = second_deg
self.minute_deg = minute_deg
self.hour_deg = hour_deg

# 時針の描画(4本線)
needle(-2, 4, hour_deg, int(self.h / 2 * 0.3), lcd.BLACK)
# 分針の描画(2本線)
needle(-1, 2, minute_deg, int(self.h / 2 * 0.45), lcd.BLACK)
# 秒針の描画(1本線)
needle(0, 1, self.second_deg, int(self.h / 2 * 0.6), lcd.RED)
# 中心に赤丸
lcd.circle(self.center_x, self.center_y, 3, lcd.RED, lcd.RED)

def env_meter_update(meter_mode):
while True:
# 次の秒までの差分(ミリ秒)を求めてスリープ
time.sleep_ms(1000 - int(time.time() % 1 * 1000))

# 各メーターおよび時計の更新
localtime = time.localtime()
localtime_str = '{}-{:02d}-{:02d} {:02d}:{:02d}:{:02d}'.format(*localtime[:6])

# 時計の表示を更新
clock.update()

try:
# DHT12 から湿度を取得
dht12.measure()
h = dht12.humidity()

# BMP280 から気温、気圧を取得
t, p = bmp280.values
except Exception as e:
print('{}-{:02d}-{:02d} {:02d}:{:02d}:{:02d}'.format(*time.localtime()[:6]), ' Script Name: ', __name__)
print('Exception: ', e)
t, h, p = None, None, None

# それぞれのメーターに取得値を表示
if meter_mode == 0 or meter_mode == 1:
t_meter.update(t)
if meter_mode == 0 or meter_mode == 2:
h_meter.update(h)
if meter_mode == 0 or meter_mode == 3:
p_meter.update(p)

# Ambientへの送信は1分間隔で行う。
# localtime[5](秒) == 0 の時に送信
if localtime[5] == 0 and t is not None:
try:
am.send({'created': localtime_str,
'd1': dht12.temperature(), 'd2': h, 'd3': t, 'd4': p})
except Exception as e:
print('{}-{:02d}-{:02d} {:02d}:{:02d}:{:02d}'.format(*time.localtime()[:6]), ' Script Name: ', __name__)
print('Ambient send error: ', e)

gc.collect()

# button A が押された時は表示を切り替える
if btnA.wasPressed():
return

# M5 UI.Flow からの実行ではなく、APP.LIST に登録して実行する場合は、
# プログラム内でネットワーク接続を行う必要がある。
# Connect network
#
# UI.Flow 1.2 以前は以下
# import wifisetup
# wifisetup.auto_connect()
#
# UI.Flow 1.3 以降は以下
import wifiCfg
wifiCfg.autoConnect(lcdShow=True)

# 日本時間に同期
# UI.Flow 1.2 以前は以下
'''
rtc = machine.RTC()
rtc.ntp_sync('ntp.nict.jp', tz='JST-9')
# M5GOのfirmwareがv0.11ではntp_syncでtzを指定するとエラーになるので以下で対応
# rtc.ntp_sync('ntp.nict.jp')
# sys.tz('JST-9')

# 同期が完了するまで100ms程度かかる
for i in range(100):
if rtc.synced():
print('synced.')
break
print(i, end=' ')
time.sleep_ms(10)
'''
# UI.Flow 1.3 以降は以下
# ntptime は以下のモジュールの改訂版(オフセット指定を追加)
# https://github.com/micropython/micropython/blob/master/ports/esp8266/modules/ntptime.py
import ntptime
ntptime.settime(9*60*60) # +09:00:00 for JST

# UI.Flow 1.2 以前はタイムスタンプの出力に以下のように time.strftime を使用していたが
#print(time.strftime('%Y-%m-%d %H:%M:%S', time.localtime()), ' Script Name: ', __name__)
# UI.Flow 1.3 以降は time.strftime が使えないため以下で対応
print('{}-{:02d}-{:02d} {:02d}:{:02d}:{:02d}'.format(*time.localtime()[:6]), ' Script Name: ', __name__)

am = ambient.Ambient(AMBIENT_CHANNEL_ID, AMBIENT_WRITE_KEY)

i2c = i2c_bus.get(i2c_bus.M_BUS)
dht12 = DHT12(i2c)
bmp280 = BMP280(i2c)

lcd.setColor(lcd.BLACK, lcd.WHITE)
lcd.setTextColor(lcd.BLACK, lcd.WHITE)
lcd.clear(lcd.BLACK)

win_w, win_h = lcd.winsize() # (320, 240)

meter_mode = 0
while True:
lcd.clear(lcd.BLACK)

if meter_mode == 0:
# 画面を4分割して、気温計、湿度計、気圧計、CO2濃度計を表示
# 表示フォーマットは、表示桁数が減った時に(気圧が1000から999になった時等)
# 前の表示を消すために前後に空白を入れている
t_meter = Meter(0, 0, win_w // 2, win_h // 2, 0, 40, lcd.ORANGE, 'Temp', ' {:.1f}C ')
h_meter = Meter(win_w // 2, 0, win_w // 2, win_h // 2, 20, 100, lcd.CYAN, 'Hum', ' {:.1f}% ')
p_meter = Meter(0, win_h // 2, win_w // 2, win_h // 2, 960, 1040, lcd.YELLOW, 'Baro', ' {:.1f}hPa ')
clock = Clock(win_w // 2, win_h // 2, win_w // 2, win_h // 2, lcd.GREENYELLOW)
elif meter_mode == 1:
# 全画面で気温計を表示
t_meter = Meter(0, 0, win_w, win_h, 0, 40, lcd.ORANGE, 'Temp', ' {:.1f}C ')
clock = Clock_digital(win_w // 2 - 43, 10)
elif meter_mode == 2:
# 全画面で湿度計を表示
h_meter = Meter(0, 0, win_w, win_h, 20, 100, lcd.CYAN, 'Hum', ' {:.1f}% ')
clock = Clock_digital(win_w // 2 - 43, 10)
elif meter_mode == 3:
# 全画面で気圧計を表示
p_meter = Meter(0, 0, win_w, win_h, 960, 1040, lcd.YELLOW, 'Baro', ' {:.1f}hPa ')
clock = Clock_digital(win_w // 2 - 43, 10)
elif meter_mode == 4:
# 全画面で時計を表示
clock = Clock(0, 0, win_w, win_h, lcd.GREENYELLOW)

env_meter_update(meter_mode)

meter_mode = (meter_mode + 1) % 5



補足

以下は M5GO Cloud (io.m5go.com Firmware 0.15) で実行した場合の動作でしたが、 UI.Flow の Firmware ではプログラムは MainThread で動くので「stack overflow」で落ちることは(多分)ありません。

上記プログラムで一応動きますが、時間が経つと(今日動かしたときは半日ほどして)何が原因なのかはよくわかりませんが、以下のように「stack overflow」で落ちることがあります。

***ERROR*** A stack overflow in task m5go_run has been detected.

abort() was called at PC 0x4009230c on core 1

「stack overflow」を回避する代替案として以下のような方法が考えられますが、1./2. で長時間(何日も動かしても)問題ないかは確認したことなく、ちゃんと動かすなら 3. の方法を考えた方がいいように思います。


  1. Stack Sizeを大きくして別スレッドで動かす(M5GO Cloudから動かした場合はStack Sizeが4096のスレッドで動く)

  2. Timer起動でメインスレッドで動かす

  3. M5GO Cloudから動かすのをやめ、boot.py/main.py を置き換えて直接動かす