MENU

Raspberry Pi Pico と e-Paper と Google Spreadsheet で 服薬管理システムを作る。

目次

 1.Raspberry Pi Pico で服薬管理システムを作る

私は毎朝のまなければならない薬があるのだが、マイにとのことだからこそ「あれ?今日は薬飲んだっけ?」というのが定期的に発生する。
もちろん薬に日付を書いたり、スマホに飲んだ記録でもつければいいのだけど、残念だけどそこまでマメな性格ではない。
そこで、飲んだらボタンをぽちっと押せば、その時刻を記録してくれるシステムを作ることにした。もちろんボタンを押すこと自体を忘れてしまったら意味がないわけで、その改善には飲んでいる動作を機械学習で認識させ記録するようなもの(Raspberry Pi5とAI Kitでできそうではある)が必要になる。それはそれでカメラを設置して大掛かりになるのが予想される。
今回は以下の要件で作ってみることにした
  1. ボタンを押すだけで飲んだ日時が記録されること
  2. ボタンを押した日時はクラウドに保存されること(どこからでも確認できること)
  3. その場でも飲んだ日時が確認できること

結果的に、下記の部品で作ってみることにした

  • Raspberry Pi Pico WH  ← 通信用のWifiとE-Paper用にピン付きが必要
  • E-Paper 私はWaveshareのPico e-Paper 2.13 (B)を使用したがなんでもいい。
  • Google Spread sheet

このようなものが出来上がります。

選択したフォントサイズ的には表示できて5行。
1行目を直近で飲んだ日時ということで、赤色表示にすることにしています。
データベースの代わりとして、Google Spread sheetを使用することとして、同じ内容は以下のように記録されます。

2.作り方

何よりも、Raspberry Piがなければ始まらない。Picoである必要はないが、今回はボタンを押したら記録と表示をするだけであるため、OSのないPicoでも十分と考えた。
標準のPythonライブラリが使いたかったり、自身がサーバのようになる場合にはRaspberry Pi Zero以上のほうが良いのだろう。

2.1 Raspberry Pi Picoの開発環境整備

すでに済んでいる場合はスキップ。
今回はMicroPythonを使用するのでそれ用の開発環境を。
安定はThonny

ツール→オプション→インタプリタからMicroPython(Raspberry Pi Pico)を選ぶ。

まだ、一度もRaspberry Pi Picoを使用しておらず、ファームウェアの導入もしていない場合は、BOOTSELボタンを押しながらUSB接続を実施して、 ThonnyのInstall or update MicroPython ※画面右下のリンク

を選択してファームウェアの導入を実施する必要がある。

VSCODEを使用することもできる。

2.2必要なライブラリ

今回はこの3つを組み合わせるだけでほぼすべてできる。
オリジナルのコードはほとんどなし、、、😓

MicroPython-GoogleSheet

Micro PythonからGoogle Spread sheetを操作するためのライブラリ。
APIキーを使用せずにGoogle Apps Scriptから古オープンにしてしまっているので、セキュリティが少し気なるところはあるが、まあ薬を飲んだ時刻しか記録されていないので、関係のない人にはどうでもいい情報なので気にしないことにする。
こちらの手順に沿って、Thonnyからライブラリのインストールを実施する。
そのあと、記録をしたいSpreadsheetを作成して、
こちらの手順でアクセスするためのクレデンシャルを入手しておく。
以上で、
        # Create Instance 
        ggsheet = MicroGoogleSheet(google_sheet_url,google_sheet_name)
        ggsheet.set_DeploymentID(google_app_deployment_id)
        ggsheet.appendRow(1,[str(month) + '月' + str(day) + '日' + str(hour) + ':' + str(min) + 'に服用しました'])
        
        dispnow = ggsheet.getRow(1)
        now_string = ''.join(dispnow)
このような形で、シートへの行の追加や読み取りを行うことができるようになる。

現在時刻を取得する

Raspberry pi picoにはOSがないため、現在時刻は都度取らなければならない。
WIFIに接続し、現在時刻を取ってくるところ。
私は、ntp.nict.jpを使用させていただいている。
# Connect to Network
        sta_if = WLAN(STA_IF)
        sta_if.active(True)
        if not sta_if.isconnected():
            print("Connecting to wifi: ", ssid)
            sta_if.connect(ssid, password)
            while not sta_if.isconnected():
                pass
            print("Connection successful")

        ntptime.host = "ntp.nict.jp"
        ntptime.settime()
        (year, month, day, hour, min, sec, weekday_num,_) = time.localtime(time.time() + 9 * 60 * 60)

E-Paperに表示を行う。

ここは、メーカーのサンプルを使用するのが一番良い。
英語、および線を描画するサンプルが手に入るが肝心の日本語であったりまたフォントサイズを大きくすることができない。
そこはこちらのサイトが大変に参考になりました。
そのまま使わせていただいています。ありがとうございます。

ボタンを押したときだけ稼働する、稼働中はLEDで知らせる

E-Paperの描画にはとても時間がかかるため、ボタンを押して本当に動いているのか、処理中なのか、終わったのか。この辺は管理できるようにしたい。
ボタンを外に出してもよかったが今回はBOOTSELボタンを活用し、LEDも標準のものを使用することにした。簡易的に。

rp2.bootsel_button() == 1:
        led.on()       
 

3.コード

私はインフラエンジニアを糧としているため、コードのきれいさに関してはご勘弁をいただきたい。
また、使用したライブラリ、参考にしたサイトに感謝を申し上げます。
 
 

# Import Library 
from ggsheet import MicroGoogleSheet
from network import WLAN,STA_IF
import ntptime
import time
from machine import Pin, SPI
from machine import Pin, SPI
import framebuf
import utime
from mfont import mfont

EPD_WIDTH       = 122
EPD_HEIGHT      = 250

RST_PIN         = 12
DC_PIN          = 8
CS_PIN          = 9
BUSY_PIN        = 13

class EPD_2in13_B_V4_Portrait:
    def __init__(self):
        self.reset_pin = Pin(RST_PIN, Pin.OUT)
        
        self.busy_pin = Pin(BUSY_PIN, Pin.IN, Pin.PULL_UP)
        self.cs_pin = Pin(CS_PIN, Pin.OUT)
        if EPD_WIDTH % 8 == 0:
            self.width = EPD_WIDTH
        else :
            self.width = (EPD_WIDTH // 8) * 8 + 8
        self.height = EPD_HEIGHT
        
        self.spi = SPI(1)
        self.spi.init(baudrate=4000_000)
        self.dc_pin = Pin(DC_PIN, Pin.OUT)
        
        
        self.buffer_balck = bytearray(self.height * self.width // 8)
        self.buffer_red = bytearray(self.height * self.width // 8)
        self.imageblack = framebuf.FrameBuffer(self.buffer_balck, self.width, self.height, framebuf.MONO_HLSB)
        self.imagered = framebuf.FrameBuffer(self.buffer_red, self.width, self.height, framebuf.MONO_HLSB)
        self.init()

    def digital_write(self, pin, value):
        pin.value(value)

    def digital_read(self, pin):
        return pin.value()

    def delay_ms(self, delaytime):
        utime.sleep(delaytime / 1000.0)

    def spi_writebyte(self, data):
        self.spi.write(bytearray(data))

    def module_exit(self):
        self.digital_write(self.reset_pin, 0)

    # Hardware reset
    def reset(self):
        self.digital_write(self.reset_pin, 1)
        self.delay_ms(50)
        self.digital_write(self.reset_pin, 0)
        self.delay_ms(2)
        self.digital_write(self.reset_pin, 1)
        self.delay_ms(50)


    def send_command(self, command):
        self.digital_write(self.dc_pin, 0)
        self.digital_write(self.cs_pin, 0)
        self.spi_writebyte([command])
        self.digital_write(self.cs_pin, 1)

    def send_data(self, data):
        self.digital_write(self.dc_pin, 1)
        self.digital_write(self.cs_pin, 0)
        self.spi_writebyte([data])
        self.digital_write(self.cs_pin, 1)
        
    def send_data1(self, buf):
        self.digital_write(self.dc_pin, 1)
        self.digital_write(self.cs_pin, 0)
        self.spi.write(bytearray(buf))
        self.digital_write(self.cs_pin, 1)
        
    def ReadBusy(self):
        print('busy')
        while(self.digital_read(self.busy_pin) == 1): 
            self.delay_ms(10) 
        print('busy release')
        self.delay_ms(20)
        
    def TurnOnDisplay(self):
        self.send_command(0x20)  # Activate Display Update Sequence
        self.ReadBusy()

    def SetWindows(self, Xstart, Ystart, Xend, Yend):
        self.send_command(0x44) # SET_RAM_X_ADDRESS_START_END_POSITION
        self.send_data((Xstart>>3) & 0xFF)
        self.send_data((Xend>>3) & 0xFF)

        self.send_command(0x45) # SET_RAM_Y_ADDRESS_START_END_POSITION
        self.send_data(Ystart & 0xFF)
        self.send_data((Ystart >> 8) & 0xFF)
        self.send_data(Yend & 0xFF)
        self.send_data((Yend >> 8) & 0xFF)
        
    def SetCursor(self, Xstart, Ystart):
        self.send_command(0x4E) # SET_RAM_X_ADDRESS_COUNTER
        self.send_data(Xstart & 0xFF)

        self.send_command(0x4F) # SET_RAM_Y_ADDRESS_COUNTER
        self.send_data(Ystart & 0xFF)
        self.send_data((Ystart >> 8) & 0xFF)
    

    def init(self):
        print('init')
        self.reset()
        
        self.ReadBusy()   
        self.send_command(0x12)  #SWRESET
        self.ReadBusy()   

        self.send_command(0x01) #Driver output control      
        self.send_data(0xf9)
        self.send_data(0x00)
        self.send_data(0x00)

        self.send_command(0x11) #data entry mode       
        self.send_data(0x03)

        self.SetWindows(0, 0, self.width-1, self.height-1)
        self.SetCursor(0, 0)

        self.send_command(0x3C) #BorderWaveform
        self.send_data(0x05)

        self.send_command(0x18) #Read built-in temperature sensor
        self.send_data(0x80)

        self.send_command(0x21) #  Display update control
        self.send_data(0x80)
        self.send_data(0x80)

        self.ReadBusy()
        
        return 0       
        
    def display(self):
        self.send_command(0x24)
        self.send_data1(self.buffer_balck)
        
        self.send_command(0x26)
        self.send_data1(self.buffer_red)  

        self.TurnOnDisplay()

    
    def Clear(self, colorblack, colorred):
        self.send_command(0x24)
        self.send_data1([colorred] * self.height * int(self.width / 8))
        
        self.send_command(0x26)
        self.send_data1([colorred] * self.height * int(self.width / 8))
                                
        self.TurnOnDisplay()

    def sleep(self):
        self.send_command(0x10) 
        self.send_data(0x01)
        
        self.delay_ms(2000)
        self.module_exit()
        
class EPD_2in13_B_V4_Landscape:
    def __init__(self):
        self.reset_pin = Pin(RST_PIN, Pin.OUT)
        
        self.busy_pin = Pin(BUSY_PIN, Pin.IN, Pin.PULL_UP)
        self.cs_pin = Pin(CS_PIN, Pin.OUT)
        if EPD_WIDTH % 8 == 0:
            self.width = EPD_WIDTH
        else :
            self.width = (EPD_WIDTH // 8) * 8 + 8
        self.height = EPD_HEIGHT
        
        self.spi = SPI(1)
        self.spi.init(baudrate=4000_000)
        self.dc_pin = Pin(DC_PIN, Pin.OUT)
        
        
        self.buffer_balck = bytearray(self.height * self.width // 8)
        self.buffer_red = bytearray(self.height * self.width // 8)
        self.imageblack = framebuf.FrameBuffer(self.buffer_balck, self.height, self.width, framebuf.MONO_VLSB)
        self.imagered = framebuf.FrameBuffer(self.buffer_red, self.height, self.width, framebuf.MONO_VLSB)
        self.init()

    def digital_write(self, pin, value):
        pin.value(value)

    def digital_read(self, pin):
        return pin.value()

    def delay_ms(self, delaytime):
        utime.sleep(delaytime / 1000.0)

    def spi_writebyte(self, data):
        self.spi.write(bytearray(data))

    def module_exit(self):
        self.digital_write(self.reset_pin, 0)

    # Hardware reset
    def reset(self):
        self.digital_write(self.reset_pin, 1)
        self.delay_ms(50)
        self.digital_write(self.reset_pin, 0)
        self.delay_ms(2)
        self.digital_write(self.reset_pin, 1)
        self.delay_ms(50)


    def send_command(self, command):
        self.digital_write(self.dc_pin, 0)
        self.digital_write(self.cs_pin, 0)
        self.spi_writebyte([command])
        self.digital_write(self.cs_pin, 1)

    def send_data(self, data):
        self.digital_write(self.dc_pin, 1)
        self.digital_write(self.cs_pin, 0)
        self.spi_writebyte([data])
        self.digital_write(self.cs_pin, 1)
        
    def send_data1(self, buf):
        self.digital_write(self.dc_pin, 1)
        self.digital_write(self.cs_pin, 0)
        self.spi.write(bytearray(buf))
        self.digital_write(self.cs_pin, 1)
        
    def ReadBusy(self):
        print('busy')
        while(self.digital_read(self.busy_pin) == 1): 
            self.delay_ms(10) 
        print('busy release')
        self.delay_ms(20)
        
    def TurnOnDisplay(self):
        self.send_command(0x20)  # Activate Display Update Sequence
        self.ReadBusy()

    def SetWindows(self, Xstart, Ystart, Xend, Yend):
        self.send_command(0x44) # SET_RAM_X_ADDRESS_START_END_POSITION
        self.send_data((Xstart>>3) & 0xFF)
        self.send_data((Xend>>3) & 0xFF)

        self.send_command(0x45) # SET_RAM_Y_ADDRESS_START_END_POSITION
        self.send_data(Ystart & 0xFF)
        self.send_data((Ystart >> 8) & 0xFF)
        self.send_data(Yend & 0xFF)
        self.send_data((Yend >> 8) & 0xFF)
        
    def SetCursor(self, Xstart, Ystart):
        self.send_command(0x4E) # SET_RAM_X_ADDRESS_COUNTER
        self.send_data(Xstart & 0xFF)

        self.send_command(0x4F) # SET_RAM_Y_ADDRESS_COUNTER
        self.send_data(Ystart & 0xFF)
        self.send_data((Ystart >> 8) & 0xFF)
    

    def init(self):
        print('init')
        self.reset()
        
        self.ReadBusy()   
        self.send_command(0x12)  #SWRESET
        self.ReadBusy()   

        self.send_command(0x01) #Driver output control      
        self.send_data(0xf9)
        self.send_data(0x00)
        self.send_data(0x00)

        self.send_command(0x11) #data entry mode       
        self.send_data(0x07)

        self.SetWindows(0, 0, self.width-1, self.height-1)
        self.SetCursor(0, 0)

        self.send_command(0x3C) #BorderWaveform
        self.send_data(0x05)

        self.send_command(0x18) #Read built-in temperature sensor
        self.send_data(0x80)

        self.send_command(0x21) #  Display update control
        self.send_data(0x80)
        self.send_data(0x80)

        self.ReadBusy()
        
        return 0       
        
    def display(self):
        self.send_command(0x24)
        for j in range(int(self.width / 8) - 1, -1, -1):
            for i in range(0, self.height):
                self.send_data(self.buffer_balck[i + j * self.height])
        
        self.send_command(0x26)
        for j in range(int(self.width / 8) - 1, -1, -1):
            for i in range(0, self.height):
                self.send_data(self.buffer_red[i + j * self.height])

        self.TurnOnDisplay()

    
    def Clear(self, colorblack, colorred):
        self.send_command(0x24)
        self.send_data1([colorblack] * self.height * int(self.width / 8))
        
        self.send_command(0x26)
        self.send_data1([colorred] * self.height * int(self.width / 8))
                                
        self.TurnOnDisplay()

    def sleep(self):
        self.send_command(0x10) 
        self.send_data(0x01)
        
        self.delay_ms(2000)
        self.module_exit()
      
def jpredchar(fd, epd, x, y, width, height):
    bn = (width+7)>>3
    py = y
    for i in range(0, len(fd), bn):
        px = x
        for j in range(bn):
            for k in range(8 if (j+1)*8 <=width else width % 8):
                if fd[i+j] & 0x80>>k:
                    epd.imagered.pixel(px + k, py, 0x00)
            px+=8
        py+=1

def jpblackchar(fd, epd, x, y, width, height): 
    bn = (width+7)>>3
    py = y
    for i in range(0, len(fd), bn):
        px = x
        for j in range(bn):
            for k in range(8 if (j+1)*8 <=width else width % 8):
                if fd[i+j] & 0x80>>k:
                    epd.imageblack.pixel(px + k, py, 0x00)
            px+=8
        py+=1


def jpredtext(str, x, y, mf, epd):
    mf.begin()
    for c in str:
        d = mf.getFont(ord(c))
        jpredchar(d, epd, x, y, mf.getWidth(), mf.getHeight())
        x = x + mf.getWidth()
    mf.end()

def jpblacktext(str, x, y, mf, epd):
    mf.begin()
    for c in str:
        d = mf.getFont(ord(c))
        jpblackchar(d, epd, x, y, mf.getWidth(), mf.getHeight())
        x = x + mf.getWidth() 
    mf.end()


# Network Creadential 
ssid = "<WIFIのSSIDを入れてください>"    
password = "<WIFI接続用のパスワード入れてください>"

# Google Sheet Credential 
google_sheet_url = "https://docs.google.com/spreadsheets/<Google Spread sheetのURLにしてください>"
google_sheet_name = "<Google Spread sheetのシート名にしてください>"
google_app_deployment_id = "Google appのIDにしてください"

led = Pin('LED', Pin.OUT)
led.off() 

while True:
    if rp2.bootsel_button() == 1:
        led.on()

        # Connect to Network
        sta_if = WLAN(STA_IF)
        sta_if.active(True)
        if not sta_if.isconnected():
            print("Connecting to wifi: ", ssid)
            sta_if.connect(ssid, password)
            while not sta_if.isconnected():
                pass
            print("Connection successful")

        ntptime.host = "ntp.nict.jp"
        ntptime.settime()
        (year, month, day, hour, min, sec, weekday_num,_) = time.localtime(time.time() + 9 * 60 * 60)

        print(f"Successfully got datetime: {year:04}-{month:02}-{day:02} {hour:02}:{min:02}:{sec:02}")

        # Create Instance 
        ggsheet = MicroGoogleSheet(google_sheet_url,google_sheet_name)
        ggsheet.set_DeploymentID(google_app_deployment_id)

        # Update the data to a specific cell (Row,Column,Data)
        #ggsheet.updateCell(1,1,"Hello this is my first data")
        # Get the data from a specific cell (Row,Column)
        #print(ggsheet.getCell(1,1))

        # Delete the data from a specific cell (Row,Column)
        #ggsheet.deleteCell(1,1)

        # Append the data to a specific row (Row, Data List)
        ggsheet.appendRow(1,[str(month) + '月' + str(day) + '日' + str(hour) + ':' + str(min) + 'に服用しました'])
        
        dispnow = ggsheet.getRow(1)
        now_string = ''.join(dispnow)
        
        rec1 = ggsheet.getRow(2)
        rec1_string = ''.join(rec1)
        
        rec2 = ggsheet.getRow(3)
        rec2_string = ''.join(rec2)
        
        rec3 = ggsheet.getRow(4)
        rec3_string = ''.join(rec3)
        
        epd = EPD_2in13_B_V4_Landscape()
        epd.Clear(0xff, 0xff)
        epd.imageblack.fill(0xff)
        epd.imagered.fill(0xff)
        mf = mfont()
        mf.setFontSize(16)    
        disps="◆ お薬管理システム ◆"
        jpblacktext(disps, 5, 10, mf, epd)
        jpredtext(now_string, 5, 35, mf, epd)
        jpblacktext(rec1_string, 5, 60, mf, epd)
        jpblacktext(rec2_string, 5, 85, mf, epd)
        jpblacktext(rec3_string, 5, 110, mf, epd)
        epd.display()
        epd.delay_ms(2000)
        epd.sleep()
        
        
        # Update the data in a specific row (Row, Data List) 
        #ggsheet.updateRow(1,[3,2,1,"Row 1 Updated!"])

        # Get all of the data from a specific row (Row)
        #print(ggsheet.getRow(1))
        # Delete the data in a specific row (Row)
        #ggsheet.deleteRow(1)

        # Append the data to a specific column (Column, Data List)
        #ggsheet.appendColumn(1,[1,2,3,"Column 1 Appended!"])

        # Update the data to a specific column (Column, Data List)
        #ggsheet.updateColumn(1,[3,2,1,"Column 1 Updated!"])

        # Get all of the data from a specific column (Column)
        #print(ggsheet.getColumn(1))

        # Delete the data in a specific column (Column)
        #ggsheet.deleteColumn(1)

        led.off() 
        time.sleep(5)
  

よかったらシェアしてね!
  • URLをコピーしました!
  • URLをコピーしました!

この記事を書いた人

コメント

コメントする

目次