Hyodotのアイコン画像
Hyodot 2024年10月27日作成
製作品 製作品 閲覧数 802
Hyodot 2024年10月27日作成 製作品 製作品 閲覧数 802

ワイングラスの音色を楽器にしてMIDI演奏する

外観

全体の構成

動画

編集画面やMIDI演奏の動画です。
Facebookに紹介動画

用途(目的)

  • 音楽の楽しみ方は色々あると思いますが、この作品は、楽器の音色を自分好みに編集して演奏を楽しむためのものです。
  • 楽器が弾ける人なら、編集した楽器で演奏できますが、私は楽器を演奏できないので、インターネットからパソコンにMIDIファイルをダウンロードして、無料ソフト「Domino」で演奏させています。

概要(作品全体の説明)

  • FM音源IC(YMF825)を利用して、手持ちのワイングラスの音色を真似た楽器作り。
  • ワイングラスは1つあればOK。音色の特徴を設定すれば、音階はYMF825にお任せ。
  • 2つのPicoをシリアル通信で連携させ、1つをPicoMainとし音源編集用、もう1つをPicoSubとしYMF825を使ったMIDI再生(発音)用にしました。
  • パラメータを設定しやすいように編集用のPicoMainにはPS/2マウスとPS/2キーボードを繋げ、TFT-LCDを接続しました。
  • パソコンからのMIDI信号をUSBに出力して、PicoSub側でDIN-MIDIインターフェースで受けて、再生(発音)します。
  • Pico側でUSB-MIDI対応するのはハードルが高いので、DIN-MIDIインターフェースで受けることにしました。
  • この関係で、パソコンのUSB端子に接続してDIN-MIDIに変換して出力できるケーブルが必要です。(Amazonで購入しました)
  • 本作品とは直接関係ありませんが、声紋のようなスペクトログラム表示しているモジュールは、液晶表示付きマイコンボードDISC-F746です。(スペクトログラム表示のアプリを入れています)
  • 音楽やシンセサイザーは ド素人ですが、音色は倍音の構成を真似るといいようです。声紋のようなスペクトログラム表示が分析ツールになります。楽器では線状に倍音の模様が現われますので、それを真似るようにYMF825の音源パラメータを調整します。
  • YMF825は、16音源16和音対応ですが、動画ではワイングラスの音ということで、1音源だけにしています。

主要部品

キャプションを入力できます

  • この作品は、PS/2キーボードに対応していますが、USBキーボードからPS/2キーボードに逆変換する 「PS/2 USB逆変換アダプタ」を使うと、USB無線のキーボードが使えます。
  • 「PS/2 USB逆変換アダプタ」は、マウスには対応していませんので、マウスを無線化することはできません。

回路図

回路全体の写真

全体ブロック図

(UARTデバイスやSPIデバイスとの接続は、個別接続図があります)
全体ブロック図

UARTデバイスやSPIデバイスの個別接続図

PicoMainとTFT-LCD ST7789との接続図

キャプションを入力できます

PicoMainとRFID RC522との接続図

キャプションを入力できます

PicoMainとPS/2キーボード、PS/2マウスとのgpio接続

PicoMainとPS/2キーボード、PS/2マウスとのgpio接続

PicoSubとYMF825とのSPI-0接続図

PicoSubとYMF825とのSPI-0接続図

PicoMainとPicoSubとのgpio接続図

PicoMainとPicoSubとのgpio接続図

MIDIインターフェースとPicoMain、PicoSubとのUART-0接続図

キャプションを入力できます

MIDIインターフェースの改良図面

MIDI入出力の精度を上げるため、MIDI入力をカレントミラー回路に、MIDI出力にバッファを入れました。赤色ボールペン手書きの部分。
MIDIインターフェースの改良図面

プログラム

プログラムは数が多くサイズも大きいので、末尾に掲載します。

ライブラリ・プログラムについて

RC522 RFID リーダー モジュールを使用するため、次のサイトのライブラリを引用しました。
https://how2electronics.com/using-rc522-rfid-reader-module-with-raspberry-pi-pico/
ST7789(TFT-LCD)用のライブラリはDFROBOTサイトのライブラリを引用しました。
https://community.dfrobot.com/makelog-311712.html
なお、このST7789ライブラリにはSPI設定に問題があります。
spi1 = machine.SPI(1, baudrate=40000000, polarity=1)
とあるところは
spi1 = machine.SPI(1, baudrate=40000000, polarity=1, phase=1)
と、phase=1の設定が追加すべきです。(表示しないLCDがあり困りました)
上記以外のライブラリは、自作です。
YMF825のライブラリは、以前には存在したYAMAHAのサイトからダウンロードしていたC言語のライブラリをベースに、大平駿介様の了解を得て次のサイトの記事を参考にMicroPuthonで作成しました。
https://github.com/ohira-s/YMF825pico

開発環境

  • PicoMain PicoSub、それぞれを、Raspberry Pi 4BとUSB接続し、標準でインストールされていたThonnyでソフト開発しました。
  • シリアル出力と、Picoに接続したLEDのみでデバッグしました。
  • 回路図にはデバッグに使用したLEDの配線を省略しています。

編集画面の紹介

初期画面

初期画面は、音源選択画面になっています。YMF825初期設定の16音源の名称一覧を表示し、どの音源を編集するか選択します。
初期画面

YMF825にはYAMAHAのサイトからダウンロードした16音源を初期セットしています。新しく編集する場合でも初期設定の音源から近いものを選んで編集し、編集の最後で名称変更できるようにしています。

音源編集画面

音源編集画面は、YMF825の16の音源毎に共通項である3要素(BO、LFO、ALG)の編集画面と、4オペレータ17要素(SR、XOF、KSR、RR、DR、AR、SL、TL、KSL、DAM、EAM、DVB、EVB、MUL、DT、WS、FB)の編集画面があります。
音源編集画面

  • 各要素の項目移動には、キーボードの矢印キー、または、マウスを動かします。Enterキー、または、マウスの左クリックで項目の編集になります。
  • 3要素の編集画面、または、17要素の編集画面から次の編集画面に移行するには、マウスで右クリック、または、Escキーです。
  • パラメータの値変更は、マウスのスクロール、または、テンキーを使います。(BackSpaceキーで末尾をクリア、Deleteキーで項目をクリア)

音源編集の最後に保存場所(0-15)と名称編集します。
キャプションを入力できます

  • 設定したパラメータは、Pico のメモリには保存されません。
  • RFIDのタグ(またはカード)を利用してパラメータの保存と読み込みができるようRead RFID と Write RFID のメニューを用意してます。保存しておくと後で音源のパラメータの読み込みができます。(次項で説明)

音源編集以外のメニュー

音源編集以外のメニューは、初期画面の音源選択画面で、キーボードの「PtSc」(PrintScreen)キーで表示します。
音源編集以外のメニュー

音源編集以外のメニューとしては、

  • Edit Tone:前記で紹介した音源編集
  • Read RFID:RFID Tag(または Card)からの読み込み(音源設定パラメータの読み込み)
  • Write RFID:RFID Tag(または Card)への書き込み(音源設定パラメータの書き込み)
  • Play MIDI:MIDI入力を演奏(パソコンとつなぐ、鍵盤とつなぐ)
    などがあります。

YMF825の音色編集について

  • YMF825は、16音源設定でき、音源毎に共通項である3要素(BO、LFO、ALG)と、4オペレータ17要素(SR、XOF、KSR、RR、DR、AR、SL、TL、KSL、DAM、EAM、DVB、EVB、MUL、DT、WS、FB)合計71項目のパラメータがあります。
  • 各パラメータの概要は次のサイトを参照してください。
  • https://fukuno.jig.jp/1867
  • 音色編集は奥が深く、最初は何から手をつけたらよいか分かりません。YMF825は、4オペレータですが、最初は直列接続の2オペレータ(YMF825では、ALG=2)にして初段(モジュレータ)側の出力を0(YMF825では、TL=63)にし後段(キャリア)側のみ有効にして始めると分かりやすいようです。
  • 経験的に音量への影響度の大きなパラメータは、TL、SL、SR、DRなどです。
  • 音色は、倍音の構成で決まってきます。倍音への影響度の大きなパラメータは、MUL、WS、KSLなどです。このあたりが音色設定のキモでしょうね。
  • 今回、ワイングラスの音色を真似るために利用したのは、初期設定で「TENOR_SAX」をベースに編集しました。

編集前と編集後のパラメータの一覧

編集前と編集後のパラメータの一覧は次のようになります。
編集前と編集後のパラメータの一覧

「Edt」または「e1」~「e4」の項目が編集後、「Org」または「o1」~「o4」の項目が編集前の初期状態の値です。「Max」または「Mx」と書かれているのは、そのパラメータに設定できる最大値です。最小値は、どのパラメータも0です。
(編集画面上でも、右側の方に編集前の初期値と設定できる最大値の項目があります)

余談

この作品は、「手持ちのワイングラスの音色を楽器にしてMIDI演奏する」をテーマとして紹介しましたが、回路とプログラムは、色々な音出しの実験を兼ねています。
本テーマの以外に次のようなことができます。(ご参考)
MIDI-OUTにポケットミク(またはeVY1)を接続してミクの音声を外してMIDI演奏(16音源16和音、GM対応)、ミクの音声で演奏、ミクの音声で百人一首読み上げ、AquesTalkPicoを使った音声出力、DFPlayerを利用したSDカードのmp3ファイルの再生などです。
(ポケットミク、または、eVY1は、USBインターフェースなので、MIDI-OUTに接続するにはUSB MIDIコンバーターが必要です)

プログラム

  • プログラムはすべてMicroPythonですが、PS/2キーボードとPS/2マウスのドライバの一部はPicoのPIO(機械語)を利用しています。
  • MicroPythonコードの一部が、PIO(機械語)対応になっています。

PicoMainプログラム

PicoM_main_py.py

PicoM_main_py.py

############################################################################# # PicoM_main_py.py Programed by Hyodo # PicoMainに「main.py」として保存するか、Thonnyからこのプログラムを実行する。 # # ps2KeyBoard と PicoMain との接続 ps2KeyBoard # <<< Pico >>> レベル変換 USB-A USB-A # (メス) (オス) # gpio-No (pin) 3V 5V GND (4) | | # gpio-13 (17)---clock -- 青------------- LV2 HV2 --- D+ --(3)--| | # gpio-12 (16)---data -- 緑------------- LV1 HV1 --- D- --(2)--| | # +5V (1) |___| # (TopView) # # ps2Mouse と PicoMain との接続 ps2Mouse # <<< Pico >>> 74HC07 レベル変換 USB-A USB-A # (バッファ) (メス) (オス) # gpio-No (pin) 3V 5V GND (4) | | # gpio-27 (32)---clock -- 青------+------ LV4 HV4 --- D+ --(3)--| | # gpio-26 (31)---data -- 緑----- | -+--- LV3 HV3 --- D- --(2)--| | # | | +5V (1) |___| # gpio-21 (27)----(13) ▷ (12)-----+ | (TopView) # gpio-20 (26)----(11) ▷ (10)--------+ # # # ST7789 との接続 # <<< Pico >>> <<< ST7789 >>> # gpio-No (pin) (pin) # gpio-6 ( 9) ----赤---- (7) RT # gpio-7 (10) ----青---- (8) DC # gpio-8 MISO (11) ----緑---- (5) SD # gpio-9 CE (12) ----黄---- (6) CS # gpio-10 CKL (14) ----白---- (3) CK # gpio-11 MOSI (15) ----紫---- (4) SI # 3.3V ----赤---- (1) V # GND ----白---- (2) G # # # RFID-RC522 との接続 # <<< Pico >>> <<< RFID-RC522 >>> # gpio-No (pin) (pin) # gpio-22 (29) ----紫---- (7) RST # 3.3V ----灰---- (8) 3V3 # gpio-19 MOSI (24) ----橙---- (3) MOSI # gpio-18 SKL (23) ----赤---- (2) SCK # GND ----青---- (6) GND # gpio-17 CE (21) ----茶---- (1) SDA-CE # gpio-16 MISO (20) ----黄---- (4) MISO # pull-up 10K No Connect--緑---- (5) IRQ # # # # PicoMain と PicoSub との接続 # <<< PicoMain >>> <<< PicoSub >>> # gpio-No (pin) (pin) # gpio-2 respT (4) --紫-- --黄-- (4) respT gpio-2 # X # gpio-3 respR (5) --黄-- --紫-- (5) respR gpio-3 # # gpio-4 TX (6) --緑-- --青-- (6) TX gpio-4 # X # gpio-5 RX (7) --青-- --緑-- (7) RX gpio-5 # # # MIDI-OUT側に、ポケットミク、または、eVY1 を接続する場合 # ポケット・ミクは、Channel 0 がミクの音声固定になっている。(GM音源指定を無視する) # ミクの音声ではなく楽器に割り当てて演奏するために、機能追加として # Channel 1(ミクの音声になる)を 別のチャンネル(Swapping Ch)に割り当てて # MIDI-OUTに出力する機能(Play MIDI)を用意した。 # Swapping Chは、MIDI信号が無音のチャンネルを指定する。 # 発音のあるチャンネルだとミクの音声に変換されて発音されるので注意。 # # Channel 1 ------------ --- Channel 1(ミクの音声になる) # X # Swapping Ch(無音) --- --- Swapping Ch(GM TONE指定) # # そのチャンネルの音源を GM TONE NO 1-128 で指定できるようにした。 # GM TONE NO 1-128 以外を指定すると、Grand Pianoにする。 # 但し、MIDI信号の中に「プログラムチェンジ」=音色変更があれば、それに従うので留意。 # ############################################################################# from machine import Timer, SPI, UART, Pin import time import utime import uos # from micropython import const # メモリ使用状況 micropython.mem_info() を表示させるため、すべてをインポート import micropython from mfrc522 import MFRC522 import ymf825py as ymf825 import midipy2 as midi import mojipy as moji import st7789py as st7789 from fonts import vga2_8x8 as font1 from fonts import vga1_16x32 as font2 from ps2mus import PS2MouseCommand from ps2mus import PS2MouseRecive from ps2kb3 import PS2KeyRecive ############################################################################### # 上記インポートファイルは、予め、Picoに書き込んでおくこと # Thonnyで 当該ファイルを開いて、Picoに書き込む。 # 直下に # ps2mus.py : PS2Mouse 接続用 # ps2kb3.py : PS2Keyboard 接続用 # mfrc522.py : RFID-RC522 接続用 # ymf825py.py : YMF825 接続用 # midipy2.py : MIDI 接続用 # mojipy.py : 文字処理用 # st7789py.py : ST7789 TFT用 # <<< fonts フォルダ下に >>> # vga1_16x32.py: ST7789用フォント1 # vga2_8x8.py : ST7789用フォント2 Non Use ############################################################################### # 20240813 百人一首用に設定 HectOne = [ # address, VoiceRome, OctKey, Velocity, onTime, offTime # adr, voi, key, vel, ont, oft # Title ['Ad', 'Voi', 'Key', 'Vel', 'oN', 'oF'], ['A1', 'a', '4C', 90, 3, 0], ['A2', 'ma', '4D', 90, 5, 0], ['A3', 'no', '4D', 90, 5, 0], ['A4', 'ha', '4D', 90, 5, 0], ['A5', 'ra', '4D', 90, 6, 3], # ['B1', 'hu', '4D', 90, 3, 0], ['B2', 'ri', '4F', 90, 5, 0], ['B3', 'sa', '4F', 90, 5, 0], ['B4', 'ke', '4F', 90, 5, 0], ['B5', 'mi', '4D', 90, 5, 0], ['B6', 're', '4D', 90, 5, 0], ['B7', 'ba', '4C', 90, 6, 3], # ['C1', 'ka', '4D', 90, 4, 0], ['C2', 'su', '4D', 90, 5, 0], ['C3', 'ga', '3A', 90, 5, 0], ['C4', 'na', '3A', 90, 5, 0], ['C5', 'ru', '3A', 90, 6, 3], # ['D1', 'mi', '4D', 90, 3, 0], ['D2', 'ka', '4F', 90, 5, 0], ['D3', 'sa', '4F', 90, 5, 0], ['D4', 'no', '4F', 90, 5, 0], ['D5', 'ya', '4F', 90, 5, 0], ['D6', 'ma', '4F', 90, 5, 0], ['D7', 'ni', '4F', 90, 6, 3], # ['E1', 'i', '4G', 90, 3, 0], ['E2', 'de', '4G', 90, 5, 0], ['E3', 'si', '4G', 90, 5, 0], ['E4', 'tu', '4G', 90, 5, 0], ['E5', 'ki', '4G', 90, 5, 0], ['E6', 'ka', '4F', 90, 5, 0], ['E7', 'mo', '4G', 90, 9, 0], ] HectOneVoice = [ # 01 tenchitennno- ['akinotano','karihonoiono', 'tomawoarami', 'wagakoromodeha', 'tuyuninuretutu'], # 02 jito-tennno- ['harusugite', 'natukinikerasi', 'sirotaeno', 'koromohosutyo', 'amanokaguyama'], # 03 kakinomotonmo hitomaro ['asibikino', 'yamadorinoono', 'sidariono', 'naganagasiyowo', 'hitorikamonenn'], # 04 yamabeno akahito ['tagonoura', 'utidetemireba', 'sirotaeno', 'fujinotakaneni', 'yukihafuritutu'], # 05 sarumaru dayu- ['okuyamani', 'momijikakiwake', 'nakusikano', 'koekikutokizo', 'akihakanasiki'], # 06 tyuunagionn nakamoti ['kasasagino', 'wataseruhasini', 'okusimono', 'sirokiwomireba', 'yozofukenikeru'], # 07 ['amanohara', 'furisakemireba', 'kasuganaru', 'mikasanoyamani', 'idesitukikamo'], # 08 ['', '', '', '', ''], # 09 ['', '', '', '', ''], # 10 ['', '', '', '', ''], # 11 ['', '', '', '', ''], # 12 ['', '', '', '', ''], # 13 ['', '', '', '', ''], # 14 ['', '', '', '', ''], # 15 ['', '', '', '', ''], ] ST7789_COL_SIZE = const(15) # st7789 Col Size 20240714 Adj 16->15 TONE_NAME_SIZE = const(15) # Tone Name Size CURSOL_MAX = const(15) # = Tone Name Size DOT_R = const(32) # charactor Dot size High = Row DOT_C = const(16) # charactor Dot size Width = colum RFID_OK = const(0) uartRespT = Pin(2, Pin.OUT) # Low TRUE uartRespT.value(1) uartRespR = Pin(3, Pin.IN) # Low TRUE # ST7789 SPI(1) default pins spi1_sck = const(10) spi1_mosi = const(11) spi1_miso = const(8) # not use # 20240418 Change # st7789_res = const(12) # st7789_dc = const(13) st7789_res = const(6) st7789_dc = const(7) st7789_cs = const(9) disp_width = const(240) disp_height = const(240) CENTER_Y = int(disp_width/2) CENTER_X = int(disp_height/2) # 20240723 Copy from ymf825py.py ここから # 配列の上限 ROW_SIZE = const(10) # 行の最大 COL_SIZE = const(10) # 列の最大 # VOICE COMMON PARAMETER ROW s Number COM = const(0) BO = const(1) LFO = const(2) ALG = const(3) VOICE_COM_COL_NUM = const(4) # VOICE COMMON PARAMETER COLUMON s Number CM0 = const(0) # Oparator Name CM1 = const(1) # Edit CM2 = const(2) # Orig CM3 = const(3) # Max VOICE_COM_ROW_NUM = const(4) # 20240326 Change # VOICE OPTIONAL PARAMETER ROW s Number OPT = const(0) SR = const(1) XOF = const(2) KSR = const(3) RR = const(4) DR = const(5) AR = const(6) SL = const(7) TL = const(8) KSL = const(9) DAM = const(10) EAM = const(11) DVB = const(12) EVB = const(13) MUL = const(14) DT = const(15) WS = const(16) FB = const(17) VOICE_OP_ROW_NUM = const(18) # VOICE OPTIONAL PARAMETER COLUMN s Number OP0 = const(0) # Oparator Name OP1 = const(1) # Edit 1 OP2 = const(2) # Edit 2 OP3 = const(3) # Edit 3 OP4 = const(4) # Edit 4 OP5 = const(5) # Orig 1 OP6 = const(6) # Orig 2 OP7 = const(7) # Orig 3 OP8 = const(8) # Orig 4 OP9 = const(9) # Max VOICE_OP_COL_NUM = const(10) # 20240723 Copy from ymf825py.py ここまで # st7789 表示枠の大きさ DSP_ROW_SIZE = const(5) # 表示行の大きさ DSP_COL_SIZE = const(4) # 表示列の大きさ TONE_MENU_COL_NUM = const(2) # number of tones menu TONE_MENU_ROW_NUM = const(17) # number of tones menu TONE_TABLE_COL_NUM = const(2) # number of tones menu TONE_TABLE_ROW_NUM = const(17) # number of tones menu FUNCTION_COL_NUM = const(2) # Fxx FunctionName FUNCTION_ROW_NUM = const(13) # Index + F1 ~ F12 FunctionName = [None] * (FUNCTION_ROW_NUM) FunctionName[0] = ' Function' # Title FunctionName[1] = 'Edit Tone' # F1 FunctionName[2] = 'Read RFID' # F2 FunctionName[3] = 'Write RFID' # F3 FunctionName[4] = 'Play MIDI' # F4 FunctionName[5] = 'DFP(mp3)Test' # F5 FunctionName[6] = 'Edt Tn Table' # F6 FunctionName[7] = 'Edt Mk Voice' # F7 FunctionName[8] = '' # F8 FunctionName[9] = '' # F9 FunctionName[10] = '' # F10 FunctionName[11] = '' # F11 FunctionName[12] = '' # F12 # FunctionTable の生成 FunctionTable = [None] * FUNCTION_ROW_NUM # 13 for i in range(FUNCTION_ROW_NUM): FunctionTable[i] = [None] * FUNCTION_COL_NUM # 2 for i in range(FUNCTION_ROW_NUM): FunctionTable[i][0] = i FunctionTable[i][1] = FunctionName[i] SubMenu = [None] *7 for i in range(7): SubMenu[i] = '' AlarmMsg = [None] *7 for i in range(7): AlarmMsg[i] = '' InputMsg = [None] *7 for i in range(7): InputMsg[i] = '' # clorFont, Msg EditMsg = [None] *7 for i in range(7): EditMsg[i] = [None] *2 for i in range(7): EditMsg[i][0] = st7789.WHITE EditMsg[i][1] = '' # PS2 Keyboard PIO Pin設定 # 5V ⇔ 3.3V レベル変換回路入れること(Key側:5V、Pico側:3V) KB_DATA = const(12) # PS2 keyboard data GPIO-12 KB_CLOCK = const(13) # PS2 keyboard clock GPIO-13 KB_PIO_FRQ = const(1000000) # PS2 Keyboard 設定 kb_freq = KB_PIO_FRQ kb_data = Pin(KB_DATA, Pin.IN, Pin.PULL_UP) # PS2 Keyboard Data kb_clock = Pin(KB_CLOCK, Pin.IN, Pin.PULL_UP) # PS2 Keyboard Clock # PS2 Mouse PIO Pin設定 # 5V ⇔ 3.3V レベル変換回路入れること(Key側:5V、Pico側:3V) RCV_DATA = const(26) # PS2 mouse data 入力 GPIO-26 RCV_CLOCK = const(27) # PS2 mouse clock 入力 GPIO-27 SEND_DATA = const(20) # 出力 GPIO-20 バッファを介して GPIO-26 に接続 SEND_CLOCK = const(21) # 出力 GPIO-21 バッファを介して GPIO-27 に接続 # PIO PS/2 Mouse 動作周波数 ps2mus.py PS2KeyRecive PIO 動作周波数:1MHz # PIO_KB_FRQ = const(1000000) MUS_PIO_FRQ = const(1000000) # PS2 Mouse 設定 pio_freq = MUS_PIO_FRQ rcv_data = Pin(RCV_DATA, Pin.IN, Pin.PULL_UP) # PS2 Mouse Recive Data rcv_clock = Pin(RCV_CLOCK, Pin.IN, Pin.PULL_UP) # PS2 Mouse Recive Clock send_data = Pin(SEND_DATA, Pin.OUT) # PS2 Mouse Send Data send_clock = Pin(SEND_CLOCK, Pin.OUT) # PS2 Mouse Send Clock # PS2MouseCommand ドライバ および StateMachine の設定と起動 ---------------------- ps2mus_cmd = PS2MouseCommand(pio_freq, send_data, send_clock, rcv_data, rcv_clock) ################################################################################## # PS2MouseCommand ドライバの起動・・・ここでは、まだ、行わない # ps2mus_cmd.set_active() ############################################################ # PS2MouseRecive ドライバ および StateMachine の設定と起動 ---------------------- ps2mus_rcv = PS2MouseRecive(pio_freq, rcv_data, rcv_clock) ################################################################################## # PS2MouseRecive ドライバの起動とタイマー設定・・・ここでは、まだ、行わない # ps2mus_rcv.set_active() ############################################################ # 1Frame Time の設定 # マウスを連続して動かすと、12ms周期くらいでデータ送信してくる。 # 何もない時間が 4ms続けば区切りとして、インターラプトを発生させる。 # 4ms:(1MHz*1ms*1000us)/PIOクロック数 2 # time_out = int((MUS_PIO_FRQ * 4) / (1000 * 2)) ############################################################ # 時分割処理のためのタイマー # global変数定義。関数で利用するグローバル変数は、関数の定義前に書くのが無難。 uart_timer = 0 # UART RTS CTS Timer # 10ms 毎に割り込みが発生する なぜか引数 (t) は必須である。 def every10ms(t): ########################################################## global uart_timer ########################################################## if uart_timer < 10: uart_timer = 0 else: uart_timer -= 10 # 20240416 10ms # 10ms 毎に every10ms を呼び出す呪文。これ以前に callback関数を定義のこと。 # Timer(period=10, mode=Timer.PERIODIC, callback=every10ms) # mode=Timer.PERIODIC は、指定しなくても大丈夫のようだ。 # xxx Timer(period=10, callback=every10ms) でも動作するが # MicroPython説明サイトに従ったタイマーの初期化 tim10 = Timer(period=10, mode=Timer.PERIODIC, callback=every10ms) tim10.init(period=10, callback=every10ms) # 97ms 毎に割り込みが発生する なぜか引数 (t) は必須である。 def every97ms(t): if midi.stop == 0 and midi.end == 0: # PS2Keyboard 「Esc」キーのみ受け付ける # ファンクション・キー入力なら if ps2key.rcv_size() > 1: ps2key_data = ps2key.get_code() # .lower 文字の一致を取りやすくするため、shift low で照合 if ps2key_data.lower() == "Esc".lower(): msg_f = "Esc" print(msg_f) midi.stop = 1 # midi_play() に停止を送信 # MicroPython説明サイトに従ったタイマーの初期化 tim97 = Timer(period=97, mode=Timer.PERIODIC, callback=every97ms) tim97.init(period=97, callback=every97ms) timer997sw = 0 backColor = st7789.RED # 編集箇所をわかりやすくするため、背景色を白と黄色にする仕掛け # 997ms 毎に割り込みが発生する なぜか引数 (t) は必須である。 def every997ms(t): global timer997sw, backColor if timer997sw == 0: backColor = st7789.WHITE timer997sw = 1 else: backColor = st7789.YELLOW timer997sw = 0 # MicroPython説明サイトに従ったタイマーの初期化 tim997 = Timer(period=997, mode=Timer.PERIODIC, callback=every997ms) tim997.init(period=997, callback=every997ms) ####################################################################### # HandShake Send RTS & Recive CTS Cross Connection # PicoMain と PicoSub との接続 # <<< PicoMain >>> <<< PicoSub >>> # gpio-2 respT (4) --紫-- --黄-- (4) respT gpio-2 # X # gpio-3 respR (5) --黄-- --紫-- (5) respR gpio-3 # def waitCTS(rts, time, msg, cnt): global uart_timer if rts == 0: # Send RTS On uartRespT.value(0) uart_timer = time # Wait CTS On while uartRespR.value() == 1: # On == Low になるのを待つ if uart_timer == 0: print(msg) uart_timer = time if cnt > 0: cnt -= 1 else: break # Pico Sub と UART 通信できない utime.sleep_ms(10) else: # Send RTS Off uartRespT.value(1) uart_timer = time # Wait CTS Off while uartRespR.value() == 0: # Off == High になるのを待つ if uart_timer == 0: print(msg) uart_timer = time if cnt > 0: cnt -= 1 else: break # Pico Sub と UART 通信できない utime.sleep_ms(10) return cnt ####################################################################### def setToneDataPicoSub(tone_num): print('setToneDataPicoSub(tone_num)') # Sub Pico へ送信要求を送出 print('RTS On to Sub_Pico') cnt = 3 cnt = waitCTS(0, 1000, 'Wait Sub_Pico CTS On', cnt) if cnt > 0: # print('CTS On(0) from Sub_Pico') # ToneEdit をシリアル出力 # 20240727 Comment Out # ymf825.serialOutToneData(tone_num) # ToneVoice をシリアル出力 ymf825.copyTone2_Voice(tone_num) ymf825.serialOutVoiceData(tone_num) # Tone_Com_Edit, Tone_Op_Edit を ASCII2文字の send_buffに変換 ymf825.takeOutToneData(tone_num) # send_buff を UART で送信 ymf825.uartOutToneData() utime.sleep_ms(10) # Sub Pico へ送信要求の解除を送出 print('RTS Off to Sub_Pico') cnt = 3 cnt = waitCTS(1, 1000, 'Wait Sub_Pico CTS Off', cnt) if cnt > 0: print('CTS Off(1) from Sub_Pico') print('ymf825.uartOutToneData() was performed') else: print() print('CTS Off(1) signal not comming from Sub-Pico') print('Check the connection with Sub-Pico') print() else: print() print('CTS On(0) signal not comming from Sub-Pico') print('ymf825.uartOutToneData() was not performed !!!') print('Check the connection with Sub-Pico') print() # Send RTS Off uartRespT.value(1) ####################################################################### def setToneTablePicoSub(): print('setToneTablePicoSub()') # Sub Pico へ送信要求を送出 print('RTS On to Sub_Pico') cnt = 3 cnt = waitCTS(0, 1000, 'Wait Sub_Pico CTS On', cnt) if cnt > 0: # print('CTS On(0) from Sub_Pico') # ToneTable を ASCII2文字の send_buffに変換 ymf825.takeOutToneTable() # ToneTableリストを UART に転送 ymf825.uartOutToneTable() utime.sleep_ms(10) # Sub Pico へ送信要求の解除を送出 print('RTS Off to Sub_Pico') cnt = 3 cnt = waitCTS(1, 1000, 'Wait Sub_Pico CTS Off', cnt) if cnt > 0: print('CTS Off(1) from Sub_Pico') print('ymf825.uartOutToneTable() was performed') else: print() print('CTS Off(1) signal not comming from Sub-Pico') print('Check the connection with Sub-Pico') print() else: print() print('CTS On(0) signal not comming from Sub-Pico') print('ymf825.uartOutToneTable() was not performed !!!') print('Check the connection with Sub-Pico') print() # Send RTS Off uartRespT.value(1) ####################################################################### # PicoSubに接続された PrintDFP用の mp3再生メッセージ # 先頭に %DFP コマンド 末尾に \n改行 def setDFP_PicoSub(command): print('setDFP_PicoSub(buffDFP)') # Sub Pico へ送信要求を送出 print('RTS On to Sub_Pico') cnt = 3 cnt = waitCTS(0, 1000, 'Wait Sub_Pico CTS On', cnt) if cnt > 0: # print('CTS On(0) from Sub_Pico') # commandを UARTで PicoSubに送出 commandDFP = '%DFP '+ command + '\n' print('PrintDFP Command = ', commandDFP) print(f'ymf825.uart_write_multi({commandDFP})') ####################################################### ymf825.uart_write_multi(commandDFP) ####################################################### utime.sleep_ms(10) # Sub Pico へ送信要求の解除を送出 print('RTS Off to Sub_Pico') cnt = 3 cnt = waitCTS(1, 1000, 'Wait Sub_Pico CTS Off', cnt) if cnt > 0: print('CTS Off(1) from Sub_Pico') print('ymf825.uartOutToneTable() was performed') else: print() print('CTS Off(1) signal not comming from Sub-Pico') print('Check the connection with Sub-Pico') print() else: print() print('CTS On(0) signal not comming from Sub-Pico') print('ymf825.uartOutToneTable() was not performed !!!') print('Check the connection with Sub-Pico') print() # Send RTS Off uartRespT.value(1) ####################################################################### # line_n(0-6)で、画面上の編集行を指定する。 # new_name = editName(EditMsg, line_n, old_name) def editName(EditMsg, line_n, old_name): line_n = line_n % 7 # 表示枠外に出ないよう規制 # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(old_name, TONE_NAME_SIZE) temp_name = new_name # ST7789 画面をクリア display.fill(st7789.BLACK) # 編集画面を表示、文字色指定あり col_n = 0 for i in range(7): if len(EditMsg[i][1]) > 0: # DOT_C = 16 : col_n , DOT_R = 32 : line_n display.text(font2, EditMsg[i][1],\ DOT_C * col_n, DOT_R * i, EditMsg[i][0]) # 編集行全体を表示 col_n = 0 # DOT_C = 16 : col_n , DOT_R = 32 : line_n display.text(font2, moji.strFixL_n(new_name, ST7789_COL_SIZE),\ DOT_C * col_n, DOT_R * line_n, st7789.WHITE) # 初期表示・カーソル now_cursol = 0 # now_cursol 文字 表示 cursol_data = new_name[now_cursol] col_n = now_cursol # DOT_C = 16 : col_n , DOT_R = 32 : line_n display.text(font2, cursol_data,\ DOT_C * col_n, DOT_R * line_n, st7789.RED, st7789.WHITE) old_cursol = now_cursol # マウス data_cnt_old = 999 btn_l = 0 btn_r = 0 btn_c = 0 cnt_x = 0 cnt_y = 0 cnt_s = 0 # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) data_cnt_old = 999 # マウスの左クリックが押されていない状態 busy_l = 0 # マウスの右クリックが押されていない状態 busy_r = 0 while True: # 文字入力 または カーソル移動があれば表示する # Enter または Esc で終了 if temp_name != new_name or old_cursol != now_cursol: # 編集行全体を表示 col_n = 0 display.text(font2, moji.strFixL_n(new_name, ST7789_COL_SIZE),\ DOT_C * col_n, DOT_R * line_n, st7789.WHITE) # now_cursol 文字 表示 cursol_data = new_name[now_cursol] col_n = now_cursol display.text(font2, cursol_data,\ DOT_C * col_n, DOT_R * line_n, st7789.RED, st7789.WHITE) temp_name = new_name old_cursol = now_cursol # PS2Keyboard # 1文字・記号入力なら if ps2key.rcv_size() == 1: ps2key_data = ps2key.get_code() # if now_cursol >= 0 and now_cursol < TONE_NAME_SIZE: # 0-14 ############################################################ # 挿入モードで編集(表示) temp_name = new_name[:] # copy new_name = temp_name[: now_cursol] + ps2key_data + temp_name[now_cursol:TONE_NAME_SIZE -1] ############################################################ old_cursol = now_cursol now_cursol += 1 # ファンクション・キー入力なら elif ps2key.rcv_size() > 1: ps2key_data = ps2key.get_code() # # .lower 文字の一致を取りやすくするため、shift low で照合 if ps2key_data.lower() == "Esc".lower(): msg_f = "Esc" print(msg_f) new_name = moji.strFixL_n(old_name, TONE_NAME_SIZE) break elif ps2key_data.lower() == "Enter".lower(): msg_f = "Enter" print(msg_f) break elif ps2key_data.lower() == "BS".lower(): msg_f = "BS" print(msg_f) if now_cursol > 0 and now_cursol < TONE_NAME_SIZE: # 1-14 temp_name = new_name[:] # copy new_name = temp_name[: now_cursol -1] + temp_name[now_cursol:TONE_NAME_SIZE] + ' ' now_cursol -= 1 elif ps2key_data.lower() == "Del".lower(): msg_f = "Del" print(msg_f) if now_cursol >= 0 and now_cursol < TONE_NAME_SIZE: # 0-14 temp_name = new_name[:] # copy new_name = temp_name[: now_cursol] + temp_name[now_cursol +1:TONE_NAME_SIZE] + ' ' elif ps2key_data.lower() == "Arw_L".lower(): # ← msg_f = "ArwL" print(msg_f) if now_cursol > 0 and now_cursol < (TONE_NAME_SIZE): # 1-14 old_cursol = now_cursol now_cursol -= 1 elif ps2key_data.lower() == "Arw_R".lower(): # → msg_f = "ArwR" print(msg_f) if now_cursol >= 0 and now_cursol < (TONE_NAME_SIZE -1): # 0-13 old_cursol = now_cursol now_cursol += 1 elif ps2key_data.lower() == "Space".lower(): # ' ' msg_f = "Space" print(msg_f) if now_cursol >= 0 and now_cursol < TONE_NAME_SIZE: # 0-14 temp_name = new_name[:] # copy new_name = temp_name[: now_cursol] + ' ' + temp_name[now_cursol:TONE_NAME_SIZE -1] old_cursol = now_cursol now_cursol += 1 elif ps2key_data == "F1": msg_f = "F1" print(msg_f) # マウスからボタンクリックの情報、座標情報を取得する (data_cnt, btn_l, btn_r, btn_c, cnt_x, cnt_y, cnt_s) = ps2mus_rcv.get_data() # PS2 Mouse に入力があるなら if data_cnt_old != data_cnt: data_cnt_old = data_cnt # 左クリック・ボタンが新たに押されたなら if busy_l == 0 and btn_l ==1: busy_l = 1 # 左クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_l == 1 and btn_l ==0: busy_l = 0 msg_f = "Mouse L cricked" print(msg_f) break # 右クリック・ボタンが新たに押されたなら elif busy_r == 0 and btn_r ==1: busy_r = 1 # 右クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_r == 1 and btn_r ==0: busy_r = 0 msg_f = "Mouse R cricked" print(msg_f) break # break でここに来る return new_name ####################################################################### # PrintMtxLED へのコマンド編集と UART送信 # 上段表示は、左から右へ、キャラクター(charU)が移動 # 下段表示は、右から左へ、キャラクター(charL)が移動 # 移動間隔時間を、timeで指定する。 def printMtxLedChr(charU, charL, time): # PrintMtxLED 画面消去 ymf825.uart_write_multi('%IJ.CLS\n') # % I J sp mtxled_header = [37, 73, 74, 32, ] mtxled_col = 4 mtxled_upper = [32] *mtxled_col mtxled_lower = [32] *mtxled_col mtxled_LF = [10] # LF \n for i in range(mtxled_col): mtxled_upper[i] = charU mtxled_lower[mtxled_col -i -1] = charL mtxledB = bytes(mtxled_header + mtxled_upper + mtxled_lower + mtxled_LF) # PrintMtxLED へシリアル送信 ymf825.uart_write_multi(mtxledB) # データを元に戻す mtxled_upper[i] = 32 mtxled_lower[mtxled_col -i -1] = 32 utime.sleep_ms(time) # PrintMtxLED 画面消去 # ymf825.uart_write_multi('%IJ.CLS\n') ####################################################################### # PrintMtxLED へのコマンド編集と UART送信 # 上段表示は、左から右へ、キャラクター(msgU)が移動 # 下段表示は、右から左へ、キャラクター(msgL)が移動 # 移動間隔時間を、timeで指定する。 def printMtxLedMsg(msgU, msgL, time): # PrintMtxLED 画面消去 ymf825.uart_write_multi('%IJ.CLS\n') # % I J sp mtxled_header = [37, 73, 74, 32, ] mtxled_col = 4 mtxled_upper = [32] *mtxled_col mtxled_lower = [32] *mtxled_col mtxled_LF = [10] # LF \n for i in range(mtxled_col): mtxled_upper[i] = msgU[i] mtxled_lower[i] = msgL[i] mtxledB = bytes(mtxled_header + mtxled_upper + mtxled_lower + mtxled_LF) # PrintMtxLED へシリアル送信 ymf825.uart_write_multi(mtxledB) utime.sleep_ms(time) # PrintMtxLED 画面消去 # utime.sleep_ms(1000) # ymf825.uart_write_multi('%IJ.CLS\n') ####################################################################### # PrintDFP へのコマンド編集と UART送信 # line_n(0-6)で、画面上の編集行を指定する。 def printDFP(EditMsg, line_n, old_name): line_n = line_n % 7 # 表示枠外に出ないよう規制 # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(old_name, TONE_NAME_SIZE) temp_name = new_name # ST7789 画面をクリア display.fill(st7789.BLACK) # 編集画面を表示、文字色指定あり col_n = 0 for i in range(7): if len(EditMsg[i][1]) > 0: # DOT_C = 16 : col_n , DOT_R = 32 : line_n display.text(font2, EditMsg[i][1],\ DOT_C * col_n, DOT_R * i, EditMsg[i][0]) # 編集行全体を表示 col_n = 0 # DOT_C = 16 : col_n , DOT_R = 32 : line_n display.text(font2, moji.strFixL_n(new_name, ST7789_COL_SIZE),\ DOT_C * col_n, DOT_R * line_n, st7789.WHITE) # 初期表示・カーソル now_cursol = 0 # now_cursol 文字 表示 cursol_data = new_name[now_cursol] col_n = now_cursol # DOT_C = 16 : col_n , DOT_R = 32 : line_n display.text(font2, cursol_data,\ DOT_C * col_n, DOT_R * line_n, st7789.RED, st7789.WHITE) old_cursol = now_cursol # マウス data_cnt_old = 999 btn_l = 0 btn_r = 0 btn_c = 0 cnt_x = 0 cnt_y = 0 cnt_s = 0 # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) data_cnt_old = 999 # マウスの左クリックが押されていない状態 busy_l = 0 # マウスの右クリックが押されていない状態 busy_r = 0 while True: # 文字入力 または カーソル移動があれば表示する # Enter または Esc で終了 if temp_name != new_name or old_cursol != now_cursol: # 編集行全体を表示 col_n = 0 display.text(font2, moji.strFixL_n(new_name, ST7789_COL_SIZE),\ DOT_C * col_n, DOT_R * line_n, st7789.WHITE) # now_cursol 文字 表示 cursol_data = new_name[now_cursol] col_n = now_cursol display.text(font2, cursol_data,\ DOT_C * col_n, DOT_R * line_n, st7789.RED, st7789.WHITE) temp_name = new_name old_cursol = now_cursol # PS2Keyboard # 1文字・記号入力なら if ps2key.rcv_size() == 1: ps2key_data = ps2key.get_code() # if now_cursol >= 0 and now_cursol < TONE_NAME_SIZE: # 0-14 ############################################################ # 挿入モードで編集(表示) temp_name = new_name[:] # copy new_name = temp_name[: now_cursol] + ps2key_data + temp_name[now_cursol:TONE_NAME_SIZE -1] ############################################################ old_cursol = now_cursol now_cursol += 1 # ファンクション・キー入力なら elif ps2key.rcv_size() > 1: ps2key_data = ps2key.get_code() # # .lower 文字の一致を取りやすくするため、shift low で照合 if ps2key_data.lower() == "Esc".lower(): msg_f = "Esc" print(msg_f) new_name = moji.strFixL_n(old_name, TONE_NAME_SIZE) break elif ps2key_data.lower() == "Enter".lower(): msg_f = "Enter" print(msg_f) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED charU = 0xEE charL = 0xEE time = 400 ####################################################### printMtxLedChr(charU, charL, time) ####################################################### utime.sleep_ms(4000) ymf825.uart_write_multi('%IJ.CLS\n') elif ps2key_data.lower() == "BS".lower(): msg_f = "BS" print(msg_f) if now_cursol > 0 and now_cursol < TONE_NAME_SIZE: # 1-14 temp_name = new_name[:] # copy new_name = temp_name[: now_cursol -1] + temp_name[now_cursol:TONE_NAME_SIZE] + ' ' now_cursol -= 1 elif ps2key_data.lower() == "Del".lower(): msg_f = "Del" print(msg_f) if now_cursol >= 0 and now_cursol < TONE_NAME_SIZE: # 0-14 temp_name = new_name[:] # copy new_name = temp_name[: now_cursol] + temp_name[now_cursol +1:TONE_NAME_SIZE] + ' ' elif ps2key_data.lower() == "Arw_L".lower(): # ← msg_f = "ArwL" print(msg_f) if now_cursol > 0 and now_cursol < (TONE_NAME_SIZE): # 1-14 old_cursol = now_cursol now_cursol -= 1 elif ps2key_data.lower() == "Arw_R".lower(): # → msg_f = "ArwR" print(msg_f) if now_cursol >= 0 and now_cursol < (TONE_NAME_SIZE -1): # 0-13 old_cursol = now_cursol now_cursol += 1 elif ps2key_data.lower() == "Space".lower(): # ' ' msg_f = "Space" print(msg_f) if now_cursol >= 0 and now_cursol < TONE_NAME_SIZE: # 0-14 temp_name = new_name[:] # copy new_name = temp_name[: now_cursol] + ' ' + temp_name[now_cursol:TONE_NAME_SIZE -1] old_cursol = now_cursol now_cursol += 1 elif ps2key_data == "F1": msg_f = "F1" print(msg_f) temp_name = new_name[:] new_name = 'PLAY 1 59' old_cursol = now_cursol now_cursol = len(new_name) # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(new_name, TONE_NAME_SIZE) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED # Hart msgU = [0xE5, 0xE5, 0xE5, 0xE5] # Hart msgL = [0xE5, 0xE5, 0xE5, 0xE5] time = 10 ####################################################### printMtxLedMsg(msgU, msgL, time) ####################################################### utime.sleep_ms(4000) ymf825.uart_write_multi('%IJ.CLS\n') elif ps2key_data == "F2": msg_f = "F2" print(msg_f) new_name = 'PLAY 1 55' old_cursol = now_cursol now_cursol = len(new_name) # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(new_name, TONE_NAME_SIZE) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED # Clover msgU = [0xE6, 0xE6, 0xE6, 0xE6] # Clover msgL = [0xE6, 0xE6, 0xE6, 0xE6] time = 10 ####################################################### printMtxLedMsg(msgU, msgL, time) ####################################################### utime.sleep_ms(4000) ymf825.uart_write_multi('%IJ.CLS\n') elif ps2key_data == "F3": msg_f = "F3" print(msg_f) new_name = 'PLAY 1 3' old_cursol = now_cursol now_cursol = len(new_name) # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(new_name, TONE_NAME_SIZE) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED # Spade msgU = [0xE4, 0xE4, 0xE4, 0xE4] # Spade msgL = [0xE4, 0xE4, 0xE4, 0xE4] time = 10 ####################################################### printMtxLedMsg(msgU, msgL, time) ####################################################### utime.sleep_ms(8000) ymf825.uart_write_multi('%IJ.CLS\n') elif ps2key_data == "F4": msg_f = "F4" print(msg_f) new_name = 'PLAY 1 4' old_cursol = now_cursol now_cursol = len(new_name) # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(new_name, TONE_NAME_SIZE) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED # Diamonds msgU = [0xE7, 0xE7, 0xE7, 0xE7] # Diamonds msgL = [0xE7, 0xE7, 0xE7, 0xE7] time = 10 ####################################################### printMtxLedMsg(msgU, msgL, time) ####################################################### utime.sleep_ms(8000) ymf825.uart_write_multi('%IJ.CLS\n') elif ps2key_data == "F5": msg_f = "F5" print(msg_f) new_name = 'PLAY 1 45' old_cursol = now_cursol now_cursol = len(new_name) # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(new_name, TONE_NAME_SIZE) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED charU = 0xED charL = 0xED time = 400 ####################################################### printMtxLedChr(charU, charL, time) ####################################################### utime.sleep_ms(6000) ymf825.uart_write_multi('%IJ.CLS\n') elif ps2key_data == "F6": msg_f = "F6" print(msg_f) new_name = 'PLAY 1 50' old_cursol = now_cursol now_cursol = len(new_name) # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(new_name, TONE_NAME_SIZE) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED charU = 0xFB charL = 0xFD time = 400 ####################################################### printMtxLedChr(charU, charL, time) ####################################################### utime.sleep_ms(6000) ymf825.uart_write_multi('%IJ.CLS\n') elif ps2key_data == "F7": msg_f = "F7" print(msg_f) new_name = 'PLAY 2 131' old_cursol = now_cursol now_cursol = len(new_name) # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(new_name, TONE_NAME_SIZE) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED # wa nn wa nn msgU = [0x20, 0xDC, 0xDD, 0x20] # msgL = [0x20, 0x20, 0x20, 0x20] time = 400 ####################################################### printMtxLedMsg(msgU, msgL, time) ####################################################### utime.sleep_ms(4000) ymf825.uart_write_multi('%IJ.CLS\n') elif ps2key_data == "F8": msg_f = "F8" print(msg_f) new_name = 'PLAY 2 141' old_cursol = now_cursol now_cursol = len(new_name) # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(new_name, TONE_NAME_SIZE) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED charU = 0xEC charL = 0xEC time = 400 ####################################################### printMtxLedChr(charU, charL, time) ####################################################### utime.sleep_ms(4000) ymf825.uart_write_multi('%IJ.CLS\n') elif ps2key_data == "F9": msg_f = "F9" print(msg_f) new_name = 'PLAY 2 231' old_cursol = now_cursol now_cursol = len(new_name) # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(new_name, TONE_NAME_SIZE) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED # ka ko msgU = [0xB6, 0xAF, 0xBA, 0x2D] # msgL = [0x20, 0x20, 0x20, 0x20] time = 10 ####################################################### printMtxLedMsg(msgU, msgL, time) ####################################################### utime.sleep_ms(4000) ymf825.uart_write_multi('%IJ.CLS\n') elif ps2key_data == "F10": msg_f = "F10" print(msg_f) new_name = 'PLAY 3 12' old_cursol = now_cursol now_cursol = len(new_name) # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(new_name, TONE_NAME_SIZE) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED # ji " a ne msgU = [0xBC, 0xDE, 0xAC, 0xC8] # yo ro si ku msgL = [0xD6, 0xDB, 0xBC, 0xB8] time = 200 ####################################################### printMtxLedMsg(msgU, msgL, time) ####################################################### utime.sleep_ms(4000) ymf825.uart_write_multi('%IJ.CLS\n') elif ps2key_data == "F11": msg_f = "F11" print(msg_f) new_name = 'PLAY 1 11' old_cursol = now_cursol now_cursol = len(new_name) # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(new_name, TONE_NAME_SIZE) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED charU = 0xEF charL = 0xEF time = 400 ####################################################### printMtxLedChr(charU, charL, time) ####################################################### utime.sleep_ms(4000) ymf825.uart_write_multi('%IJ.CLS\n') elif ps2key_data == "F12": msg_f = "F12" print(msg_f) new_name = 'PLAY 1 12' old_cursol = now_cursol now_cursol = len(new_name) # 末尾にスペースを入れて15桁にする new_name = moji.strFixL_n(new_name, TONE_NAME_SIZE) # PrintDFP command = new_name.strip() setDFP_PicoSub(command) utime.sleep_ms(100) # PrintMtxLED charU = 0xF1 charL = 0xF1 time = 400 ####################################################### printMtxLedChr(charU, charL, time) ####################################################### utime.sleep_ms(4000) ymf825.uart_write_multi('%IJ.CLS\n') # マウスからボタンクリックの情報、座標情報を取得する (data_cnt, btn_l, btn_r, btn_c, cnt_x, cnt_y, cnt_s) = ps2mus_rcv.get_data() # PS2 Mouse に入力があるなら if data_cnt_old != data_cnt: data_cnt_old = data_cnt # 左クリック・ボタンが新たに押されたなら if busy_l == 0 and btn_l ==1: busy_l = 1 # 左クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_l == 1 and btn_l ==0: busy_l = 0 msg_f = "Mouse L cricked" print(msg_f) # commandDFP = 'PRINT"%DFP '+ new_name.strip() + '"' commandDFP = '%DFP '+ new_name.strip() + '\n' print('PrintDFP Command = ', commandDFP) print(f'ymf825.uart_write_multi({commandDFP})') ####################################################### ymf825.uart_write_multi(commandDFP) ####################################################### # 右クリック・ボタンが新たに押されたなら elif busy_r == 0 and btn_r ==1: busy_r = 1 # 右クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_r == 1 and btn_r ==0: busy_r = 0 msg_f = "Mouse R cricked" print(msg_f) new_name = moji.strFixL_n(old_name, TONE_NAME_SIZE) break # break でここに来る return new_name ####################################################################### # 数文字入力。return値を、int(num)で、「数値」に変換できる。 # line_n(0-6)で、画面上の入力行を指定する。 # prompt は、入力行でのプロンプト(例 'ch ? ') # num_asc(文字列 ='000')は初期値、'000'を入力桁数のサイズとする。 # prompt の桁数と、num_asc の桁数合計が、st7789の桁数(15)を超えないこと。 def inputNum(InputMsg, line_n, prompt, num_asc): # line_n = line_n % 7 # 表示枠外に出ないよう規制 # prompt の桁数 try: if len(prompt) >= 1: prompt_size = len(prompt) else: prompt = '?' prompt_size = len(prompt) except: prompt = '?' prompt_size = len(prompt) # 念のため、文字列の先頭と末尾、および、文字間のスペースを取り除く num_asc = num_asc.replace(' ', '') # num_size = len(num_asc) の桁数内で編集する。最低でも1桁とする。 try: if len(num_asc) >= 1: num_size = len(num_asc) else: num_asc = '0' num_size = len(num_asc) except: num_asc = '0' num_size = len(num_asc) # 先頭にスペースを入れて num_size 桁にする old_num = moji.strFixR_n(str(int(num_asc)), num_size) new_num = old_num temp_num = new_num # ST7789 画面をクリア display.fill(st7789.BLACK) # InputMsg の表示 0行目〜6行目 for i in range(7): if len(InputMsg[i]) > 0: display.text(font2, moji.strFixL_n(InputMsg[i], ST7789_COL_SIZE), 0, \ DOT_R *i, st7789.WHITE) # マウス data_cnt_old = 999 btn_l = 0 btn_r = 0 btn_c = 0 cnt_x = 0 cnt_y = 0 cnt_s = 0 # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) data_cnt_old = 999 # マウスの左クリックが押されていない状態 busy_l = 0 # マウスの右クリックが押されていない状態 busy_r = 0 # 表示位置変数の設定 col_n = 0 cursol = prompt_size + num_size -1 # 初期表示・全体 # DOT_C = 16 : col_n , DOT_R = 32 : line_n dsp_data = prompt + moji.strFixR_n(str(int(new_num)), num_size) display.text(font2, moji.strFixL_n(dsp_data, ST7789_COL_SIZE),\ DOT_C * col_n, DOT_R * line_n, \ st7789.WHITE) # 初期表示・カーソル cursol_data = str(int(new_num) % 10) display.text(font2, cursol_data,\ DOT_C * cursol, DOT_R * line_n, \ st7789.RED, st7789.WHITE) while True: # 数字入力があれば表示する Enter または Esc で終了 if temp_num != new_num: col_n = 0 dsp_data = prompt + moji.strFixR_n(str(int(new_num)), num_size) display.text(font2, moji.strFixL_n(dsp_data, ST7789_COL_SIZE),\ DOT_C * col_n, DOT_R * line_n, \ st7789.WHITE) cursol_data = str(int(new_num) % 10) display.text(font2, cursol_data,\ DOT_C * cursol, DOT_R * line_n, \ st7789.RED, st7789.WHITE) temp_num = new_num[:] # PS2Keyboard # 1文字・記号入力なら if ps2key.rcv_size() == 1: ps2key_data = ps2key.get_code() if ps2key_data >= '0' and ps2key_data <= '9': ####################################################### # 右詰め表示で編集 temp_num = new_num[:] # copy new_num += ps2key_data new_num = moji.strFixR_n(str(int(new_num)), num_size) ####################################################### # ファンクション・キー入力なら elif ps2key.rcv_size() > 1: ps2key_data = ps2key.get_code() # .lower 文字の一致を取りやすくするため、shift low で照合 if ps2key_data.lower() == "Esc".lower(): msg_f = "Esc" print(msg_f) new_num = moji.strFixR_n(str(int(old_num)), num_size) break elif ps2key_data.lower() == "Enter".lower(): msg_f = "Enter" print(msg_f) break elif ps2key_data.lower() == "BS".lower(): msg_f = "BS" print(msg_f) temp_num = new_num[:] # copy new_num = str(int(int(new_num) / 10)) new_num = moji.strFixR_n(str(int(new_num)), num_size) elif ps2key_data.lower() == "Del".lower(): msg_f = "Del" print(msg_f) temp_num = new_num[:] # copy new_num = str(int(int(new_num) / 10)) new_num = moji.strFixR_n(str(int(new_num)), num_size) # マウスからボタンクリックの情報、座標情報を取得する (data_cnt, btn_l, btn_r, btn_c, cnt_x, cnt_y, cnt_s) = ps2mus_rcv.get_data() # PS2 Mouse に入力があるなら if data_cnt_old != data_cnt: data_cnt_old = data_cnt # 左クリック・ボタンが新たに押されたなら if busy_l == 0 and btn_l ==1: busy_l = 1 # 左クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_l == 1 and btn_l ==0: busy_l = 0 msg_f = "Mouse L cricked" print(msg_f) break # 右クリック・ボタンが新たに押されたなら elif busy_r == 0 and btn_r ==1: busy_r = 1 # 右クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_r == 1 and btn_r ==0: busy_r = 0 msg_f = "Mouse R cricked" print(msg_f) new_num = moji.strFixR_n(str(int(old_num)), num_size) break return new_num ####################################################################### # 数文字入力。return値を、int(num)で、「数値」に変換できる。 # line_n(0-6)で、画面上の入力行を指定する。末尾に tableの検索結果を表示するので 0-5 # prompt は、入力行でのプロンプト(例 'ch ? ') # num_asc(文字列 ='000')は初期値、'000'を入力桁数のサイズとする。 # prompt の桁数と、num_asc の桁数合計が、st7789の桁数(15)を超えないこと。 # 入力された整数で、tableを検索して、prompt の次の行に表示する。 def inputNumTable(InputMsg, table, line_n, prompt, num_asc): # line_n = line_n % 6 # 表示枠外に出ないよう規制 name_line = line_n +1 # tableの検索結果を表示する # prompt の桁数 try: if len(prompt) >= 1: prompt_size = len(prompt) else: prompt = '?' prompt_size = len(prompt) except: prompt = '?' prompt_size = len(prompt) # 念のため、文字列の先頭と末尾、および、文字間のスペースを取り除く num_asc = num_asc.replace(' ', '') # num_size = len(num_asc) の桁数内で編集する。最低でも1桁とする。 try: if len(num_asc) >= 1: num_size = len(num_asc) else: num_asc = '0' num_size = len(num_asc) except: num_asc = '0' num_size = len(num_asc) # 先頭にスペースを入れて num_size 桁にする old_num = moji.strFixR_n(str(int(num_asc)), num_size) new_num = old_num temp_num = new_num # ST7789 画面をクリア display.fill(st7789.BLACK) # InputMsg の表示 0行目〜6行目 for i in range(7): if len(InputMsg[i]) > 0: display.text(font2, moji.strFixL_n(InputMsg[i], ST7789_COL_SIZE), 0, \ DOT_R *i, st7789.WHITE) # マウス data_cnt_old = 999 btn_l = 0 btn_r = 0 btn_c = 0 cnt_x = 0 cnt_y = 0 # 20240725 Change # cnt_s = 0 new_num_int = int(new_num) cnt_s = new_num_int # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) data_cnt_old = 999 # マウスの左クリックが押されていない状態 busy_l = 0 # マウスの右クリックが押されていない状態 busy_r = 0 # 表示位置変数の設定 col_n = 0 cursol = prompt_size + num_size -1 # 初期表示・全体 # DOT_C = 16 : col_n , DOT_R = 32 : line_n dsp_data = prompt + moji.strFixR_n(str(int(new_num)), num_size) display.text(font2, moji.strFixL_n(dsp_data, ST7789_COL_SIZE),\ DOT_C * col_n, DOT_R * line_n, \ st7789.WHITE) # 初期表示・カーソル cursol_data = str(int(new_num) % 10) display.text(font2, cursol_data,\ DOT_C * cursol, DOT_R * line_n, \ st7789.RED, st7789.WHITE) while True: # 数字入力があれば表示する Enter または Esc で終了 if temp_num != new_num: col_n = 0 dsp_data = prompt + moji.strFixR_n(str(int(new_num)), num_size) display.text(font2, moji.strFixL_n(dsp_data, ST7789_COL_SIZE),\ DOT_C * col_n,\ DOT_R * line_n,\ st7789.WHITE) # GmName表示 table の有効範囲に限定 if new_num_int > 0 and new_num_int < len(table): ######################################## dsp_data = table[new_num_int] ######################################## display.text(font2, moji.strFixL_n(dsp_data, ST7789_COL_SIZE),\ DOT_C * col_n,\ DOT_R * name_line,\ st7789.WHITE) else: dsp_data = '' display.text(font2, moji.strFixL_n(dsp_data, ST7789_COL_SIZE),\ DOT_C * col_n,\ DOT_R * name_line,\ st7789.WHITE) # cursol_data 1文字を赤文字/白背景で表示する cursol_data = str(int(new_num) % 10) display.text(font2, cursol_data,\ DOT_C * cursol,\ DOT_R * line_n,\ st7789.RED, st7789.WHITE) temp_num = new_num[:] # PS2Keyboard # 1文字・記号入力なら if ps2key.rcv_size() == 1: ps2key_data = ps2key.get_code() if ps2key_data >= '0' and ps2key_data <= '9': ####################################################### # 右詰め表示で編集 temp_num = new_num[:] # copy new_num += ps2key_data new_num = moji.strFixR_n(str(int(new_num)), num_size) ####################################################### new_num_int = int(new_num) # 整数にする # マウス座標・現在地を更新 スクロールカウント ps2mus_rcv.set_cnt(cnt_x, cnt_y, new_num_int) ####################################################### # ファンクション・キー入力なら elif ps2key.rcv_size() > 1: ps2key_data = ps2key.get_code() # .lower 文字の一致を取りやすくするため、shift low で照合 if ps2key_data.lower() == "Esc".lower(): msg_f = "Esc" print(msg_f) new_num = moji.strFixR_n(str(int(old_num)), num_size) break elif ps2key_data.lower() == "Enter".lower(): msg_f = "Enter" print(msg_f) break elif ps2key_data.lower() == "BS".lower(): msg_f = "BS" print(msg_f) temp_num = new_num[:] # copy new_num = str(int(int(new_num) / 10)) new_num = moji.strFixR_n(str(int(new_num)), num_size) elif ps2key_data.lower() == "Del".lower(): msg_f = "Del" print(msg_f) temp_num = new_num[:] # copy new_num = str(int(int(new_num) / 10)) new_num = moji.strFixR_n(str(int(new_num)), num_size) # マウスからボタンクリックの情報、座標情報を取得する (data_cnt, btn_l, btn_r, btn_c, cnt_x, cnt_y, cnt_s) = ps2mus_rcv.get_data() # PS2 Mouse に入力があるなら if data_cnt_old != data_cnt: data_cnt_old = data_cnt print('cnt_s = ', cnt_s) # スクロールが変化しているなら if cnt_s != new_num_int: ####################################################### # 右詰め表示で編集 cnt_s で更新 temp_num = new_num[:] # copy new_num = moji.strFixR_n(str(cnt_s), num_size) ####################################################### new_num_int = int(new_num) # 整数にする # 左クリック・ボタンが新たに押されたなら elif busy_l == 0 and btn_l ==1: busy_l = 1 # 左クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_l == 1 and btn_l ==0: busy_l = 0 msg_f = "Mouse L cricked" print(msg_f) break # 右クリック・ボタンが新たに押されたなら elif busy_r == 0 and btn_r ==1: busy_r = 1 # 右クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_r == 1 and btn_r ==0: busy_r = 0 msg_f = "Mouse R cricked" print(msg_f) new_num = moji.strFixR_n(str(int(old_num)), num_size) break return new_num ####################################################################### # ymf825 def selectTone(num): をベースに、マウスとキーボード入力に変更 # Tone情報を、メモリから読み出す場所(tone_num) # または、Tone情報を、メモリに保存する場所(tone_num)を指定する # 初期値として numとして与え、デフォルトの位置とする # dsp_msg は、画面の表題表示する内容 # return は、tone_name のみとし、SysRq は global変数とする def selectTone(dsp_msg, num): tone_num = num & 0x0F # 0 - 15 tone_num_old = tone_num tone_num_now = tone_num # SysRq は結果を returnするのではなく、global変数として扱う global SysRq SysRq = 0 # SysRq キーは押されていない # 4 10 # ymf825.Voice_Op_Edit # 4 4 # ymf825.Voice_Com_Edit # 4 2 # ymf825.ToneMenuTable if DSP_COL_SIZE < TONE_MENU_COL_NUM: dsp_col_max = DSP_COL_SIZE else: dsp_col_max = TONE_MENU_COL_NUM -1 # 5 18 # ymf825.Voice_Op_Edit # 5 4 # ymf825.Voice_Com_Edit # 5 17 # ymf825.ToneMenuTable if DSP_ROW_SIZE < TONE_MENU_ROW_NUM: dsp_row_max = DSP_ROW_SIZE else: dsp_row_max = TONE_MENU_ROW_NUM -1 # 与えられた tone_num に表示を合わせるため cell_y home_y を求める # cell_y を計算 cell_y = tone_num +1 # tone_num:0-15 ==> cell_y:1-16 # home_y を計算 if cell_y <= DSP_ROW_SIZE: home_y = 1 # 表示エリアホーム y else: home_y = cell_y - DSP_ROW_SIZE +1 cell_x = 1 # 列 home_x = 1 # 表示エリアホーム x display.fill(st7789.BLACK) # 「Menu」表示内容 関数呼び出し時に dsp_msg = 'Select Tone' などと指定する display.text(font2, moji.strFixL_n(dsp_msg, ST7789_COL_SIZE), \ 16*0, 32*0, st7789.YELLOW) # 画面2行目に、セル・アドレス表示 [A99] display.text(font2, chr(64 + cell_x) + moji.int2str_n(cell_y, 2), \ 16*0, 32*1, st7789.CYAN) # 画面2行目に、列タイトルの表示 [Select Tone] for j in range(home_x, home_x + dsp_col_max): display.text(font2, moji.strFixL_n(ymf825.ToneMenuTable[0][j], 12), \ 16*3 + 16*3*(j - home_x), 32, st7789.CYAN) # 行タイトルの表示 [ 0] [ 1] [ 2] ... for i in range(home_y, home_y + dsp_row_max): display.text(font2, moji.int2str_n(ymf825.ToneMenuTable[i][0], 2), \ 0, 32*2 + 32*(i - home_y), st7789.CYAN) # ymf825.ToneMenuTable データの表示 for i in range(home_y, home_y + dsp_row_max): for j in range(home_x, home_x + dsp_col_max): display.text(font2, moji.strFixL_n(ymf825.ToneMenuTable[i][j], 12), \ 16*3 + 16*3*(j - home_x), \ 32*2 + 32*(i - home_y), st7789.WHITE) # 現在のセル表示を [RED] にする display.text(font2, moji.strFixL_n(ymf825.ToneMenuTable[cell_y][cell_x], 12), \ 16*3 + 16*3*(cell_x - home_x), \ 32*2 + 32*(cell_y - home_y), st7789.RED, st7789.WHITE) cell_y_old = cell_y # 行 cell_x_old = cell_x # 列 home_y_old = home_y # 表示エリアホーム y home_x_old = home_x # 表示エリアホーム x # マウス data_cnt_old = 999 btn_l = 0 btn_r = 0 btn_c = 0 cnt_x = 0 cnt_y = 0 cnt_s = 0 # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) data_cnt_old = 999 # マウスの左クリックが押されていない状態 busy_l = 0 # マウスの右クリックが押されていない状態 busy_r = 0 col = 0 row = 0 col_old = col row_old = row while True: # SysRq Enter または Esc で終了 if home_y_old != home_y: # 行タイトルの表示 [ 0] [ 1] [ 2] ... for i in range(home_y, home_y + dsp_row_max): display.text(font2, moji.int2str_n(ymf825.ToneMenuTable[i][0], 2), 0, \ 32*2 + 32*(i - home_y), st7789.CYAN) # ymf825.ToneMenuTable データの表示 for i in range(home_y, home_y + dsp_row_max): for j in range(home_x, home_x + dsp_col_max): display.text(font2, moji.strFixL_n(ymf825.ToneMenuTable[i][j], 12), \ 16*3 + 16*3*(j - home_x), \ 32*2 + 32*(i - home_y), st7789.WHITE) if cell_y_old != cell_y: # 画面2行目に、セル・アドレス表示 [A99] display.text(font2, chr(64 + cell_x) + moji.int2str_n(cell_y, 2), \ 16*0, 32*1, st7789.CYAN) # 直前のセル表示を [WHITE] に戻す display.text(font2, moji.strFixL_n(ymf825.ToneMenuTable[cell_y_old][cell_x_old], 12), \ 16*3 + 16*3*(cell_x_old - home_x), \ 32*2 + 32*(cell_y_old - home_y), st7789.WHITE) # 現在のセル表示を [RED] にする display.text(font2, moji.strFixL_n(ymf825.ToneMenuTable[cell_y][cell_x], 12), \ 16*3 + 16*3*(cell_x - home_x), \ 32*2 + 32*(cell_y - home_y), st7789.RED, st7789.WHITE) if cell_y_old != cell_y or home_y_old != home_y: cell_y_old = cell_y # 行 cell_x_old = cell_x # 列 home_y_old = home_y # 表示エリアホーム y home_x_old = home_x # 表示エリアホーム x # PS2Keyboard # 入力は Arw_U:「↑」 Arw_D:「↓」 「Enter」 「Esc」キーのみ受け付ける # ファンクション・キー入力なら if ps2key.rcv_size() > 1: ps2key_data = ps2key.get_code() # .lower 文字の一致を取りやすくするため、shift low で照合 if ps2key_data.lower() == "PrtSc".lower(): msg_f = "PrtSc = SysRq" print(msg_f) tone_num = tone_num_old ################################################### SysRq = 1 # PrtSc を SysRq として扱う ################################################### break elif ps2key_data.lower() == "Esc".lower(): msg_f = "Esc" print(msg_f) tone_num = tone_num_old break elif ps2key_data.lower() == "Enter".lower(): msg_f = "Enter" print(msg_f) tone_num = cell_y -1 # 0-15 break elif ps2key_data.lower() == "Arw_U".lower(): # ↑ msg_f = "ArwU" print(msg_f) if cell_y > 1 and cell_y < (TONE_MENU_ROW_NUM): # 2-16 cell_y_old = cell_y cell_y -= 1 # Table表示位置を調整(home_y の計算) if cell_y < home_y: home_y_old = home_y home_y = cell_y elif ps2key_data.lower() == "Arw_D".lower(): # ↓ msg_f = "ArwD" print(msg_f) if cell_y > 0 and cell_y < (TONE_MENU_ROW_NUM -1): # 1-15 cell_y_old = cell_y cell_y += 1 # Table表示位置を調整(home_y の計算) if cell_y > (home_y + (dsp_row_max -1)): home_y_old = home_y home_y = cell_y - (dsp_row_max -1) elif ps2key_data == "F1": msg_f = "F1" print(msg_f) # マウスからボタンクリックの情報、座標情報を取得する (data_cnt, btn_l, btn_r, btn_c, cnt_x, cnt_y, cnt_s) = ps2mus_rcv.get_data() # PS2 Mouse に入力があるなら if data_cnt_old != data_cnt: data_cnt_old = data_cnt # 左クリック、右クリック、ともに押されていないなら if busy_l == 0 and busy_r == 0: # マウスの座標の差分を取り出し、ArwD、ArwU キーと同じように扱う # 列 if cnt_x > 50: col_direc = 1 elif cnt_x < -50: col_direc = -1 else: col_direc = 0 # 行 if cnt_y < -50: row_direc = 1 elif cnt_y > 50: row_direc = -1 else: row_direc = 0 # 差分だけ取り出すので、クリア cnt_x = 0 cnt_y = 0 cnt_s = 0 # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) if row_direc == 1: # ↓ msg_f = "Down" print(msg_f) if cell_y > 0 and cell_y < (TONE_MENU_ROW_NUM -1): # 1-15 cell_y_old = cell_y cell_y += 1 # Table表示位置を調整(home_y の計算) if cell_y > (home_y + (dsp_row_max -1)): home_y_old = home_y home_y = cell_y - (dsp_row_max -1) elif row_direc == -1: # ↑ msg_f = "Up" print(msg_f) if cell_y > 1 and cell_y < (TONE_MENU_ROW_NUM): # 2-16 cell_y_old = cell_y cell_y -= 1 # Table表示位置を調整(home_y の計算) if cell_y < home_y: home_y_old = home_y home_y = cell_y # 左クリック・ボタンが新たに押されたなら if busy_l == 0 and btn_l ==1: busy_l = 1 # 左クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_l == 1 and btn_l ==0: busy_l = 0 msg_f = "Mouse L cricked" print(msg_f) tone_num = cell_y -1 # 0-15 break # 右クリック・ボタンが新たに押されたなら elif busy_r == 0 and btn_r ==1: busy_r = 1 # 右クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_r == 1 and btn_r ==0: busy_r = 0 msg_f = "Mouse R cricked" print(msg_f) tone_num = tone_num_old break utime.sleep_ms(100) return tone_num ####################################################################### # 処理メニューの選択 # 初期値として numとして与え、デフォルトの位置とする # dsp_msg は、画面の表題表示する内容 # return は、tone_name のみとし、SysRq は global変数とする def selectFunction(dsp_msg, num): function_num = num % FUNCTION_ROW_NUM # 0 - 12 function_num_old = function_num function_num_now = function_num # SysRq は結果を returnするのではなく、global変数として扱う global SysRq SysRq = 0 # SysRq キーは押されていない # 4 10 # ymf825.Voice_Op_Edit # 4 4 # ymf825.Voice_Com_Edit # 4 2 # ymf825.ToneMenuTable # 4 2 # FunctionTable if DSP_COL_SIZE < FUNCTION_COL_NUM: dsp_col_max = DSP_COL_SIZE else: dsp_col_max = FUNCTION_COL_NUM -1 # 5 18 # ymf825.Voice_Op_Edit # 5 4 # ymf825.Voice_Com_Edit # 5 17 # ymf825.ToneMenuTable # 5 13 # FunctionTable if DSP_ROW_SIZE < FUNCTION_ROW_NUM: dsp_row_max = DSP_ROW_SIZE else: dsp_row_max = FUNCTION_ROW_NUM -1 # 与えられた function_num に表示を合わせるため cell_y home_y を求める # cell_y を計算 cell_y = function_num # function_num:1-12 ==> cell_y:1-12 # home_y を計算 if cell_y <= DSP_ROW_SIZE: home_y = 1 # 表示エリアホーム y else: home_y = cell_y - DSP_ROW_SIZE +1 cell_x = 1 # 列 home_x = 1 # 表示エリアホーム x display.fill(st7789.BLACK) # 「Menu」表示内容 関数呼び出し時に dsp_msg = 'Select Function' などと指定する display.text(font2, moji.strFixL_n(dsp_msg, ST7789_COL_SIZE), \ 16*0, 32*0, st7789.MAGENTA, st7789.YELLOW) # 画面2行目に、セル・アドレス表示 [A99] display.text(font2, chr(64 + cell_x) + moji.int2str_n(cell_y, 2), \ 16*0, 32*1, st7789.CYAN) # 画面2行目に、列タイトルの表示 [Select Tone] for j in range(home_x, home_x + dsp_col_max): display.text(font2, moji.strFixL_n(FunctionTable[0][j], 12), \ 16*3 + 16*3*(j - home_x), 32, st7789.CYAN) # 行タイトルの表示 [ 0] [ 1] [ 2] ... for i in range(home_y, home_y + dsp_row_max): display.text(font2, moji.int2str_n(FunctionTable[i][0], 2), 0, \ 32*2 + 32*(i - home_y), st7789.CYAN) # FunctionTable データの表示 for i in range(home_y, home_y + dsp_row_max): for j in range(home_x, home_x + dsp_col_max): display.text(font2, moji.strFixL_n(FunctionTable[i][j], 12), \ 16*3 + 16*3*(j - home_x), \ 32*2 + 32*(i - home_y), st7789.WHITE) # 現在のセル表示を [RED] にする display.text(font2, moji.strFixL_n(FunctionTable[cell_y][cell_x], 12), \ 16*3 + 16*3*(cell_x - home_x), \ 32*2 + 32*(cell_y - home_y), st7789.RED, st7789.WHITE) cell_y_old = cell_y # 行 cell_x_old = cell_x # 列 home_y_old = home_y # 表示エリアホーム y home_x_old = home_x # 表示エリアホーム x # マウス data_cnt_old = 999 btn_l = 0 btn_r = 0 btn_c = 0 cnt_x = 0 cnt_y = 0 cnt_s = 0 # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) data_cnt_old = 999 # マウスの左クリックが押されていない状態 busy_l = 0 # マウスの右クリックが押されていない状態 busy_r = 0 col = 0 row = 0 col_old = col row_old = row while True: # if home_y_old != home_y: # 行タイトルの表示 [ 0] [ 1] [ 2] ... for i in range(home_y, home_y + dsp_row_max): display.text(font2, moji.int2str_n(FunctionTable[i][0], 2), 0, \ 32*2 + 32*(i - home_y), st7789.CYAN) # FunctionTable データの表示 for i in range(home_y, home_y + dsp_row_max): for j in range(home_x, home_x + dsp_col_max): display.text(font2, moji.strFixL_n(FunctionTable[i][j], 12), \ 16*3 + 16*3*(j - home_x), \ 32*2 + 32*(i - home_y), st7789.WHITE) if cell_y_old != cell_y: # 画面2行目に、セル・アドレス表示 [A99] display.text(font2, chr(64 + cell_x) + moji.int2str_n(cell_y, 2), \ 16*0, 32*1, st7789.CYAN) # 直前のセル表示を [WHITE] に戻す display.text(font2, moji.strFixL_n(FunctionTable[cell_y_old][cell_x_old], 12), \ 16*3 + 16*3*(cell_x_old - home_x), \ 32*2 + 32*(cell_y_old - home_y), st7789.WHITE) # 現在のセル表示を [RED] にする display.text(font2, moji.strFixL_n(FunctionTable[cell_y][cell_x], 12), \ 16*3 + 16*3*(cell_x - home_x), \ 32*2 + 32*(cell_y - home_y), st7789.RED, st7789.WHITE) if cell_y_old != cell_y or home_y_old != home_y: cell_y_old = cell_y # 行 cell_x_old = cell_x # 列 home_y_old = home_y # 表示エリアホーム y home_x_old = home_x # 表示エリアホーム x # PS2Keyboard # 入力は Arw_U:「↑」 Arw_D:「↓」 「Enter」 「Esc」キーのみ受け付ける # ファンクション・キー入力なら if ps2key.rcv_size() > 1: ps2key_data = ps2key.get_code() # .lower 文字の一致を取りやすくするため、shift low で照合 if ps2key_data.lower() == "PrtSc".lower(): msg_f = "PrtSc = SysRq" print(msg_f) function_num = function_num_old ################################################### # 20240712 この関数では、SysRq として扱わない # SysRq = 1 # PrtSc を SysRq として扱う ################################################### break elif ps2key_data.lower() == "Esc".lower(): msg_f = "Esc" print(msg_f) function_num = function_num_old break elif ps2key_data.lower() == "Enter".lower(): msg_f = "Enter" print(msg_f) function_num = cell_y # 1-12 break elif ps2key_data.lower() == "Arw_U".lower(): # ↑ msg_f = "ArwU" print(msg_f) if cell_y > 1 and cell_y < (FUNCTION_ROW_NUM): # 2-12 cell_y_old = cell_y cell_y -= 1 # Table表示位置を調整(home_y の計算) if cell_y < home_y: home_y_old = home_y home_y = cell_y elif ps2key_data.lower() == "Arw_D".lower(): # ↓ msg_f = "ArwD" print(msg_f) if cell_y > 0 and cell_y < (FUNCTION_ROW_NUM -1): # 1-11 cell_y_old = cell_y cell_y += 1 # Table表示位置を調整(home_y の計算) if cell_y > (home_y + (dsp_row_max -1)): home_y_old = home_y home_y = cell_y - (dsp_row_max -1) elif ps2key_data == "F1": msg_f = "F1" print(msg_f) function_num = 1 break elif ps2key_data == "F2": msg_f = "F2" print(msg_f) function_num = 2 break elif ps2key_data == "F3": msg_f = "F3" print(msg_f) function_num = 3 break elif ps2key_data == "F4": msg_f = "F4" print(msg_f) function_num = 4 break elif ps2key_data == "F5": msg_f = "F5" print(msg_f) function_num = 5 break elif ps2key_data == "F6": msg_f = "F6" print(msg_f) function_num = 6 break elif ps2key_data == "F7": msg_f = "F7" print(msg_f) function_num = 7 break # マウスからボタンクリックの情報、座標情報を取得する (data_cnt, btn_l, btn_r, btn_c, cnt_x, cnt_y, cnt_s) = ps2mus_rcv.get_data() # PS2 Mouse に入力があるなら if data_cnt_old != data_cnt: data_cnt_old = data_cnt # 左クリック、右クリック、ともに押されていないなら if busy_l == 0 and busy_r == 0: # マウスの座標の差分を取り出し、ArwD、ArwU キーと同じように扱う # 列 if cnt_x > 50: col_direc = 1 elif cnt_x < -50: col_direc = -1 else: col_direc = 0 # 行 if cnt_y < -50: row_direc = 1 elif cnt_y > 50: row_direc = -1 else: row_direc = 0 # 差分だけ取り出すので、クリア cnt_x = 0 cnt_y = 0 cnt_s = 0 # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) if row_direc == 1: # ↓ msg_f = "Down" print(msg_f) if cell_y > 0 and cell_y < (FUNCTION_ROW_NUM -1): # 1-11 cell_y_old = cell_y cell_y += 1 # Table表示位置を調整(home_y の計算) if cell_y > (home_y + (dsp_row_max -1)): home_y_old = home_y home_y = cell_y - (dsp_row_max -1) elif row_direc == -1: # ↑ msg_f = "Up" print(msg_f) if cell_y > 1 and cell_y < (FUNCTION_ROW_NUM): # 2-12 cell_y_old = cell_y cell_y -= 1 # Table表示位置を調整(home_y の計算) if cell_y < home_y: home_y_old = home_y home_y = cell_y # 左クリック・ボタンが新たに押されたなら if busy_l == 0 and btn_l ==1: busy_l = 1 # 左クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_l == 1 and btn_l ==0: busy_l = 0 msg_f = "Mouse L cricked" print(msg_f) function_num = cell_y # 1-12 break # 右クリック・ボタンが新たに押されたなら elif busy_r == 0 and btn_r ==1: busy_r = 1 # 右クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_r == 1 and btn_r ==0: busy_r = 0 msg_f = "Mouse R cricked" print(msg_f) function_num = function_num_old break utime.sleep_ms(100) return function_num ####################################################################### # SubMenu の選択 # リスト SubMenu[0]~SubMenu[6] は、画面表示する内容 # return は、Enter=True または Esc=False のみとする def selectSubMenu(SubMenu): display.fill(st7789.BLACK) # SubMenu の表示 for i in range(7): display.text(font2, moji.strFixL_n(SubMenu[i], ST7789_COL_SIZE), \ DOT_C *0, DOT_R *i, st7789.WHITE) # マウス data_cnt_old = 999 btn_l = 0 btn_r = 0 btn_c = 0 cnt_x = 0 cnt_y = 0 cnt_s = 0 # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) data_cnt_old = 999 # マウスの左クリックが押されていない状態 busy_l = 0 # マウスの右クリックが押されていない状態 busy_r = 0 while True: # PS2Keyboard # 入力は 「Enter」 「Esc」キーのみ受け付ける # ファンクション・キー入力なら if ps2key.rcv_size() > 1: ps2key_data = ps2key.get_code() # .lower 文字の一致を取りやすくするため、shift low で照合 if ps2key_data.lower() == "Esc".lower(): msg_f = "Esc" print(msg_f) return False elif ps2key_data.lower() == "Enter".lower(): msg_f = "Enter" print(msg_f) return True # マウスからボタンクリックの情報、座標情報を取得する (data_cnt, btn_l, btn_r, btn_c, cnt_x, cnt_y, cnt_s) = ps2mus_rcv.get_data() # PS2 Mouse に入力があるなら if data_cnt_old != data_cnt: data_cnt_old = data_cnt # 左クリック・ボタンが新たに押されたなら if busy_l == 0 and btn_l ==1: busy_l = 1 # 左クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_l == 1 and btn_l ==0: busy_l = 0 msg_f = "Mouse L cricked" print(msg_f) return True # 右クリック・ボタンが新たに押されたなら if busy_r == 0 and btn_r ==1: busy_r = 1 # 右クリック・ボタンが押された状態から離れた(=クリック)なら elif busy_r == 1 and btn_r ==0: busy_r = 0 msg_f = "Mouse R cricked" print(msg_f) return False utime.sleep_ms(100) ####################################################################### # Alarm 画面表示のみ # 全画面の 文字色(colorFont)を指定できる def alarm(AlarmMsg, colorFont=st7789.RED): # display.fill(st7789.BLACK) # AlarmMsg の表示 for i in range(7): if len(AlarmMsg[i]) > 0: display.text(font2, moji.strFixL_n(AlarmMsg[i], ST7789_COL_SIZE), \ 0, 32 *i, colorFont, st7789.WHITE) ####################################################################### # MFRC522(13.56MHz)RFID にタグやカードが置かれるか、取り除かれるまで待つ # タグやカードが置かれるか、取り除かれるまで Alarm画面を表示する # OnOff(True False)で論理指定する # True:取り除かれるまで待つ False:置かれるまで待つ def waitTag(AlarmMsg, OnOff=False): ptn = 0 display.fill(st7789.BLACK) while midi.checkOnRfid() == OnOff: if ptn == 0: colorFont = st7789.RED alarm(AlarmMsg, colorFont) ptn = 1 else: colorFont = st7789.BLUE alarm(AlarmMsg, colorFont) ptn = 0 utime.sleep_ms(500) ####################################################################### # def selectTone(num): をベースに、チャンネル・トーンのテーブルをキー入力で作成する # # def editToneTable(ToneTable, ToneName): tone_num = 16 tone_num_old = tone_num display.fill(st7789.BLACK) # ST7789 表示窓の設定 # 4 10 # ymf825.Voice_Op_Edit # 4 4 # ymf825.Voice_Com_Edit # 4 2 # ymf825.ToneMenuTable # 4 2 # ymf825.ToneTable if DSP_COL_SIZE < TONE_TABLE_COL_NUM: dsp_col_max = DSP_COL_SIZE else: dsp_col_max = TONE_TABLE_COL_NUM -1 # 5 18 # ymf825.Voice_Op_Edit # 5 4 # ymf825.Voice_Com_Edit # 5 17 # ymf825.ToneMenuTable # 5 17 # ymf825.ToneTable if DSP_ROW_SIZE < TONE_TABLE_ROW_NUM: dsp_row_max = DSP_ROW_SIZE else: dsp_row_max = TONE_TABLE_ROW_NUM -1 cell_x = 1 # 列 cell_y = 1 # home_x = 1 # 表示エリアホーム x home_y = 1 # 表示エリアホーム x # ST7789 初期画面表示 # 「Menu」= 'Save Tone As...' の場合 dsp_data = 'Edit Tone Table' display.text(font2, dsp_data, 16*0, 32*0, st7789.YELLOW) # 画面2行目に、セル・アドレス表示 [A99] display.text(font2, chr(64 + cell_x) + moji.int2str_n(cell_y, 2), \ 16*0, 32*1, st7789.CYAN) # 画面2行目に、列タイトルの表示 [ Select Tone] for j in range(home_x, home_x + dsp_col_max): display.text(font2, moji.strFixL_n(ToneTable[0][j], 12), \ 16*3 + 16*3*(j - home_x), 32, st7789.CYAN) # 行タイトルの表示 [ 0] [ 1] [ 2] ... for i in range(home_y, home_y + dsp_row_max): display.text(font2, moji.int2str_n(ToneTable[i][0], 2), 0, \ 32*2 + 32*(i - home_y), st7789.CYAN) # ToneTable データの表示 for i in range(home_y, home_y + dsp_row_max): for j in range(home_x, home_x + dsp_col_max): dsp_data = ToneName[ToneTable[i][j]] display.text(font2, moji.strFixL_n(dsp_data, 12), \ 16*3 + 16*3*(j - home_x), \ 32*2 + 32*(i - home_y), st7789.WHITE) # 現在のセル表示を [RED] にする dsp_data = ToneName[ToneTable[cell_y][cell_x]] display.text(font2, moji.strFixL_n(dsp_data, 12), \ 16*3 + 16*3*(cell_x - home_x), \ 32*2 + 32*(cell_y - home_y), st7789.RED, st7789.WHITE) cell_y_old = cell_y # 行 cell_x_old = cell_x # 列 home_y_old = home_y # 表示エリアホーム y home_x_old = home_x # 表示エリアホーム x end = 0 # end != 0 で終了 while end == 0: # Enter または Esc で終了 if home_y_old != home_y: # 行タイトルの表示 [ 0] [ 1] [ 2] ... for i in range(home_y, home_y + dsp_row_max): display.text(font2, moji.int2str_n(ToneTable[i][0], 2), 0, \ 32*2 + 32*(i - home_y), st7789.CYAN) # ToneTable データの表示 for i in range(home_y, home_y + dsp_row_max): for j in range(home_x, home_x + dsp_col_max): dsp_data = ToneName[ToneTable[i][j]] display.text(font2, moji.strFixL_n(dsp_data, 12), \ 16*3 + 16*3*(j - home_x), \ 32*2 + 32*(i - home_y), st7789.WHITE) if cell_y_old != cell_y: # 画面2行目に、セル・アドレス表示 [A99] display.text(font2, chr(64 + cell_x) + moji.int2str_n(cell_y, 2), \ 16*0, 32*1, st7789.CYAN) # 直前のセル表示を [WHITE] に戻す dsp_data = ToneName[ToneTable[cell_y_old][cell_x_old]] display.text(font2, moji.strFixL_n(dsp_data, 12), \ 16*3 + 16*3*(cell_x_old - home_x), \ 32*2 + 32*(cell_y_old - home_y), st7789.WHITE) # 現在セルの tone_num を取得する tone_num = ToneTable[cell_y][cell_x] # 現在のセル表示を [RED] にする dsp_data = ToneName[ToneTable[cell_y][cell_x]] display.text(font2, moji.strFixL_n(dsp_data, 12), \ 16*3 + 16*3*(cell_x - home_x), \ 32*2 + 32*(cell_y - home_y), st7789.RED, st7789.WHITE) if tone_num_old != tone_num: # 現在セルの tone_num を取得する tone_num_old = ToneTable[cell_y][cell_x] # 現在のセル表示を [RED] にする dsp_data = ToneName[ToneTable[cell_y][cell_x]] display.text(font2, moji.strFixL_n(dsp_data, 12), \ 16*3 + 16*3*(cell_x - home_x), \ 32*2 + 32*(cell_y - home_y), st7789.RED, st7789.WHITE) if cell_y_old != cell_y or home_y_old != home_y: cell_y_old = cell_y # 行 cell_x_old = cell_x # 列 home_y_old = home_y # 表示エリアホーム y home_x_old = home_x # 表示エリアホーム x # PS2Keyboard # 入力は Arw_U:「↑」 Arw_D:「↓」 Arw_L:「←」 Arw_R:「→」 # 「Enter」 「Esc」キーのみ受け付ける # ファンクション・キー入力なら if ps2key.rcv_size() > 1: ps2key_data = ps2key.get_code() # .lower 文字の一致を取りやすくするため、shift low で照合 if ps2key_data.lower() == "Esc".lower(): msg_f = "Esc" print(msg_f) end = 1 elif ps2key_data.lower() == "Enter".lower(): msg_f = "Enter" print(msg_f) end = 1 elif ps2key_data.lower() == "Arw_U".lower(): # ↑ msg_f = "ArwU" print(msg_f) if cell_y > 1 and cell_y < (TONE_TABLE_ROW_NUM): # 2-16 cell_y_old = cell_y cell_y -= 1 # Table表示位置を調整(home_y の計算) if cell_y < home_y: home_y_old = home_y home_y = cell_y elif ps2key_data.lower() == "Arw_D".lower(): # ↓ msg_f = "ArwD" print(msg_f) if cell_y > 0 and cell_y < (TONE_TABLE_ROW_NUM -1): # 1-15 cell_y_old = cell_y cell_y += 1 # Table表示位置を調整(home_y の計算) if cell_y > (home_y + (dsp_row_max -1)): home_y_old = home_y home_y = cell_y - (dsp_row_max -1) elif ps2key_data.lower() == "Arw_L".lower(): # ← msg_f = "ArwL" print(msg_f) if tone_num > 0: # 1-16 tone_num_old = tone_num tone_num -= 1 else: tone_num_old = tone_num tone_num = 16 # data keep ToneTable[cell_y][cell_x] = tone_num elif ps2key_data.lower() == "Arw_R".lower(): # → msg_f = "ArwR" print(msg_f) if tone_num < (TONE_TABLE_ROW_NUM -1): # 0-15 tone_num_old = tone_num tone_num += 1 else: tone_num_old = tone_num tone_num = 0 # data keep ToneTable[cell_y][cell_x] = tone_num elif ps2key_data == "F1": msg_f = "F1" print(msg_f) ############################################################################# # 20240723 Move Here frm ymf825py.py # Edit Voice Operation Parameter # old def editVoiceOp(num, OpNum): def editVoiceOp(Voice_Op_Edit, VoiceOpName, num, OpNum): tone_num = num & 0x0F # global Voice_Op_Edit # global VoiceOpName global backColor display.fill(st7789.BLACK) # 4 10 # Voice_Op_Edit # 4 4 # Voice_Com_Edit # 4 2 # ToneMenuTable if DSP_COL_SIZE < VOICE_OP_COL_NUM: dsp_col_max = DSP_COL_SIZE else: dsp_col_max = VOICE_OP_COL_NUM -1 # 5 18 # Voice_Op_Edit # 5 4 # Voice_Com_Edit # 5 17 # ToneMenuTable if DSP_ROW_SIZE < VOICE_OP_ROW_NUM: dsp_row_max = DSP_ROW_SIZE else: dsp_row_max = VOICE_OP_ROW_NUM -1 try: if OpNum >= VOICE_OP_ROW_NUM: OpNum = VOICE_OP_ROW_NUM -1 elif OpNum <= 0: OpNum = 1 except: print('TypeError ComNum in def editVoiceOp(Voice_Op_Edit, VoiceOpName, num, OpNum):') OpNum = 0 # OpNum の範囲を 1 - 17 に制限する if OpNum >= VOICE_OP_ROW_NUM: OpNum = VOICE_OP_ROW_NUM -1 elif OpNum <= 0: OpNum = 1 # 与えられた OpNum に表示を合わせるため cell_y home_y を求める # cell_y を計算 cell_y = OpNum # 行 1 - 17 # home_y を計算 if cell_y <= DSP_ROW_SIZE: home_y = 1 # 表示エリアホーム y else: home_y = cell_y - DSP_ROW_SIZE +1 cell_x = 1 # 列 home_x = 1 # 表示エリアホーム x menu_num = 4 # ホーム座標表示 Menu_bar の表示 「 OK」「 BK」「 DF」 menu_bar = [None] * menu_num menu_bar[0] = " " # 6 menu_bar[1] = " OK" # 3 menu_bar[2] = " BK" # 3 menu_bar[3] = " DF" # 3 dsp_data = "" for i in range(menu_num): dsp_data += menu_bar[i] display.text(font2, dsp_data, 16*0, 32*0, st7789.GREEN) # セル・アドレス表示 [A99] display.text(font2, chr(64 + cell_x) + moji.int2str_n(cell_y, 2), \ 16*0, 32*1, st7789.CYAN) # 列タイトルの表示 [ e1] [ e2] [ e3] [ e4] [ o1] [ o2] [ o3] [ o4] [ Mx] for j in range(home_x, home_x + dsp_col_max): display.text(font2, moji.strFixR_n(Voice_Op_Edit[OPT][j], 3), \ 16*3 + 16*3*(j - home_x), 32, st7789.CYAN) # 行タイトルの表示 [SR ] [XOF] ... [WS ] [FB ] for i in range(home_y, home_y + dsp_row_max): display.text(font2, moji.strFixL_n(Voice_Op_Edit[i][OP0], 3), 0, \ 32*2 + 32*(i - home_y), st7789.CYAN) # Voice_Op_Edit データの表示 for y in range(home_y, home_y + dsp_row_max): for x in range(home_x, home_x + dsp_col_max): display.text(font2, moji.int2str_n(Voice_Op_Edit[y][x], 3), \ 16*3 + 16*3*(x - home_x), \ 32*2 + 32*(y - home_y), st7789.WHITE) # 現在のセル表示を [RED] にする display.text(font2, moji.int2str_n(Voice_Op_Edit[cell_y][cell_x], 3), \ 16*3 + 16*3*(cell_x - home_x), \ 32*2 + 32*(cell_y - home_y), st7789.RED, st7789.WHITE) cell_y_old = cell_y # 行 cell_x_old = cell_x # 列 # 20240727 Add cell_x_now = cell_x cell_y_now = cell_y home_y_old = home_y # 表示エリアホーム y home_x_old = home_x # 表示エリアホーム x cnt_y = (cell_y *100 +50) *-1 data_cnt = 0 btn_l = 0 btn_r = 0 btn_c = 0 cnt_x = 0 # cnt_y = 0 cnt_s = 0 # data_cnt_old = data_cnt btn_l_old = btn_l btn_r_old = btn_r btn_c_old = btn_c cnt_x_old = cnt_x cnt_y_old = cnt_y cnt_s_old = cnt_s # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) data_cnt_old = 999 # mode:0 編集セルを選択するモード、mode:1 選択したセルを編集するモード mode = 0 # マウスの左クリックが押された状態(離れたことを確認できていない状態) busy_l = 0 # マウスの右クリックが押された状態(離れたことを確認できていない状態) busy_r = 0 # mode == 3 で終了 while mode != 3: flag_M = 0 # Mouse flag_K = 0 # Key msg_f = '' # FunctionKey ############################################################## # PS2Keyboard入力があれば、マウス情報に置き換える。 # マウスのクリック押下時点で反応するように設計したのだ。 # 入力は Arw_U:「↑」, Arw_D:「↓」, Arw_L:「←」, Arw_R:「→」 # 「Enter」 「Esc」キーのみ受け付ける ############################################################## # 1文字・記号入力なら if ps2key.rcv_size() == 1: ps2key_data = ps2key.get_code() # 数字キーならば if ps2key_data >= '0' and ps2key_data <= '9': flag_K = 1 ####################################################### # 右詰め表示で編集 # Voice_Com_Editの場合は、Maxが10以上なら2桁 if Voice_Op_Edit[cell_y_now][OP9] > 10: cnt_s = cnt_s *10 + int(ps2key_data) else: cnt_s = int(ps2key_data) # マウス座標・現在地を更新 スクロールカウント ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) ####################################################### # ファンクション・キー入力なら elif ps2key.rcv_size() > 1: ps2key_data = ps2key.get_code() flag_K = 1 # .lower 文字の一致を取りやすくするため、shift low で照合 if ps2key_data.lower() == "PrtSc".lower(): msg_f = "PrtSc = SysRq" print(msg_f) ################################################### SysRq = 1 # PrtSc を SysRq として扱う ################################################### # break elif ps2key_data.lower() == "BS".lower(): msg_f = "BS" print(msg_f) # 末尾の1桁を消すということは、1/10にすること cnt_s = int(cnt_s / 10) flag_M = 1 elif ps2key_data.lower() == "Del".lower(): msg_f = "Del" print(msg_f) # 0クリア cnt_s = 0 flag_M = 1 elif ps2key_data.lower() == "Esc".lower(): msg_f = "Esc" print(msg_f) return OpNum elif ps2key_data.lower() == "Enter".lower(): msg_f = "Enter" print(msg_f) flag_M = 1 elif ps2key_data.lower() == "Arw_U".lower(): # ↑ msg_f = "ArwU" print(msg_f) cnt_y += 100 print(f'{cnt_y} cnt_y += 100') flag_M = 1 elif ps2key_data.lower() == "Arw_D".lower(): # ↓ msg_f = "ArwD" print(msg_f) cnt_y -= 100 print(f'{cnt_y} cnt_y -= 100') flag_M = 1 elif ps2key_data.lower() == "Arw_L".lower(): # ← msg_f = "ArwL" print(msg_f) cnt_x -= 100 print(f'{cnt_x} cnt_x -= 100') flag_M = 1 elif ps2key_data.lower() == "Arw_R".lower(): # → msg_f = "ArwR" print(msg_f) cnt_x += 100 print(f'{cnt_x} cnt_x += 100') flag_M = 1 elif ps2key_data.lower() == "PgUp".lower(): # PageUp msg_f = "PageUp" print(msg_f) cnt_y += 100 *8 print(f'{cnt_y} cnt_y += 100 *8') flag_M = 1 elif ps2key_data.lower() == "PgDn".lower(): # PageDown msg_f = "PageDown" print(msg_f) cnt_y -= 100 *8 print(f'{cnt_y} cnt_y -= 100 *8') flag_M = 1 elif ps2key_data.lower() == "Home".lower(): # Home msg_f = "Home" print(msg_f) cnt_x -= 100 *7 print(f'{cnt_x} cnt_x -= 100 *7') flag_M = 1 elif ps2key_data.lower() == "End".lower(): # End msg_f = "End" print(msg_f) cnt_x += 100 *7 print(f'{cnt_x} cnt_x += 100 *7') flag_M = 1 elif ps2key_data == "F1": msg_f = "F1" print(msg_f) if flag_M == 1: # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) # マウス入力ありなら else: # マウスからボタンクリックの情報、座標情報を取得する (data_cnt, btn_l, btn_r, btn_c, cnt_x, cnt_y, cnt_s) = ps2mus_rcv.get_data() # PS2 Mouse に入力があるなら if data_cnt_old != data_cnt: data_cnt_old = data_cnt flag_M = 1 # マウス入力か、キー入力があれば if flag_M == 1 or flag_K == 1: # セル・選択モード #################################################### if mode == 0: # 左クリック・ボタンが押されたままでないなら if busy_l == 0: # 列 Voice_Op_Edit VOICE_OP_COL_NUM if cnt_x > VOICE_OP_COL_NUM *100 -1: cnt_x = VOICE_OP_COL_NUM *100 -1 elif cnt_x < 100: cnt_x = 100 # 行 Voice_Op_Edit VOICE_OP_ROW_NUM if cnt_y < (VOICE_OP_ROW_NUM *100 -1) *-1: cnt_y = (VOICE_OP_ROW_NUM *100 -1) *-1 elif cnt_y > -100: cnt_y = -100 # セル・アドレスの表示データ生成 [x:1-9, y:1-9] # 列 cell_x = int(cnt_x / 100) # 行 cell_y = int((-1 * cnt_y) / 100) if cell_x_old < cell_x: cell_x_direc = 1 elif cell_x_old > cell_x: cell_x_direc = -1 else: cell_x_direc = 0 if cell_y_old < cell_y: cell_y_direc = 1 elif cell_y_old > cell_y: cell_y_direc = -1 else: cell_y_direc = 0 # Table表示位置を調整(home_x home_y の計算) ############################################################ if cell_x_direc == 1 and cell_x > (home_x + (dsp_col_max -1)): home_x = cell_x - (dsp_col_max -1) elif cell_x_direc == -1 and cell_x < home_x: home_x = cell_x ############################################################ if cell_y_direc == 1 and cell_y > (home_y + (dsp_row_max -1)): home_y = cell_y - (dsp_row_max -1) elif cell_y_direc == -1 and cell_y < home_y: home_y = cell_y ############################################################ # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) # セル・アドレス表示 [A99] adr_dsp = chr(64 + cell_x) + moji.int2str_n(cell_y, 2) display.text(font2, adr_dsp, 16*0, 32*1, st7789.CYAN) ########################################################### # DEBUG home_x home_y を左上隅に表示 # menu_bar[0] = moji.int2str_n(home_x, 3) + moji.int2str_n(home_y, 3) # display.text(font2, menu_bar[0], 0, 0, st7789.GREEN) ########################################################### # セル値を、左上隅に表示 val = Voice_Op_Edit[cell_y][cell_x] display.text(font2, moji.int2str_n(val, 6), \ 16*0, 32*0, st7789.RED) # home_x または home_y に変化があれば、Voice_Op_Edit を再表示する if home_x_old != home_x or home_y_old != home_y: for y in range(home_y, home_y + dsp_row_max): for x in range(home_x, home_x + dsp_col_max): display.text(font2, moji.int2str_n(Voice_Op_Edit[y][x], 3), \ 16*3 + 16*3*(x - home_x), \ 32*2 + 32*(y - home_y), st7789.WHITE) # home_x に変化があれば、列タイトルの再表示 [ e1] [ e2] ... [ o1] [ o2] ... [ Mx] if home_x_old != home_x: for j in range(home_x, home_x + dsp_col_max): display.text(font2, moji.strFixR_n(Voice_Op_Edit[OPT][j], 3), \ 16*3 + 16*3*(j - home_x), 32, st7789.CYAN) # home_y に変化があれば、行タイトルの再表示 [SR ] [XOF] ... [WS ] [FB ] if home_y_old != home_y: for i in range(home_y, home_y + dsp_row_max): display.text(font2, moji.strFixL_n(Voice_Op_Edit[i][OP0], 3), 0, \ 32*2 + 32*(i - home_y), st7789.CYAN) # 現在セル座標に変化があれば if cell_x_old != cell_x or cell_y_old != cell_y: # 直前のセル表示を [WHITE] に戻す display.text(font2, moji.int2str_n(Voice_Op_Edit[cell_y_old][cell_x_old], 3), \ 16*3 + 16*3*(cell_x_old - home_x_old), \ 32*2 + 32*(cell_y_old - home_y_old), st7789.WHITE) # 現在のセル表示を 文字色:[RED] 背景色[WHITE] にする display.text(font2, moji.int2str_n(Voice_Op_Edit[cell_y][cell_x], 3), \ 16*3 + 16*3*(cell_x - home_x), \ 32*2 + 32*(cell_y - home_y), st7789.RED, st7789.WHITE) # セルの値を、左上隅に表示 val = Voice_Op_Edit[cell_y][cell_x] display.text(font2, moji.int2str_n(val, 6), \ 16*0, 32*0, st7789.RED) # セル座標を更新 cell_y_old = cell_y cell_x_old = cell_x home_y_old = home_y home_x_old = home_x # 左クリック・ボタンが押されていたら、セル・編集モードに移行 if btn_l == 1 or msg_f == 'Enter': if btn_l == 1: busy_l = 1 elif msg_f == 'Enter': msg_f = '' # 編集中の座標を保存しておく cell_x_now = cell_x cell_y_now = cell_y # 編集中のセル値を、左上隅に表示 if cell_x_now >= 1 and cell_x_now <= 4: # 変更前の内容を表示 OpNum = cell_y_now OpName = Voice_Op_Edit[cell_y_now][cell_x_now] print(f'Sub OK Voice_Op_Edit {VoiceOpName[cell_y_now]} {cell_x_now} = ', OpName, end = '') val = Voice_Op_Edit[cell_y_now][cell_x_now] display.text(font2, moji.int2str_n(val, 6), \ 16*0, 32*0, st7789.RED, st7789.WHITE) else: display.text(font2, moji.strFixL_n("N/A", 6), \ 16*0, 32*0, st7789.RED, st7789.WHITE) # menu_bar[1] = " OK", menu_bar[2] = " BK" menu_bar[3] = " DF" # [ OK]の位置にセル座標を合わせる cnt_x = 250 cnt_y = 50 cnt_s = val # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) # セル編集モードに移行 mode = 1 # 左クリック・ボタンが押されたままなら、離れるまで待つ elif btn_l == 0: busy_l = 0 # 新たに右クリック・ボタンが押されていたら if busy_r == 0 and btn_r == 1: busy_r = 1 mode = 2 # セル・編集モード #################################################### elif mode == 1: # 左クリック・ボタンが押されたままでないなら if busy_l == 0: # 列 if cnt_x > (menu_num +1) *100 -1: # cnt_x = cnt_x > (menu_num +1) *100 -1 cnt_x = (menu_num +1) *100 -1 elif cnt_x < 200: cnt_x = 200 # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) # セル・アドレスの生成 # menu_bar[1] = " OK", menu_bar[2] = " BK" menu_bar[3] = " DF" # 列 1, 2, 3 cell_x_edt = int(cnt_x / 100) -1 # 行 cell_y_edt = 0 # メニューバー表示を編集用にする(選択メニューの背景をマゼンタ色) for i in range(1, menu_num): if i == cell_x_edt: display.text(font2, menu_bar[i], \ 16*6 + 16*3*(i - 1), 0, st7789.GREEN, st7789.MAGENTA) else: display.text(font2, menu_bar[i], \ 16*6 + 16*3*(i - 1), 0, st7789.GREEN) # 編集中のセル値を、左上隅に表示 if cell_x_now >= 1 and cell_x_now <= 4: # 最大値を超えない if cnt_s > Voice_Op_Edit[cell_y_now][OP9]: val = Voice_Op_Edit[cell_y_now][OP9] # マイナス値はなし elif cnt_s < 0: val = 0 else: val = cnt_s ##################################################### # display.text(font2, moji.int2str_n(val, 6), \ # 16*0, 32*0, st7789.RED, st7789.WHITE) ##################################################### display.text(font2, moji.int2str_n(val, 6), \ 16*0, 32*0, st7789.RED, backColor) ##################################################### # 編集対象外のセルなら #N/A を表示 else: display.text(font2, moji.strFixL_n("#N/A", 6), \ 16*0, 32*0, st7789.RED, st7789.WHITE) # 左クリック・ボタンが押されていたら、 # Voice_Op_Edit に代入して、セル・編集モードを終了 if btn_l == 1 or msg_f == 'Enter': if btn_l == 1: busy_l = 1 elif msg_f == 'Enter': msg_f = '' # 左クリックしたセルが menu_bar[1] = " OK" なら # 編集中のセル値を、Voice_Op_Edit に代入して表示する if cell_x_edt == 1: Voice_Op_Edit[cell_y_now][cell_x_now] = val display.text(font2, moji.int2str_n(Voice_Op_Edit[cell_y_now][cell_x_now], 3), \ 16*3 + 16*3*(cell_x_now - home_x), \ 32*2 + 32*(cell_y_now - home_y), st7789.RED, st7789.WHITE) OpNum = cell_y_now OpName = Voice_Op_Edit[cell_y_now][cell_x_now] print(f' to {VoiceOpName[cell_y_now]} {cell_x_now} = ', OpName) # 左クリックしたセルが menu_bar[2] = " BK" なら、元の値に戻す elif cell_x_edt == 2: val = Voice_Op_Edit[cell_y_now][cell_x_now] # 左クリックしたセルが menu_bar[3] = " DF" なら # Voice_Op_Edit の Orig 値を、Voice_Op_Edit に代入して表示する elif cell_x_edt == 3 and (cell_x_now >= 1 and cell_x_now <= 4): Voice_Op_Edit[cell_y_now][cell_x_now] = Voice_Op_Edit[cell_y_now][cell_x_now +4] val = Voice_Op_Edit[cell_y_now][cell_x_now] display.text(font2, moji.int2str_n(Voice_Op_Edit[cell_y_now][cell_x_now], 3), \ 16*3 + 16*3*(cell_x_now - home_x), \ 32*2 + 32*(cell_y_now - home_y), st7789.RED, st7789.WHITE) # 編集が終了したら、セル・選択モードに戻る # メニューバー表示を元に戻す for i in range(1, menu_num): display.text(font2, menu_bar[i], \ 16*6 + 16*3*(i - 1), 0, st7789.GREEN) # 編集中の左上隅に表示を元に戻す display.text(font2, moji.int2str_n(val, 6), \ 16*0, 32*0, st7789.RED) # 編集前の座標にセット cnt_x = cell_x_now *100 + 50 cnt_y = (cell_y_now *100 + 50) * -1 # cnt_s = 0 #マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) # セル選択モードに移行 mode = 0 # 左クリック・ボタンが押されたままなら、離れるまで待つ elif btn_l == 0: busy_l = 0 # 編集終了・確認 モード ############################################### # mode:0 で、右クリック・ボタンが押されていたらここにくる elif mode == 2: # 右クリック・ボタンが押されたままなら、離れるまで待つ if btn_r == 0: print('right button clicked ... End of editVoiceOp.') busy_r = 0 mode = 3 # End of editVoiceOp else: return OpNum ############################################################################# # 20240723 Move Here frm ymf825py.py # Edit Voice Commonn Parameter # old def editVoiceCom(num, ComNum): def editVoiceCom(Voice_Com_Edit, VoiceComName, num, ComNum): tone_num = num & 0x0F # global Voice_Com_Edit # global VoiceComName global backColor display.fill(st7789.BLACK) CELL_WIDTH = const(4) # 4 10 # 4 4 if DSP_COL_SIZE < VOICE_COM_COL_NUM: dsp_col_max = DSP_COL_SIZE else: dsp_col_max = VOICE_COM_COL_NUM -1 # 5 18 # 5 4 if DSP_ROW_SIZE < VOICE_COM_ROW_NUM: dsp_row_max = DSP_ROW_SIZE else: dsp_row_max = VOICE_COM_ROW_NUM -1 try: if ComNum >= VOICE_COM_ROW_NUM: ComNum = VOICE_COM_ROW_NUM -1 elif ComNum <= 0: ComNum = 1 except: print(f'TypeError ComNum in def editVoiceCom(Voice_Com_Edit, VoiceComName, num, ComNum):') ComNum = 0 # ComNum の範囲を 1 - 3 に制限する if ComNum >= VOICE_COM_ROW_NUM: ComNum = VOICE_COM_ROW_NUM -1 elif ComNum <= 0: ComNum = 1 # 与えられた ComNum に表示を合わせるため cell_y home_y を求める # cell_y を計算 cell_y = ComNum # 行 1 - 3 # home_y を計算 if cell_y <= DSP_ROW_SIZE: home_y = 1 # 表示エリアホーム y else: home_y = cell_y - DSP_ROW_SIZE +1 cell_x = 1 # 列 home_x = 1 # 表示エリアホーム x menu_num = 4 # ホーム座標表示 Menu_bar の表示 「 OK」「 BK」「 DF」 menu_bar = [None] * menu_num menu_bar[0] = " " # 6 menu_bar[1] = " OK" # 3 menu_bar[2] = " BK" # 3 menu_bar[3] = " DF" # 3 dsp_data = "" for i in range(menu_num): dsp_data += menu_bar[i] display.text(font2, dsp_data, 16*0, 32*1, st7789.GREEN) # セル・アドレス表示 [A99] display.text(font2, chr(64 + cell_x) + moji.int2str_n(cell_y, 2), \ 16*0, 32*1, st7789.CYAN) # 列タイトルの表示 [ Edt] [ Org] [ Max] for j in range(home_x, home_x + dsp_col_max): display.text(font2, moji.strFixR_n(Voice_Com_Edit[COM][j], 3), \ 16*CELL_WIDTH + 16*CELL_WIDTH*(j - home_x), 32, st7789.CYAN) # 行タイトルの表示 [BO ] [LFO] [ALG] for i in range(home_y, home_y + dsp_row_max): display.text(font2, moji.strFixL_n(Voice_Com_Edit[i][CM0], 3), 0, \ 32*2 + 32*(i - home_y), st7789.CYAN) for y in range(home_y, home_y + dsp_row_max): for x in range(home_x, home_x + dsp_col_max): display.text(font2, moji.int2str_n(Voice_Com_Edit[y][x], 3), \ 16*CELL_WIDTH + 16*CELL_WIDTH*(x - home_x), \ 32*2 + 32*(y - home_y), st7789.WHITE) # 現在のセル表示を [RED] にする display.text(font2, moji.int2str_n(Voice_Com_Edit[cell_y][cell_x], 3), \ 16*CELL_WIDTH + 16*CELL_WIDTH*(cell_x - home_x), \ 32*2 + 32*(cell_y - home_y), st7789.RED, st7789.WHITE) cell_y_old = cell_y # 行 cell_x_old = cell_x # 列 home_y_old = home_y # 表示エリアホーム y home_x_old = home_x # 表示エリアホーム x cnt_y = (cell_y *100 +50) *-1 data_cnt = 0 btn_l = 0 btn_r = 0 btn_c = 0 cnt_x = 0 # cnt_y = 0 cnt_s = 0 # data_cnt_old = data_cnt btn_l_old = btn_l btn_r_old = btn_r btn_c_old = btn_c cnt_x_old = cnt_x cnt_y_old = cnt_y cnt_s_old = cnt_s # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) data_cnt_old = 999 # mode:0 編集セルを選択するモード、mode:1 選択したセルを編集するモード mode = 0 # マウスの左クリックが押された状態(離れたことを確認できていない状態) busy_l = 0 # マウスの右クリックが押された状態(離れたことを確認できていない状態) busy_r = 0 # 直前の選択を表示する cell_y = ComNum cnt_y = (cell_y *100 +50) *-1 # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) data_cnt_old = 999 # mode == 3 で終了 while mode != 3: flag_M = 0 # Mouse flag_K = 0 # Key msg_f = '' # FunctionKey ############################################################## # PS2Keyboard入力があれば、マウス情報に置き換える。 # マウスのクリック押下時点で反応するように設計したのだ。 # 入力は Arw_U:「↑」, Arw_D:「↓」, Arw_L:「←」, Arw_R:「→」 # 「Enter」 「Esc」キーのみ受け付ける ############################################################## # 1文字・記号入力なら if ps2key.rcv_size() == 1: ps2key_data = ps2key.get_code() # 数字キーならば if ps2key_data >= '0' and ps2key_data <= '9': flag_K = 1 ####################################################### # 右詰め表示で編集 # Voice_Com_Editの場合は、1桁なので上書き cnt_s = int(ps2key_data) # マウス座標・現在地を更新 スクロールカウント ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) ####################################################### # ファンクション・キー入力なら elif ps2key.rcv_size() > 1: ps2key_data = ps2key.get_code() flag_K = 1 # .lower 文字の一致を取りやすくするため、shift low で照合 if ps2key_data.lower() == "PrtSc".lower(): msg_f = "PrtSc = SysRq" print(msg_f) ################################################### SysRq = 1 # PrtSc を SysRq として扱う ################################################### # break elif ps2key_data.lower() == "BS".lower(): msg_f = "BS" print(msg_f) # 末尾を消すということは、1/10 cnt_s = int(cnt_s / 10) flag_M = 1 elif ps2key_data.lower() == "Del".lower(): msg_f = "Del" print(msg_f) # 0クリア cnt_s = 0 flag_M = 1 elif ps2key_data.lower() == "Esc".lower(): msg_f = "Esc" print(msg_f) return ComNum elif ps2key_data.lower() == "Enter".lower(): msg_f = "Enter" print(msg_f) flag_M = 1 elif ps2key_data.lower() == "Arw_U".lower(): # ↑ msg_f = "ArwU" print(msg_f) cnt_y += 100 print(f'{cnt_y} cnt_y += 100') flag_M = 1 elif ps2key_data.lower() == "Arw_D".lower(): # ↓ msg_f = "ArwD" print(msg_f) cnt_y -= 100 print(f'{cnt_y} cnt_y -= 100') flag_M = 1 elif ps2key_data.lower() == "Arw_L".lower(): # ← msg_f = "ArwL" print(msg_f) cnt_x -= 100 print(f'{cnt_x} cnt_x -= 100') flag_M = 1 elif ps2key_data.lower() == "Arw_R".lower(): # → msg_f = "ArwR" print(msg_f) cnt_x += 100 print(f'{cnt_x} cnt_x += 100') flag_M = 1 elif ps2key_data == "F1": msg_f = "F1" print(msg_f) if flag_M == 1: # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) # マウス入力なら else: # マウスからボタンクリックの情報、座標情報を取得する (data_cnt, btn_l, btn_r, btn_c, cnt_x, cnt_y, cnt_s) = ps2mus_rcv.get_data() # PS2 Mouse に入力があるなら if data_cnt_old != data_cnt: data_cnt_old = data_cnt flag_M = 1 # マウス入力か、キー入力があれば if flag_M == 1 or flag_K == 1: # セル・選択モード #################################################### if mode == 0: # 左クリック・ボタンが押されたままでないなら if busy_l == 0: # 列 Voice_Com_Edit Voice_Com_COL_NUM if cnt_x > VOICE_COM_COL_NUM *100 -1: cnt_x = VOICE_COM_COL_NUM *100 -1 elif cnt_x < 100: cnt_x = 100 # 行 Voice_Com_Edit Voice_Com_ROW_NUM if cnt_y < (VOICE_COM_ROW_NUM *100 -1) *-1: cnt_y = (VOICE_COM_ROW_NUM *100 -1) *-1 elif cnt_y > -100: cnt_y = -100 # セル・アドレスの表示データ生成 [x:1-9, y:1-9] # 列 cell_x = int(cnt_x / 100) # 行 cell_y = int((-1 * cnt_y) / 100) if cell_x_old < cell_x: cell_x_direc = 1 elif cell_x_old > cell_x: cell_x_direc = -1 else: cell_x_direc = 0 if cell_y_old < cell_y: cell_y_direc = 1 elif cell_y_old > cell_y: cell_y_direc = -1 else: cell_y_direc = 0 # Table表示位置を調整(home_x home_y の計算) ############################################################ if cell_x_direc == 1 and cell_x > (home_x + (dsp_col_max -1)): home_x = cell_x - (dsp_col_max -1) elif cell_x_direc == -1 and cell_x < home_x: home_x = cell_x ############################################################ if cell_y_direc == 1 and cell_y > (home_y + (dsp_row_max -1)): home_y = cell_y - (dsp_row_max -1) elif cell_y_direc == -1 and cell_y < home_y: home_y = cell_y ############################################################ # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) # セル・アドレス表示 [A99] adr_dsp = chr(64 + cell_x) + moji.int2str_n(cell_y, 2) display.text(font2, adr_dsp, 16*0, 32*1, st7789.CYAN) # セル値を、左上隅に表示 val = Voice_Com_Edit[cell_y][cell_x] display.text(font2, moji.int2str_n(val, 6), \ 16*0, 32*0, st7789.RED) # home_x または home_y に変化があれば、Voice_Com_Edit を再表示する if home_x_old != home_x or home_y_old != home_y: for y in range(home_y, home_y + dsp_row_max): for x in range(home_x, home_x + dsp_col_max): display.text(font2, moji.int2str_n(Voice_Com_Edit[y][x], 3), \ 16*CELL_WIDTH + 16*CELL_WIDTH*(x - home_x), \ 32*2 + 32*(y - home_y), st7789.WHITE) # home_x に変化があれば、列タイトルの再表示 [ edt] [ Org] [ Max]... if home_x_old != home_x: for j in range(home_x, home_x + dsp_col_max): display.text(font2, moji.strFixR_n(Voice_Com_Edit[COM][j], 3), \ 16*CELL_WIDTH + 16*CELL_WIDTH*(j - home_x), 32, st7789.CYAN) # home_y に変化があれば、行タイトルの再表示 [BO ] [LFO] [ALG]... if home_y_old != home_y: for i in range(home_y, home_y + dsp_row_max): display.text(font2, moji.strFixL_n(Voice_Com_Edit[i][CM0], 3), 0, \ 32*2 + 32*(i - home_y), st7789.CYAN) # 現在セル座標に変化があれば if cell_x_old != cell_x or cell_y_old != cell_y: # 直前のセル表示を [WHITE] に戻す display.text(font2, moji.int2str_n(Voice_Com_Edit[cell_y_old][cell_x_old], 3), \ 16*CELL_WIDTH + 16*CELL_WIDTH*(cell_x_old - home_x_old), \ 32*2 + 32*(cell_y_old - home_y_old), st7789.WHITE) # 現在のセル表示を 文字色:[RED] 背景色[WHITE] にする display.text(font2, moji.int2str_n(Voice_Com_Edit[cell_y][cell_x], 3), \ 16*CELL_WIDTH + 16*CELL_WIDTH*(cell_x - home_x), \ 32*2 + 32*(cell_y - home_y), st7789.RED, st7789.WHITE) # セルの値を、左上隅に表示 val = Voice_Com_Edit[cell_y][cell_x] display.text(font2, moji.int2str_n(val, 6), \ 16*0, 32*0, st7789.RED) # セル座標を更新 cell_y_old = cell_y cell_x_old = cell_x home_y_old = home_y home_x_old = home_x # 左クリック・ボタンが押されていたら、セル・編集モードに移行 # if btn_l == 1: if btn_l == 1 or msg_f == 'Enter': if btn_l == 1: busy_l = 1 elif msg_f == 'Enter': msg_f = '' # 編集中の座標を保存しておく cell_x_now = cell_x cell_y_now = cell_y # 編集中のセル値を、左上隅に表示 if cell_x_now == 1: # 変更前の内容を表示 ComNum = Voice_Com_Edit[cell_y_now][cell_x_now] print(f'Sub OK Voice_Com_Edit {VoiceComName[cell_y_now]} = ', ComNum, end = '') val = Voice_Com_Edit[cell_y_now][cell_x_now] display.text(font2, moji.int2str_n(val, 6), \ 16*0, 32*0, st7789.RED, st7789.WHITE) else: display.text(font2, moji.strFixL_n("N/A", 6), \ 16*0, 32*0, st7789.RED, st7789.WHITE) # menu_bar[1] = " OK", menu_bar[2] = " BK" menu_bar[3] = " DF" # [ OK]の位置にセル座標を合わせる cnt_x = 250 cnt_y = 50 cnt_s = val #マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) # セル編集モードに移行 mode = 1 # 左クリック・ボタンが離れるのを待って elif btn_l == 0: busy_l = 0 # 新たに右クリック・ボタンが押されていたら if busy_r == 0 and btn_r == 1: busy_r = 1 mode = 2 # セル・編集モード #################################################### elif mode == 1: # 左クリック・ボタンが押されたままでないなら if busy_l == 0: # 列 print(f'mode == 1 cnt_x = {cnt_x}') if cnt_x > (menu_num +1) *100 -1: # cnt_x = cnt_x > (menu_num +1) *100 -1 cnt_x = (menu_num +1) *100 -1 elif cnt_x < 200: cnt_x = 200 # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) # セル・アドレスの生成 # menu_bar[1] = " OK", menu_bar[2] = " BK" menu_bar[3] = " DF" # 列 1, 2, 3 cell_x_edt = int(cnt_x / 100) -1 # 行 cell_y_edt = 0 # メニューバー表示を編集用にする(選択メニューの背景をマゼンタ色) for i in range(1, menu_num): if i == cell_x_edt: display.text(font2, menu_bar[i], \ 16*6 + 16*3*(i - 1), 0, st7789.GREEN, st7789.MAGENTA) else: display.text(font2, menu_bar[i], \ 16*6 + 16*3*(i - 1), 0, st7789.GREEN) # 編集中のセル値を、左上隅に表示 if cell_x_now == 1: # 最大値を超えない if cnt_s > Voice_Com_Edit[cell_y_now][CM3]: val = Voice_Com_Edit[cell_y_now][CM3] # マイナス値はなし elif cnt_s < 0: val = 0 else: val = cnt_s ###################################################### # display.text(font2, moji.int2str_n(val, 6), \ # 16*0, 32*0, st7789.RED, st7789.WHITE) display.text(font2, moji.int2str_n(val, 6), \ 16*0, 32*0, st7789.RED, backColor) ###################################################### # 編集対象外のセルなら #N/A を表示 else: display.text(font2, moji.strFixL_n("#N/A", 6), \ 16*0, 32*0, st7789.RED, st7789.WHITE) # 左クリック・ボタンが押されていたら、 # Voice_Com_Edit に代入して、セル・編集モードを終了 if btn_l == 1 or msg_f == 'Enter': if btn_l == 1: busy_l = 1 elif msg_f == 'Enter': msg_f = '' # 左クリックしたセルが menu_bar[1] = " OK" なら # 編集中のセル値を、Voice_Com_Edit に代入して表示する if cell_x_edt == 1: Voice_Com_Edit[cell_y_now][cell_x_now] = val display.text(font2, moji.int2str_n(Voice_Com_Edit[cell_y_now][cell_x_now], 3), \ 16*CELL_WIDTH + 16*CELL_WIDTH*(cell_x_now - home_x), \ 32*2 + 32*(cell_y_now - home_y), st7789.RED, st7789.WHITE) ComNum = Voice_Com_Edit[cell_y_now][cell_x_now] print(f' to {VoiceComName[cell_y_now]} = ', ComNum) # 左クリックしたセルが menu_bar[2] = " BK" なら、元の値に戻す elif cell_x_edt == 2: val = Voice_Com_Edit[cell_y_now][cell_x_now] # 左クリックしたセルが menu_bar[3] = " DF" なら # Voice_Com_Edit の Orig 値を、Voice_Com_Edit に代入して表示する elif cell_x_edt == 3 and cell_x_now == 1: # Voice_Com_Edit[cell_y_now][cell_x_now] = Voice_Com_Edit[cell_y_now][cell_x_now +4] Voice_Com_Edit[cell_y_now][cell_x_now] = Voice_Com_Edit[cell_y_now][CM2] val = Voice_Com_Edit[cell_y_now][cell_x_now] display.text(font2, moji.int2str_n(Voice_Com_Edit[cell_y_now][cell_x_now], 3), \ 16*CELL_WIDTH + 16*CELL_WIDTH*(cell_x_now - home_x), \ 32*2 + 32*(cell_y_now - home_y), st7789.RED, st7789.WHITE) # 編集が終了したら、セル・選択モードに戻る # メニューバー表示を元に戻す for i in range(1, menu_num): display.text(font2, menu_bar[i], \ 16*6 + 16*3*(i - 1), 0, st7789.GREEN) # 編集中の左上隅に表示を元に戻す display.text(font2, moji.int2str_n(val, 6), \ 0, 0, st7789.RED) # 編集前の座標にセット cnt_x = cell_x_now *100 + 50 cnt_y = (cell_y_now *100 + 50) * -1 cnt_s = val # マウス座標・現在地を更新 ps2mus_rcv.set_cnt(cnt_x, cnt_y, cnt_s) # セル選択モードに移行 mode = 0 # 左クリック・ボタンが押されたままなら、離れるまで待つ elif btn_l == 0: busy_l = 0 # 編集終了・確認 モード ############################################### # mode:0 で、右クリック・ボタンが押されていたらここにくる elif mode == 2: # 右クリック・ボタンが押されたままなら、離れるまで待つ if btn_r == 0: busy_r = 0 print('right button clicked ... End of editVoiceCom.') mode = 3 # End of editVoiceCom else: return ComNum ####################################################################### # Main Python Start ####################################################################### print('Main Python Start') print() print('uos.uname = ', uos.uname()) print() print('Main Python Start メモリ使用状況') micropython.mem_info() # st7789 spi Setting # 20240321 polarity=1, phase=1 OK spi1 = SPI(1, baudrate=40000000, polarity=1, phase=1) print('spi1 = ', spi1) print() # st7789 display Setting display = st7789.ST7789(spi1, disp_width, disp_width,\ reset=Pin(st7789_res, Pin.OUT),\ dc=Pin(st7789_dc, Pin.OUT),\ xstart=0, ystart=0, rotation=0) # 変数の初期設定 ps2kb_timer = 0 ps2key_buff = "" # init_ps2mouse() ps2mus_cmd.init_ps2cmd_1() ps2mus_rcv.init_ps2rcv_1(MUS_PIO_FRQ) ps2mus_cmd.init_ps2cmd_2() ps2mus_rcv.init_ps2rcv_2(MUS_PIO_FRQ) ###################################### # PS2mouseCommnd PIO プログラムの削除 # いろいろ試したが、唯一うまく動作する方法 rp2.PIO(0).remove_program() ###################################### # PS2KeyRecive ドライバ および StateMachine の設定と起動 ---------------------- # PS2KeyRecive ドライバの初期化 ps2key = PS2KeyRecive(kb_freq, kb_data, kb_clock) # PS2KeyRecive ドライバの起動 ps2key.set_active() # 連続して文字入力がある場合・・・1Frame Time の設定 # TenKey 機能ボタンなどでは、最短で 2.5ms周期くらいで文字データが発生する。 # 5ms:(10MHz*50ms*1000us)/PIOクロック数 2 time_out = int((KB_PIO_FRQ * 5) / (1000 * 2)) ps2key.put_timer(time_out) tone_num = 0 ComNum = 0 OpNum = 0 print('Main Loop メモリ使用状況') micropython.mem_info() # PrintMtxLED 画面消去 ymf825.uart_write_multi('%IJ.CLS\n') #---------------------------------------------------------------------------- # MicroPython Main Loop while True: SysRq = 0 while SysRq == 0: # Tone(音色:楽器)指定 # tone_num = ymf825.selectTone(tone_num)をベースに内蔵した # SysRq(PrtSc)キーが押される、または、RFID-CARD、TAGが置かれても終了する dsp_msg = 'Select Tone' tone_num = selectTone(dsp_msg, tone_num) if SysRq == 0: # SysRq(PrtSc)キー押下による終了ではなく # tone_num が指定された=中心的な処理ルーチン # 指定された ToneEdit リストを編集用リスト(VoiceEdit)にコピー ymf825.copyTone2_Voice(tone_num) # Edit Voice Common parameter ComNum = editVoiceCom(ymf825.Voice_Com_Edit, ymf825.VoiceComName, tone_num, ComNum) # Edit Voice Operator Parameter OpNum = editVoiceOp(ymf825.Voice_Op_Edit, ymf825.VoiceOpName, tone_num, OpNum) # 編集結果(VoiceEdit)をシリアル出力 ymf825.serialOutVoiceData(tone_num) # 編集結果(VoiceEdit)を ToneEdit リストにコピーする(戻す) ymf825.copyVoice2_Tone(tone_num) # ToneNameの編集と ToneName の更新 old_name = ymf825.ToneName[tone_num] # 20240716 Change into main # new_name = editToneName(old_name) for i in range(7): EditMsg[i][0] = st7789.WHITE EditMsg[i][1] = '' EditMsg[0][0] = st7789.YELLOW EditMsg[0][1] = 'Edit Tone Name' EditMsg[1][0] = st7789.MAGENTA EditMsg[1][1] = 'Before Name' EditMsg[2][0] = st7789.WHITE EditMsg[2][1] = old_name EditMsg[3][0] = st7789.CYAN EditMsg[3][1] = 'After Name' # Line_n は、編集行 line_n = 4 new_name = editName(EditMsg, line_n, old_name) print('new_name =', new_name) # YMF825 ToneName の更新 ################################################ ymf825.ToneName[tone_num] = new_name ################################################ # ST7789表示用 ToneMenuTable の更新 ################################################ ymf825.ToneMenuTable[tone_num +1][1] = new_name ################################################ # Toneデータを、転送用に準備 # Tone_Com_Edit, Tone_Op_Edit を ASCII2文字の send_buffに変換 ymf825.takeOutToneData(tone_num) # UARTに送信する前に、文字列の先頭と末尾から、スペースを取り除く # RFID用に編集した名残で末尾にスペースが16個入っているのを取り除く # PicoSub側でも、受信前に空読みするよう対応済みだが ymf825.send_buff = ymf825.send_buff.strip() # PicoSub に ToneData を送信 setToneDataPicoSub(tone_num) else: if SysRq == 1: # Keyboard SysRq [PrtSc] キー押下なら 全メニューを表示して選択する dsp_msg = 'Select Function' num = 1 # 1:Edit Tone is Default num = selectFunction(dsp_msg, num) print(f'{num} = selectFunction(dsp_msg, num)') # メニュー画面で選択された num に応じた Functionを実行 if num == 1: # 1:Edit Tone pass elif num == 2: # 2:Read RFID for i in range(7): AlarmMsg[i] = '' AlarmMsg[0] = 'Set RFID' AlarmMsg[1] = ' CARD or TAG' # MFRC522(13.56MHz)RFID にタグやカードが置かれるまで待つ OnOff = False # On:False Off:True waitTag(AlarmMsg, OnOff) # RFID カード または タグを読み込み rcv_buff に格納 print('(stat, rcv_size, headerID, tone_name, rcv_tone) = midi.readRfidTone()') print('4秒以内に RFID-CAD または RFID-TAG を置いてください。') midi.tag_read_timer = midi.NO_TAG_TIME (stat, rcv_size, headerID, tone_name, rcv_tone) = midi.readRfidTone() print(f'stat={stat} rcv_size={rcv_size} headerID={headerID} tonename={tone_name}') if stat == RFID_OK: # RFID 読み取りOK if headerID == '0': # headerID == '0': Tone Data # SubMenuを表示して取り込むか、キャンセルかを選択する for i in range(7): SubMenu[i] = '' SubMenu[0] = 'RFID Data is...' SubMenu[1] = '0:ToneData' SubMenu[2] = tone_name SubMenu[4] = 'Select Action' SubMenu[5] = 'TakeIn Cancel' SubMenu[6] = '=Enter =Esc' if selectSubMenu(SubMenu): # Enter:True なら # RFID から読み込んだ ToneData をどの場所に保存するか。 print('tone_num =', tone_num) print(f'tone_num = selectTone(dsp_msg, {tone_num})') # 20240710 Change # tone_num = ymf825.selectTone(tone_num) から変更 dsp_msg = 'Save Tone As...' tone_num = selectTone(dsp_msg, tone_num) print(f'{tone_num} =selectTone(dsp_msg, tone_num)') #################################################### ymf825.rcv_tone = rcv_tone #################################################### # 指定された保存場所に ToneData を保存する print(f'ymf825.rcv_tone[0] = {tone_num}') #################################################### ymf825.rcv_tone[0] = tone_num ymf825.takeInToneData() #################################################### # ToneNameの更新、ST7789表示用 ToneMenuTable の更新 #################################################### ymf825.ToneName[tone_num] = tone_name ymf825.ToneMenuTable[tone_num +1][1] = tone_name #################################################### # MFRC522(13.56MHz)RFID にタグやカードが取り除かれるまで待つ for i in range(7): AlarmMsg[i] = '' AlarmMsg[0] = 'Read End RFID' AlarmMsg[2] = 'Remove RFID' AlarmMsg[3] = ' CARD or TAG' OnOff = True # On:False Off:True waitTag(AlarmMsg, OnOff) else: # Esc:False なら pass # 何もせず終了 else: # headerID=='0' 以外の場合 # MIDI Data ? # ToneTable ? pass # 組み込む必要あり elif num == 3: # 3:Write RFID for i in range(7): AlarmMsg[i] = '' AlarmMsg[0] = 'Set RFID' AlarmMsg[1] = ' CARD or TAG' # MFRC522(13.56MHz)RFID にタグやカードが置かれるまで待つ OnOff = False # On:False Off:True waitTag(AlarmMsg, OnOff) # RFID カード または タグを読み込み rcv_buff に格納 print('(stat, rcv_size, headerID, tone_name, rcv_tone) = midi.readRfidTone()') print('4秒以内に RFID-CAD または RFID-TAG を置いてください。') midi.tag_read_timer = midi.NO_TAG_TIME (stat, rcv_size, headerID, tone_name, rcv_tone) = midi.readRfidTone() print(f'stat={stat} rcv_size={rcv_size} headerID={headerID} tonename={tone_name}') if stat == RFID_OK: # RFID 読み取りOK # SubMenuを表示して書き込むか、キャンセルかを選択する for i in range(7): SubMenu[i] = '' SubMenu[0] = 'RFID Data is...' SubMenu[4] = 'Select Action' SubMenu[5] = 'WriteOn Cancel' SubMenu[6] = '=Enter =Esc' if headerID == '0': # headerID == '0': Tone Data SubMenu[1] = '0:ToneData' SubMenu[2] = tone_name else: # headerID != '0': Tone Data ではない SubMenu[1] = 'Not ToneData' SubMenu[2] = '' if selectSubMenu(SubMenu): # Enter:True # 書き込みを選択 # Tone_Com_Edit, Tone_Op_Edit を ASCII2文字の send_buffに変換 ymf825.takeOutToneData(tone_num) # send_buff を RFID で書き込み print('midi.writeRfidTone(ymf825.send_buff)') print('ymf825.send_buff size= ', len(ymf825.send_buff)) print('ymf825.send_buff = ', ymf825.send_buff) print('4秒以内に RFID-CAD または RFID-TAG を置いてください。') ######################################################## midi.tag_write_timer = midi.NO_TAG_TIME midi.writeRfidTone(ymf825.send_buff) ######################################################## print('書き込みが完了しました。 RFID-CAD または RFID-TAG を取り除いてください。') # MFRC522(13.56MHz)RFID にタグやカードが取り除かれるまで待つ for i in range(7): AlarmMsg[i] = '' AlarmMsg[0] = 'Write End RFID' AlarmMsg[2] = 'Remove RFID' AlarmMsg[3] = ' CARD or TAG' OnOff = True # On:False Off:True waitTag(AlarmMsg, OnOff) elif num == 4: # 4:Play MIDI # Select Swapping Channel for i in range(7): InputMsg[i] = '' InputMsg[0] = 'Play MIDI' InputMsg[1] = 'Select' InputMsg[2] = ' Channel 1' InputMsg[3] = ' Swap to 1-16' InputMsg[4] = 'No Swapping = 1' prompt_row = 5 prompt = 'ch ? ' ch_asc = '00' ch_asc = inputNum(InputMsg, prompt_row, prompt, ch_asc) print(f'{ch_asc} = inputNum(InputMsg, prompt_row, {prompt}, ch_asc)') ch_num = int(ch_asc) if ch_num < 1 or ch_num > 16: # 1-16 以外が選択されたなら 1(No Swapping)にする。 ch_num = 1 # 1-16が選択されたなら 0-15にする midi_ex_ch = int(ch_num -1) % 16 # 整数を2桁の16進数文字(0-9 A-F)に変換 (asc_Hn, asc_Ln) = moji.int2asc2(midi_ex_ch) # Select GM Musical No for i in range(7): InputMsg[i] = '' InputMsg[0] = 'Play MIDI' InputMsg[1] = "Swapping Ch's" InputMsg[2] = ' TONE is' InputMsg[3] = ' GM No 1-128' InputMsg[4] = ' 0:Grand Piano' prompt_row = 5 prompt = 'GM Tone ? ' gm_asc = '000' # '000':3桁の編集を指示 table = ymf825.GmName gm_asc = inputNumTable(InputMsg, table, prompt_row, prompt, gm_asc) print(f'{gm_asc} = inputNumTable(InputMsg, table, prompt_row, {prompt}, gm_asc)') # 数字文字を整数に変換 gm_num = int(gm_asc) print(f'{gm_num} = int(gm_asc)') if gm_num < 1 or gm_num > 128: # 1-128 以外が選択されたなら 1(Grand Piano)にする。 gm_num = 1 # 1-128が選択されたなら 0-127にする。 # 整数を2桁の16進数文字(0-9 A-F)に変換 (asc_H, asc_L) = moji.int2asc2(gm_num -1) print(f'({asc_H}, {asc_L}) = moji.int2asc2({gm_num -1})') # Program Change 文字列の生成(CnXX) prgChangeB = 'C' + asc_Ln + asc_H + asc_L # Program Change 文字列の生成(CnXX)を MIDI-Outに出力する ########################################## midi.midi_writeB(prgChangeB) ########################################## display.fill(st7789.BLACK) for i in range(7): AlarmMsg[i] = '' AlarmMsg[0] = 'Playing MIDI' # 選択された楽器の名称を表示 AlarmMsg[1] = moji.strFixL_n(ymf825.GmName[gm_num], ST7789_COL_SIZE) AlarmMsg[2] = ' STOP' AlarmMsg[3] = ' =Esc' # colorFont = backColor # def every997ms(t): で決まる色 colorFont = st7789.RED alarm(AlarmMsg) # 「Esc」キー押下で終了 # MIDI-Inから入力、チャンネル変換し MIDI-Outに出力、midi_listに保存する。 ################################################ # midi.midi_frame_timer = midi.NO_VOICE_TIME # 時間制限をなくした midi.midi_play(midi_ex_ch) ################################################ elif num == 5: # 5:'Print DFP Test' # F5 for i in range(7): EditMsg[i][0] = st7789.WHITE EditMsg[i][1] = '' EditMsg[0][0] = st7789.YELLOW EditMsg[0][1] = 'Print DFP' EditMsg[1][0] = st7789.WHITE EditMsg[1][1] = 'PLAY mp3' EditMsg[2][0] = st7789.CYAN EditMsg[2][1] = 'Command ?' # Line_n は、編集行 line_n = 3 old_command = 'PLAY 1 1' new_command = printDFP(EditMsg, line_n, old_command) elif num == 6: # 6:Edt Tn Table ################################################### # ToneTable を使う場面はありや? # この関数は、キーボード対応のみ(マウスは対応できていない) ################################################### # editToneTable() 関数の入力テスト editToneTable(ymf825.ToneTable, ymf825.ToneName) # PicoSub に ToneTable を送信 setToneTablePicoSub() elif num == 7: # 7:Edt Mk Voice midi.playHectOne(HectOne) # Copy HectOnVoice to HectOne # Test 7 syu for i in range(7): # ローマ字 31文字文を連結する voiceRm = HectOneVoice[i][0] + \ HectOneVoice[i][1] + \ HectOneVoice[i][2] + \ HectOneVoice[i][3] + \ HectOneVoice[i][4] # 文字数が不足するとエラーが発生するはず try: pt = 0 for j in range(31): if voiceRm[pt].lower() == 'a'.lower() or \ voiceRm[pt].lower() == 'i'.lower() or \ voiceRm[pt].lower() == 'u'.lower() or \ voiceRm[pt].lower() == 'e'.lower() or \ voiceRm[pt].lower() == 'o'.lower(): HectOne[j +1][1] = voiceRm[pt] pt += 1 elif voiceRm[pt +1].lower() == 'y'.lower() or \ voiceRm[pt +1].lower() == 'h'.lower() or \ voiceRm[pt +1].lower() == 'w'.lower() or \ voiceRm[pt +1].lower() == 's'.lower(): HectOne[j +1][1] = voiceRm[pt] + \ voiceRm[pt +1] + \ voiceRm[pt +2] pt += 3 else: HectOne[j +1][1] = voiceRm[pt] + \ voiceRm[pt +1] pt += 2 except: print('Error In Copy HectOnVoice to HectOne') pass midi.playHectOne(HectOne)

PicoSubプログラム

PicoS_main_py.py

PicoS_main_py.py

############################################################################# # PicoS_main_py.py Programed by Hyodo # PicoS(YMF825)HectOne-026.py をベースに、イコライザー(フィルター)機能を追加 # # MIDI Keyboard(鍵盤)からの入力で和音を実現するため、MONO TONEで # チャンネルを割り振って利用する方法を試行する。 # # PicoMain と PicoSub で分担して処理する。 # このプログラムは、PicoSub用のプログラムである。 # PicoSub には、YMF825のみが接続されており、 # PicoMain とシリアル通信で接続し、PicoMain からシリアルで送られてくる # ToneData と ToneTable を利用して、YMF825 から音出し(演奏)する。 # MIDI信号を受信し、音出し(演奏)ができる。 # # Shunsuke Ohira さんのプログラムを参考にした。 # # YMF825 との接続 # PICO gpio name (pin) (pin) YMF825 # gpio-17 SPI CS (22) ----白---- (1) SS # gpio-19 SPI MOSI (25) ----青---- (2) MOSI # gpio-16 SPI MISO (21) ----緑---- (3) MISO # gpio-18 SPI CLK (24) ----黄---- (4) CLK # GND -----------白---- (5) GND # 5V -----------橙---- (6) VCC # gpio-22 RESET (29) ----紫---- (7) RST # No Connect (8) Audio Jack # # YMF825 Monitor LED-BLUE # gpio-28 (34) --- BLUE-LED Anode --->--- Cathode (GND) # # # PicoMain と PicoSub との接続 # <<< PicoMain >>> <<< PicoSub >>> # gpio-2 respT (4) --紫-- --黄-- (4) respT gpio-2 # X # gpio-3 respR (5) --黄-- --紫-- (5) respR gpio-3 # # gpio-4 TX (6) --緑-- --青-- (6) TX gpio-4 # X # gpio-5 RX (7) --青-- --緑-- (7) RX gpio-5 # # # Aques Talk との接続 # <<< PicoSub >>> <<< Aques Talk ATP3012 >>> # (Pin) (Pin:ATP3012) # gpio-12 spi1_MISO (16) --緑-- (18) MISO # gpio-13 spi1_CE (17) --黃-- (16) /SS # gpio-14 spi1_CKL (19) --紫-- (19) SCK # gpio-15 spi1_MOSI (20) --青-- (17) MOSI # +5V --赤-- VCC # GND --白-- GND # # ############################################################################# from machine import Timer, SPI, UART, Pin import time import utime import uos from micropython import const # 20240919 Add import math import ymf825py as ymf825 import mojipy as moji ############################################################################# # 上記インポートファイルは、予め、Picoに書き込んでおくこと # ymf825py.py : YMF825 接続用 # mojipy.py : 文字処理用 ############################################################################# HectOneVoice = [ # 01 tenchitennno- ['akinotano,','karihonoiono,', 'tomaoarami-,', 'wagakoromodewa,', 'tuyuninuretutu-.'], # 02 jito-tennno- ['harusugite,', 'natukinikerasi,', 'sirotaeno-,', 'koromohosutyo-,', 'amanokaguyama-.'], # 03 kakinomotonmo hitomaro ['asibikino,', 'yamadorinoono,', 'sidariono-,', 'naganagasiyowo,', 'hitorikamunenn-.'], # 04 yamabeno akahito ['tagonoura,', 'utidetemireba,', 'sirotaeno-,', 'fujinotakaneni,', 'yukiwafuritutu-.'], # 05 sarumaru dayu- ['okuyamani,', 'momijikakiwake,', 'nakusikano-,', 'koekikutokizo,', 'akiwakanasiki-.'], # 06 tyuunagionn nakamoti ['kasasagino,', 'wataseruhasini,', 'okusimono-,', 'sirokiomireba,', 'yozofukenikeru-.'], # 07 ['amanohara,', 'furisakemireba,', 'kasuganaru-,', 'mikasanoyamani,', 'idesitukikamo-.'], # 08 ['toppann,', 'ippann,', 'nippann,', 'nissann,', 'hyakuninnissyu.'], # 09 ['', '', '', '', ''], # 10 ['', '', '', '', ''], # 11 ['', '', '', '', ''], # 12 ['', '', '', '', ''], # 13 ['', '', '', '', ''], # 14 ['', '', '', '', ''], # 15 ['', '', '', '', ''], ] # MIDI Keyboard 入力を ch に振り分けるためのリスト TONE_NUM = const(16) ChList = [None] *TONE_NUM # KeyOn 起動で、NoteOff までのタイマー値 note_off_timer = [-1] *TONE_NUM NOTE_OFF_TIME = 4000 # シリアル uart_midi を生成 # MIDI は、31250bpsである。 MIDI_BAUD_RATE = 31250 # uart_midi = UART(1, MIDI_BAUD_RATE,bits=8,parity=None,stop=1,tx=Pin(8),rx=Pin(9)) uart_midi = UART(0, MIDI_BAUD_RATE,bits=8,parity=None,stop=1,tx=Pin(0),rx=Pin(1)) uartRespT = Pin(2, Pin.OUT) # Low TRUE uartRespT.value(1) uartRespR = Pin(3, Pin.IN) # Low TRUE # spi1 Aques Talk MODE=3 spi1 = SPI(1, baudrate=1000000, polarity=1, phase=1, sck=Pin(14), mosi=Pin(15), miso=Pin(12)) aques_cs = Pin(13, Pin.OUT) aques_cs.value(1) # 初期状態:1 # 時分割処理のためのタイマー # global変数定義。関数で利用するグローバル変数は、関数の定義前に書くのが無難。 uart_timer = 0 # UART RTS CTS Timer uartRtsReq = 0 # UART RTS Request 1:Request uartRtsBusy = 0 # UART RTS Busy 1:Busy MONOTONE_TIME = const(8000) # MIDI受信タイマー monoTone_timer = 0 spi_timer = 0 # SPI Time # 10ms 毎に割り込みが発生する なぜか引数 (t) は必須である。 def every10ms(t): global uart_timer, uartRespR, uartRtsReq, uartRtsBusy global monoTone_timer, note_off_timer global spi_timer if uart_timer < 10: uart_timer = 0 else: uart_timer -= 10 if uartRespR.value() == 0 and uartRtsBusy == 0: # PicoMain から送信要求あり uartRtsReq = 1 if monoTone_timer < 10: monoTone_timer = 0 else: monoTone_timer -= 10 if spi_timer < 10: spi_timer = 0 else: spi_timer -= 10 # note_off_timer for ch in range(TONE_NUM): if note_off_timer[ch] < 10 and note_off_timer[ch] >= 0: ymf825.note_off(ch) # 空きチャンネルにする ChList[ch] = 0 note_off_timer[ch] = -1 elif note_off_timer[ch] >= 10: note_off_timer[ch] -= 10 else: # note_off_timer[ch] < 0 pass # 20240416 10ms # 10ms 毎に every1ms を呼び出す呪文。これ以前に callback関数を定義のこと。 # Timer(period=10, mode=Timer.PERIODIC, callback=every1ms) # mode=Timer.PERIODIC は、指定しなくても大丈夫のようだ。 ############################################################################## # Main 直前で起動するするのが無難か # Timer(period=10, callback=every10ms) ############################################################################## # MicroPython説明サイトでは、タイマーの初期化を行っているが、なくても動作する。 # tim = Timer(period=1, mode=Timer.PERIODIC, callback=every1ms) # tim.init(period=1, callback=every1ms) ####################################################################### # HandShake Send RTS & Recive CTS Cross Connection # PicoMain と PicoSub との接続 # <<< PicoMain >>> <<< PicoSub >>> # gpio-2 respT (4) --紫-- --黄-- (4) respT gpio-2 # X # gpio-3 respR (5) --黄-- --紫-- (5) respR gpio-3 # def waitCTS(rts, time, msg, cnt): global uart_timer if rts == 0: # Send RTS On uartRespT.value(0) uart_timer = time # Wait CTS On while uartRespR.value() == 1: # On == Low になるのを待つ if uart_timer == 0: print(msg) uart_timer = time if cnt > 0: cnt -= 1 else: break # Pico Sub と UART 通信できない utime.sleep_ms(10) elif rts == 1: # Send RTS Off uartRespT.value(1) uart_timer = time # Wait CTS Off while uartRespR.value() == 0: # Off == High になるのを待つ if uart_timer == 0: print(msg) uart_timer = time if cnt > 0: cnt -= 1 else: break # Pico Sub と UART 通信できない utime.sleep_ms(10) elif rts == 2: # Sub-Pico で利用 # Send RTS On uartRespT.value(0) uart_timer = time # Wait CTS Off while uartRespR.value() == 0: # Off == High になるのを待つ if uart_timer == 0: print(msg) uart_timer = time if cnt > 0: cnt -= 1 else: break # Pico Sub と UART 通信できない utime.sleep_ms(10) return cnt ####################################################################### # PicoMain からの受信処理 def uartComRcv(): global uartRespR, uartRespT, uartRtsReq, uartRtsBusy, tone_num # 念のために確認 if uartRespR.value() == 0: # 念のためにから読みする ymf825.uart_blank_read() print('RTS arrived from Main-Pico') # Main Pico へ送信可を送出して、送信完了を待つ print('RTS On(0) to Main_Pico') cnt = 3 # RTS On(0) Wait CTS Off(1) = 送信完了 cnt = waitCTS(2, 1000, 'Wait UART data ...', cnt) # uartRespR.value() == 1 を待って if cnt > 0: uartRespT.value(1) # Toneリストを UART から読み込む # utime.sleep_ms(10) try: (rcv_size, headerID, tone_name) = ymf825.uartInTone() except: print('Error : on (rcv_size, headerID, tone_name) = ymf825.uartInTone() : NoneType?') rcv_size = 0 try: if rcv_size > 0: print() print(f'rcv_size={rcv_size} headerID={headerID} tone_name={tone_name}') # headerID == '0' : ToneData ならば if headerID == '0': # 読み込んだ Toneリストを、Tone_Com_Edit と Tone_Op_Edit に格納する # UART から受け取った Toneリストの tone_num を取得 tone_num = ymf825.takeInToneData() # ToneName 更新 ymf825.ToneName[tone_num] = tone_name # ToneMenuTable 更新 ・・・ PicoSubでは利用しないが ymf825.ToneMenuTable[tone_num +1][1] = tone_name # 内容をシリアル出力する # ymf825.serialOutToneData(tone_num) ymf825.copyTone2_Voice(tone_num) ymf825.serialOutVoiceData(tone_num) ymf825.init_tone() print('受信した音源で発音テスト(ToneTable の登録に従って音出し)') print('DoMiSo', tone_num, ymf825.ToneName[tone_num], 'Octave:4') ymf825.keyon5(0, 4, ymf825.KEY_C, 40, tone_num) utime.sleep_ms(1000) ymf825.keyon5(1, 4, ymf825.KEY_E, 40, tone_num) utime.sleep_ms(1000) ymf825.keyon5(2, 4, ymf825.KEY_G, 40, tone_num) utime.sleep_ms(1000) ymf825.keyoff(0) ymf825.keyoff(1) ymf825.keyoff(2) # headerID == '1' : ToneTable ならば elif headerID == '1': # ToneTable に格納 print('ymf825.takeInToneTable()') ymf825.takeInToneTable() # headerID == '%' : PrintDFP Command ならば elif headerID == '%': # PrintDFP Command を uart に送信 print('ymf825.uart_write_multi(PrintDFP Command)') ymf825.uart_write_multi(tone_name) except TypeError: print('uartComRcv(): rcv_size > 0:') else: print() print('CTS Off(1) signal not comming from Main-Pico') print('ymf825.uartInTone() was not performed !!!') print('Check the connection with Main-Pico') print() # Send RTS Off uartRespT.value(1) utime.sleep_ms(1000) else: # print('RTS is not arrived') uartRespT.value(1) utime.sleep_ms(10) ####################################################################### # note_off_timer[0-15] を -1:TimeOver にする def init_note_off_timer(): global note_off_timer for ch in range(TONE_NUM): # ch 0-15 note_off_timer[ch] = -1 # -1:TimeOver ####################################################################### def check_and_set_ChList(kk): global ChList try: # すでに利用中の kk であるか? ch = ChList.index(kk) # エラーでなければ(見つかったなら)、その ch をそのまま利用する。 ############################# # 一旦、ノートオフにする print(f' note off ch = {ch}') ymf825.note_off(ch) utime.sleep_ms(50) ############################# return ch except ValueError: # エラーならば(利用中ではないなら)、空き ch を探す。 try: ch = ChList.index(0) # チャンネルを kk で使用中にする ChList[ch] = kk return ch except ValueError: return 16 ####################################################################### # ChList は整数リストとする # ChList(整数リスト)を 0クリアする def initChList(): global ChList # 0:空きチャンネル for ch in range(TONE_NUM): # ch 0-15 ChList[ch] = 0 ####################################################################### # ChList は整数リストとする # ChList(整数リスト)から、最初に見つかった 0 の位置(空きインデックス)を返す # 見つからなければ 16を返す def setChList(kk): global ChList try: ch = ChList.index(0) # チャンネルを kk で使用中にする ChList[ch] = kk return ch except ValueError: return 16 ####################################################################### # ChList は整数リストとする # ChList(整数リスト)から、最初に見つかった kk の位置(インデックス)を返す # 見つからなければ 16を返す def resetChList(kk): global ChList try: ch = ChList.index(kk) # 空きチャンネルに戻す ChList[ch] = 0 return ch except ValueError: return 16 ####################################################################### def allNoteOff(): for ch in range(16): # ch 0-15 ymf825.note_off(ch) utime.sleep_ms(10) ####################################################################### # note off = 8n kk vv or 9n kk 00 # note on = 9n kk vv # midi_rcvbuff から note on, note off 情報を抽出して ymf825に送る。 def noteYmf825(midi_rcvbuff): print('noteYmf825', end='') ch = moji.asc2int(midi_rcvbuff[0:2]) % 16 kk = moji.asc2int(midi_rcvbuff[2:4]) & 0x7F vv = moji.asc2int(midi_rcvbuff[4:6]) & 0x7F # note off if midi_rcvbuff[0] == '8' or (midi_rcvbuff[0] == '9' and vv == 0): print(' note off', ch) ymf825.note_off(ch) # note on # 20240928 Edit elif midi_rcvbuff[0] == '9' and vv != 0: octave = int(kk / 12) -1 key = kk % 12 print(' note on', ch, octave, key, vv) ymf825.note_on(ch, octave, key, vv) ####################################################################### # note on = 9n kk vv のみに対応し、 # note off = 8n kk vv or 9n kk 00 には直接反応せず、一定時間後に対応する。 # midi_rcvbuff から note on, note off 情報を抽出して ymf825に送る。 # midi Keyboard 入力で和音を出すために MONO TONE でチャンネルを振り分ける # 20240927 def noteMonoToneYmf825(midi_rcvbuff): を改造 def noteChaimeModeYmf825(midi_rcvbuff): # PicoMain から受信した tone_num で演奏する global tone_num print('noteKeyboardYmf825', end='') # ch = moji.asc2int(midi_rcvbuff[0:2]) % 16 kk = moji.asc2int(midi_rcvbuff[2:4]) & 0x7F vv = moji.asc2int(midi_rcvbuff[4:6]) & 0x7F if midi_rcvbuff[0] == '8' or (midi_rcvbuff[0] == '9' and vv == 0): # note off # なにもしない pass elif midi_rcvbuff[0] == '9' and vv != 00: # note on # 同一 kk で利用チャンネルを探す、なければ、空きチャンネルを探す ch = check_and_set_ChList(kk) print(f' {ch} = check_and_set_ChList({kk})', end = '') if ch >= 0 and ch < TONE_NUM: # ch 0-15 octave = int(kk / 12) -1 key = kk % 12 print(f' note on ch={ch} octave={octave} key={key} velocity={vv} tone={tone_num}') ymf825.note_onMonoTone(ch, octave, key, vv, tone_num) # タイマーをセット note_off_timer[ch] = NOTE_OFF_TIME else: # ch 0-15 以外なら # 何もしない pass ####################################################################### # MIDIの Program Change情報に基づき、ToneTable を編集する。 # MIDI受信したGM音源(1-128)を ymf825 の音色に変換して、ToneTableに記録する。 def changeToneTable(midi_rcvbuff): print('PROGRAM CHANGE ', end='') ch = moji.asc2int(midi_rcvbuff[0:2]) % 16 gm = moji.asc2int(midi_rcvbuff[2:4]) & 0x7F ymf = ymf825.exGmYmf825.get(gm) # ToneTable の更新 ymf825.ToneTable[ch +1][1] = ymf # ドラム(9)は、無音にする ymf825.ToneTable[9 +1][1] = 16 print('ch = ', ch, 'GM = ', gm, 'YMF825 = ', ymf) ####################################################################### # MIDI 入力を演奏する # Keyboardからの MIDIは、ch=0 だけなので、 MONO TONE にして和音対応する。 # Chaime Mode = Key On で発音し、 Key off 後も一定時間(4秒)持続する。 # 20240717:def midiKeyboardPlayYmf825(): 関数を内蔵して自動切り替えする。 # PicoMain からの送信要求(uartRtsReq)があれば、中断する。 # 20240927:def midiPlayYmf825(): を改造して作成。 def midiPlayChaimeModeYmf825(): global uartRespR, uartRtsReq, uartRtsBusy global monoTone_timer midi_rcvbuff = "" note_Flag = 0 note_Byte = 0 prg_Flag = 0 prg_Byte = 0 status_Flag = 0 # 0:mono_toneでない通常のmidi:MIDI受信した Program Change(音色変更)を利用する # 1:Keyboard:ch分割で和音対応:PicoMainから受信したToneDataを利用する # MIDI信号があれば、1 から 0 に切り替える。 mono_tone = 1 # チャンネルリストを 0 クリア initChList() # ノートオフタイマーを初期設定 init_note_off_timer() # PicoMain から送信要求があれば中断する while uartRtsReq == 0: # MIDI 受信ありなら if uart_midi.any() > 0: try: # uart_midi から、1データ読み込む(decodeなし=bytesデータのまま) midi_data_rx = uart_midi.read(1) except UnicodeError: # 読み込みエラーになるの?なら print("uart_midi.read(1) Error!!") midi_data_rx = "" # 受信した1バイトを2バイトの文字に変換して、シリアル出力する # print(moji.byte2chr2(midi_data_rx), end="") # 受信データを2バイトの文字データにして追加保存する midi_rcvbuff += moji.byte2chr2(midi_data_rx) # MIDI受信が8秒以上なければ、mono_tone = 1 にする ############################################## monoTone_timer = MONOTONE_TIME ############################################## ######################################################################### # note on または note off ならば if ord(midi_data_rx) >= 0x80 and ord(midi_data_rx) <= 0x9F: note_Flag = 1 note_Byte = 1 prg_Flag = 0 status_Flag = 0 # note on または note off のデータならば elif ord(midi_data_rx) < 0x80 and note_Flag == 1: note_Byte += 1 if note_Byte == 3: # データをシリアル出力する print(" >>> midi_rcvbuff = ", midi_rcvbuff) ##################################################### if mono_tone == 1: # mono_tone ###################################### # noteMonoToneYmf825(midi_rcvbuff) noteChaimeModeYmf825(midi_rcvbuff) ###################################### else: # MIDI受信した Program Change(音色変更)を利用した演奏 noteYmf825(midi_rcvbuff) ##################################################### note_Byte = 0 #################### midi_rcvbuff = "" note_Flag = 0 prg_Flag = 0 status_Flag = 0 #################### # Program Change(音色変更)ならば elif ord(midi_data_rx) >= 0xC0 and ord(midi_data_rx) <= 0xCF: mono_tone = 0 prg_Flag = 1 prg_Byte = 1 note_Flag = 0 status_Flag = 0 # Program Change(音色変更)のデータならば elif ord(midi_data_rx) < 0x80 and prg_Flag == 1: prg_Byte += 1 if prg_Byte == 2: # ToneTable を更新する print(" >>> midi_rcvbuff = ", midi_rcvbuff) # ############################## changeToneTable(midi_rcvbuff) ############################## prg_Byte = 0 #################### midi_rcvbuff = "" note_Flag = 0 prg_Flag = 0 status_Flag = 0 #################### # その他の情報ならば、何もしない else: #################### midi_rcvbuff = "" note_Flag = 0 prg_Flag = 0 status_Flag = 0 #################### elif monoTone_timer == 0: # MIDI受信が8秒以上なければ、mono_tone = 1 にする ############################################## mono_tone = 1 ############################################## # while uartRtsReq != 0 PicoMain から送信要求があれば中断する else: print('Recived uartRtsReq !') allNoteOff() uartRtsBusy = 1 return ####################################################################### # MIDI 入力を演奏する # Keyboardからの MIDIは、ch=0 だけなので、 MONO TONE にして和音対応する。 # 20240717:def midiKeyboardPlayYmf825(): 関数を内蔵して自動切り替えする。 # PicoMain からの送信要求(uartRtsReq)があれば、中断する。 def midiPlayYmf825(): global uartRespR, uartRtsReq, uartRtsBusy global monoTone_timer midi_rcvbuff = "" note_Flag = 0 note_Byte = 0 prg_Flag = 0 prg_Byte = 0 status_Flag = 0 # 0:mono_toneでない通常のmidi:MIDI受信した Program Change(音色変更)を利用する # 1:Keyboard:ch分割で和音対応:PicoMainから受信したToneDataを利用する # MIDI信号があれば、1 から 0 に切り替える。 mono_tone = 1 # チャンネルリストを 0 クリア initChList() # PicoMain から送信要求があれば中断する while uartRtsReq == 0: # MIDI 受信ありなら if uart_midi.any() > 0: try: # uart_midi から、1データ読み込む(decodeなし=bytesデータのまま) midi_data_rx = uart_midi.read(1) except UnicodeError: # 読み込みエラーになるの?なら print("uart_midi.read(1) Error!!") midi_data_rx = "" # 受信した1バイトを2バイトの文字に変換して、シリアル出力する # print(moji.byte2chr2(midi_data_rx), end="") # 受信データを2バイトの文字データにして追加保存する midi_rcvbuff += moji.byte2chr2(midi_data_rx) # MIDI受信が8秒以上なければ、mono_tone = 1 にする ############################################## monoTone_timer = MONOTONE_TIME ############################################## ######################################################################### # note on または note off ならば if ord(midi_data_rx) >= 0x80 and ord(midi_data_rx) <= 0x9F: note_Flag = 1 note_Byte = 1 prg_Flag = 0 status_Flag = 0 # note on または note off のデータならば elif ord(midi_data_rx) < 0x80 and note_Flag == 1: note_Byte += 1 if note_Byte == 3: # データをシリアル出力する print(" >>> midi_rcvbuff = ", midi_rcvbuff) ##################################################### if mono_tone == 1: # mono_tone noteMonoToneYmf825(midi_rcvbuff) else: # MIDI受信した Program Change(音色変更)を利用する noteYmf825(midi_rcvbuff) ##################################################### note_Byte = 0 #################### midi_rcvbuff = "" note_Flag = 0 prg_Flag = 0 status_Flag = 0 #################### # Program Change(音色変更)ならば elif ord(midi_data_rx) >= 0xC0 and ord(midi_data_rx) <= 0xCF: mono_tone = 0 prg_Flag = 1 prg_Byte = 1 note_Flag = 0 status_Flag = 0 # Program Change(音色変更)のデータならば elif ord(midi_data_rx) < 0x80 and prg_Flag == 1: prg_Byte += 1 if prg_Byte == 2: # ToneTable を更新する print(" >>> midi_rcvbuff = ", midi_rcvbuff) # ############################## changeToneTable(midi_rcvbuff) ############################## prg_Byte = 0 #################### midi_rcvbuff = "" note_Flag = 0 prg_Flag = 0 status_Flag = 0 #################### # その他の情報ならば、何もしない else: #################### midi_rcvbuff = "" note_Flag = 0 prg_Flag = 0 status_Flag = 0 #################### elif monoTone_timer == 0: # MIDI受信が8秒以上なければ、mono_tone = 1 にする ############################################## mono_tone = 1 ############################################## # while uartRtsReq != 0 PicoMain から送信要求があれば中断する else: print('Recived uartRtsReq !') allNoteOff() uartRtsBusy = 1 return ####################################################################### # note off = 8n kk vv or 9n kk 00 # note on = 9n kk vv # midi_rcvbuff から note on, note off 情報を抽出して ymf825に送る。 # midi Keyboard 入力で和音を出すために MONO TONE でチャンネルを振り分ける def noteMonoToneYmf825(midi_rcvbuff): # PicoMain から受信した tone_num で演奏する global tone_num print('noteKeyboardYmf825', end='') # ch = moji.asc2int(midi_rcvbuff[0:2]) % 16 kk = moji.asc2int(midi_rcvbuff[2:4]) & 0x7F vv = moji.asc2int(midi_rcvbuff[4:6]) & 0x7F # note off if midi_rcvbuff[0] == '8' or (midi_rcvbuff[0] == '9' and vv == 0): # ノート オン したチャンネルを探す ch = resetChList(kk) print(f' {ch} = index_ChList({kk})', end = '') if ch >= 0 and ch <= 16: print(f' note off ch = {ch}') ymf825.note_off(ch) else: pass # note on # 20240928 Edit elif midi_rcvbuff[0] == '9' and vv != 0: # 空きチャンネルを探す ch = setChList(kk) print(f' {ch} = index_ChList(0)', end = '') if ch >= 0 and ch <= 16: octave = int(kk / 12) -1 key = kk % 12 print(f' note on ch={ch} octave={octave} key={key} velocity={vv} tone={tone_num}') ymf825.note_onMonoTone(ch, octave, key, vv, tone_num) else: pass ####################################################################### # MIDI Keyboard からの入力を演奏する # Keyboardは、ch=0 だけなので、 MONO TONE にして和音対応する # PicoMain からの送信要求(uartRtsReq)があれば、中断する。 # 20240928 利用されていない def midiKeyboardPlayYmf825(): global uartRespR, uartRtsReq, uartRtsBusy midi_rcvbuff = "" note_Flag = 0 note_Byte = 0 prg_Flag = 0 prg_Byte = 0 status_Flag = 0 # チャンネルリストを 0 クリア initChList() # PicoMain から送信要求があれば中断する while uartRtsReq == 0: # MIDI 受信ありなら if uart_midi.any() > 0: try: # uart_midi から、1データ読み込む(decodeなし=bytesデータのまま) midi_data_rx = uart_midi.read(1) except UnicodeError: # 読み込みエラーになるの?なら print("uart_midi.read(1) Error!!") midi_data_rx = "" # 受信した1バイトを2バイトの文字に変換して、シリアル出力する # print(moji.byte2chr2(midi_data_rx), end="") # 受信データを2バイトの文字データにして追加保存する midi_rcvbuff += moji.byte2chr2(midi_data_rx) ######################################################################### # note on または note off ならば if ord(midi_data_rx) >= 0x80 and ord(midi_data_rx) <= 0x9F: note_Flag = 1 note_Byte = 1 prg_Flag = 0 status_Flag = 0 # note on または note off 情報情報ならば elif ord(midi_data_rx) < 0x80 and note_Flag == 1: note_Byte += 1 if note_Byte == 3: # データをシリアル出力する print(" >>> midi_rcvbuff = ", midi_rcvbuff) # ################################### # 20240705 Change # noteYmf825(midi_rcvbuff) noteMonoToneYmf825(midi_rcvbuff) ################################### note_Byte = 0 #################### midi_rcvbuff = "" note_Flag = 0 prg_Flag = 0 status_Flag = 0 #################### # Program Change(音色変更)ならば elif ord(midi_data_rx) >= 0xC0 and ord(midi_data_rx) <= 0xCF: prg_Flag = 1 prg_Byte = 1 note_Flag = 0 status_Flag = 0 # Program Change(音色変更)のデータならば elif ord(midi_data_rx) < 0x80 and prg_Flag == 1: prg_Byte += 1 if prg_Byte == 2: # ToneTable を更新する print(" >>> midi_rcvbuff = ", midi_rcvbuff) # ############################## # 20240705 Change # changeToneTable(midi_rcvbuff) ############################## prg_Byte = 0 #################### midi_rcvbuff = "" note_Flag = 0 prg_Flag = 0 status_Flag = 0 #################### # その他の情報ならば、何もしない else: #################### midi_rcvbuff = "" note_Flag = 0 prg_Flag = 0 status_Flag = 0 #################### # while uartRtsReq != 0 PicoMain から送信要求があれば中断する else: print('Recived uartRtsReq !') allNoteOff() uartRtsBusy = 1 return ######################################################################## # Aques Talk Ready Chexk def aquesReady(time, msg): global spi_timer spi_timer = time buff = bytearray(1) aques_cs.value(0) utime.sleep_us(100) spi1.readinto(buff, 0xFF) # 0xFF 送出して1byte 読み込む utime.sleep_us(100) aques_cs.value(1) while buff != '>': # > = Ready になるのを待つ if spi_timer == 0: print('Time Over spi_timer') return spi_timer else: # DEBUG print # print(msg, ' ', end='') # print(f'spi1.read(1) = {buff}') utime.sleep_ms(20) aques_cs.value(0) utime.sleep_us(100) spi1.readinto(buff, 0xFF) # 0xFF 送出して1byte 読み込む utime.sleep_us(100) aques_cs.value(1) return spi_timer ######################################################################## # Aques Talk 発声 # 末尾に改行('\r')を送るので、talk_buffは本文のローマ字だけで良い。 def aquesTalk(talk_buff): if len(talk_buff) > 0: print(talk_buff) # Aques Talk Ready Check cnt = aquesReady(2000, 'Wait Aques Talk Ready') if cnt > 0: aques_cs.value(0) utime.sleep_us(100) for i in range(len(talk_buff)): spi1.write(talk_buff[i]) utime.sleep_us(30) # byte間 20us以上 spi1.write(b'\r') utime.sleep_us(100) aques_cs.value(1) else: print('Error Aques Talk is not Ready in def aquesTalk(talk_buff):') ######################################################################## # Calculate the biquad filter parameters def on_calc_biquad_filter(flt_type, fc_num, qv_num): # Cut off frequency and Q value (fc Hz, YMF825 sampling frequency is always 48000 Hz) fc = float(fc_num) qv = float(qv_num) print(f'fc = {fc}, qx = {qv}') # Zero clear the Fc and Qv if fc < 0.0 or qv < 0.0: print('if fc < 0.0 or qv < 0.0: in def on_calc_biquad_filter(fc_num, qv_num)') return # Calculate the filter parameters if qv < 0.01: qv = 0.01 # print("BIQUAD FILTER:{}, Fc={}, Q={}".format(flt_type, fc, qv)) # 20240919 Change w0 = math.pi * 2 * fc / 48000 alpha = math.sin(w0) / (qv + qv) cosw0 = math.cos(w0) print(f'w0 = {w0}, alpha = {alpha}, cosw0 = {cosw0}') a0 = 1.0 + alpha a1 = cosw0 * 2 / a0 a2 = (alpha - 1.0) / a0 if flt_type == "LPF:FcQ": b0 = (1.0 - cosw0) / (a0 + a0) b1 = (1.0 - cosw0) / a0 b2 = b0 elif flt_type == "HPF:FcQ": b0 = (1.0 + cosw0) / (a0 + a0) b1 = -(1.0 + cosw0) / a0 b2 = b0 elif flt_type == "BPFskt:FcQ": b0 = qv * alpha / a0 b1 = 0 b2 = -b0 elif flt_type == "BPF0db:FcQ": b0 = alpha / a0 b1 = 0 b2 = -b0 elif flt_type == "NOTCH:FcQ": b0 = 1 / a0 b1 = -2 * cosw0 / a0 b2 = b0 elif flt_type == "APF:FcQ": b0 = (1 - alpha) / a0 b1 = -2 * cosw0 / a0 b2 = (1 + alpha) / a0 else: print("UNKNOWN FILTER TYPE.") return # Set parameters print(f'PARMS b0 = {b0}, b1 = {b1} , b2 = {b2}, a0 = {a0}, a1 = {a1}, a2 = {a2}') # return b0, b1, b2, a1, a2 ######################################################################## # FROM ymf825pico.py 757 line~ # Set equalizer # eql:: Equalizer number (0, 1, 2) # ceq#:: ceq-eql-# # def set_equalizer( self, eql, ceq0 = 1.0, ceq1 = 0.0, ceq2 = 0.0, ceq3 = 0.0, ceq4 = 0.0 ): def set_equalizer( eql, ceq0 = 1.0, ceq1 = 0.0, ceq2 = 0.0, ceq3 = 0.0, ceq4 = 0.0 ): # 20240920 Add global equalizer_ceq def decimal_to_binary(n: float, int_digits: int = 3, frac_digits: int = 20): """Convert a decimal number to its binary representation""" # Check if number is negative negative = n < 0 n = abs(n) # Separate the integer part and fractional part int_part = int(n) # 整数部 frac_part = n - int_part # 小数部 # Convert integer part to binary int_binary = bin(int_part).replace("0b", "") while len(int_binary) < int_digits: int_binary = "0" + int_binary # Convert fractional part to binary frac_binary = "" while frac_part > 0 and len(frac_binary) < frac_digits: frac_part *= 2 bit = int(frac_part) if bit == 1: frac_part -= bit frac_binary += "1" else: frac_binary += "0" # If the number is negative, perform two's complement if negative: binary = list(int_binary + frac_binary) for i, bit in enumerate(binary): binary[i] = '0' if bit == '1' else '1' # 1's complement for i in range(len(binary) - 1, -1, -1): # 2's complement if binary[i] == '1': binary[i] = '0' else: binary[i] = '1' break int_binary = "".join(binary[:int_digits]) frac_binary = "".join(binary[int_digits:]) # return f"{'1' if negative else '0'}{int_binary}{frac_binary}" return f"{'0b1' if negative else '0b0'}{int_binary}{frac_binary}" # Make CEQ# bytes data def make_ceq_bytes( ceq_num, ceq ): print(f'ceq_num = {ceq_num}') print(f'ceq = {ceq}') ######################################## ceqB = int(decimal_to_binary(ceq), 2) ######################################## print(f'{ceqB} = int(decimal_to_binary(ceq), 2)') print(f'{ceqB:b} = int(decimal_to_binary({ceq}))') print(f'type(ceqB) = {type(ceqB)}') equalizer_ceq[ceq_num *3 +0] = (ceqB & 0xFF0000) >> 16 # 上位8bit equalizer_ceq[ceq_num *3 +1] = (ceqB & 0x00FF00) >> 8 # 中位8bit equalizer_ceq[ceq_num *3 +2] = ceqB & 0x0000FF # 下位8bit print(f'equalizer_ceq = {equalizer_ceq}') print(f'len(equalizer_ceq) = {len(equalizer_ceq)}') print() # この関数雨の実行開始位置 # Clear CEQ bytes data for b in range(15): # # 20240920 Change # self.equalizer_ceq[b] = 0 equalizer_ceq[b] = 0 # Make CEQ0 bytes data make_ceq_bytes( 0, ceq0 ) make_ceq_bytes( 1, ceq1 ) make_ceq_bytes( 2, ceq2 ) make_ceq_bytes( 3, ceq3 ) make_ceq_bytes( 4, ceq4 ) # Burst write mode and all key notes off # print("EDITOR: YMF825 Burst write mode.") # # 20240920 Change # self.spi_write_byte( 0x08, 0xF6 ) # self.delay(20) # self.spi_write_byte( 0x08, 0x00 ) # ymf825.spi_write_single( 0x08, 0xF6 ) utime.sleep_ms(20) ymf825.spi_write_single( 0x08, 0x00 ) # # Write tone data to YMF825 FIFO. # print("EDITOR: Write sound data to YMF825.") # print("EQUALIZER", eql, ":", list(self.equalizer_ceq)) # # 20240920 Change # self.spi_write( 32 + eql, bytearray(list(self.equalizer_ceq)) ) # # ymf825.spi_write_multi( 32 + eql, bytearray(list(equalizer_ceq)) ) # ymf825.spi_write_multi( 32 + eql, equalizer_ceq ) ######################################################################## def domisoTest(): print('DoMiSo', 'Octave:4') ymf825.note_on(0, 4, ymf825.KEY_C, 40) utime.sleep_ms(1000) ymf825.note_on(1, 4, ymf825.KEY_E, 40) utime.sleep_ms(1000) ymf825.note_on(2, 4, ymf825.KEY_G, 40) utime.sleep_ms(1000) ymf825.note_off(0) ymf825.note_off(1) ymf825.note_off(2) ######################################################################## # ------------------------------------------------------------------------- # Python Main Program # ------------------------------------------------------------------------- # YMF825 Setting ymf825.setup() # Max = 63 # ymf825.set_MasterVolume(32) ymf825.set_MasterVolume(24) ymf825.init_tone() print('初期設定の音源で発音テスト(= ToneTable の登録に従って音出し)') domisoTest() aquesTalk("akuesu,to-ku,kido-simasita.") ''' # 百人一首読み上げテスト aquesTalk("amanohara-,") aquesTalk("furisakemireba-,kasuganaru-,") aquesTalk("mikasanoyamani-,idesitukikamo-.") aquesTalk("a") aquesTalk("ma") aquesTalk("no") aquesTalk("ha") aquesTalk("ra,") aquesTalk("fu") aquesTalk("ri") aquesTalk("sa") aquesTalk("ke") aquesTalk("mi") aquesTalk("re") aquesTalk("ba,") aquesTalk("ka") aquesTalk("su") aquesTalk("ga") aquesTalk("na") aquesTalk("ru,") aquesTalk("mi") aquesTalk("ka") aquesTalk("sa") aquesTalk("no") aquesTalk("ya") aquesTalk("ma") aquesTalk("ni,") aquesTalk("i") aquesTalk("de") aquesTalk("si") aquesTalk("tu") aquesTalk("ki") aquesTalk("ka") aquesTalk("mo.") # Copy HectOnVoice to HectOne # Test 7 syu for i in range(8): # ローマ字 31文字文を連結する voiceRm = HectOneVoice[i][0] + \ HectOneVoice[i][1] + \ HectOneVoice[i][2] + \ HectOneVoice[i][3] + \ HectOneVoice[i][4] # 百人一首読み上げテスト aquesTalk(voiceRm) ''' utime.sleep_ms(3000) equalizer_ceq = bytearray(15) ''' ################################# # 以下は EQ(フィルター)テスト ################################# flt_type = 'LPF:FcQ' flt_fc = 1760 flt_qv = 0.7 # Calculate the biquad filter parameters (b0, b1, b2, a1, a2) = on_calc_biquad_filter(flt_type, flt_fc, flt_qv) # Set equalizer set_equalizer( 0, b0, b1, b2, a1, a2) utime.sleep_ms(1000) domisoTest() flt_type = 'HPF:FcQ' flt_fc = 440 flt_qv = 0.7 # Calculate the biquad filter parameters (b0, b1, b2, a1, a2) = on_calc_biquad_filter(flt_type, flt_fc, flt_qv) # Set equalizer set_equalizer( 1, b0, b1, b2, a1, a2) utime.sleep_ms(1000) domisoTest() flt_type = 'BPFskt:FcQ' flt_fc = 880 flt_qv = 0.7 # Calculate the biquad filter parameters (b0, b1, b2, a1, a2) = on_calc_biquad_filter(flt_type, flt_fc, flt_qv) # Set equalizer set_equalizer( 2, b0, b1, b2, a1, a2) utime.sleep_ms(1000) domisoTest() flt_type = 'LPF:FcQ' flt_fc = 1760 flt_qv = 0.7 # Calculate the biquad filter parameters (b0, b1, b2, a1, a2) = on_calc_biquad_filter(flt_type, flt_fc, flt_qv) # Set equalizer set_equalizer( 2, b0, b1, b2, a1, a2) utime.sleep_ms(1000) domisoTest() # Test Calculate the biquad filter parameters # on_calc_biquad_filter(flt_type, fc_num, qv_num) # (b0, b1, b2, a1, a2) = on_calc_biquad_filter("LPF:FcQ", 880, 4) flt_fc = 500.0 flt_qv = 0.7 for i in range(6): if i == 0: flt_type = 'LPF:FcQ' (b0, b1, b2, a1, a2) = on_calc_biquad_filter(flt_type, flt_fc, flt_qv) # Set equalizer Test set_equalizer( 0, b0, b1, b2, a1, a2) elif i == 1: flt_type = 'HPF:FcQ' (b0, b1, b2, a1, a2) = on_calc_biquad_filter(flt_type, flt_fc, flt_qv) # Set equalizer Test set_equalizer( 0, b0, b1, b2, a1, a2) elif i == 2: flt_type = 'BPFskt:FcQ' (b0, b1, b2, a1, a2) = on_calc_biquad_filter(flt_type, flt_fc, flt_qv) # Set equalizer Test set_equalizer( 0, b0, b1, b2, a1, a2) elif i == 3: flt_type = 'BPF0db:FcQ' (b0, b1, b2, a1, a2) = on_calc_biquad_filter(flt_type, flt_fc, flt_qv) # Set equalizer Test set_equalizer( 0, b0, b1, b2, a1, a2) elif i == 4: flt_type = 'NOTCH:FcQ' (b0, b1, b2, a1, a2) = on_calc_biquad_filter(flt_type, flt_fc, flt_qv) # Set equalizer Test set_equalizer( 0, b0, b1, b2, a1, a2) elif i == 5: # 【注意】強烈な音になる flt_type = 'APF:FcQ' (b0, b1, b2, a1, a2) = on_calc_biquad_filter(flt_type, flt_fc, flt_qv) # Set equalizer Test set_equalizer( 0, b0, b1, b2, a1, a2) utime.sleep_ms(1000) domisoTest() ''' flt_type = 'LPF:FcQ' flt_fc = 5000.0 flt_qv = 0.7 # Calculate the biquad filter parameters (b0, b1, b2, a1, a2) = on_calc_biquad_filter(flt_type, flt_fc, flt_qv) # Set equalizer set_equalizer( 0, b0, b1, b2, a1, a2) utime.sleep_ms(1000) domisoTest() tone_num = 0 # ComNum = 0 # OpNum = 0 uartRtsReq = 0 # PicoMain からの送信要求 uartRtsBusy = 0 # UART COM RECIVE 実行中 ############################################################################## # Main 直前で起動する Timer(period=10, callback=every10ms) ############################################################################## #---------------------------------------------------------------------------- # MicroPython Main Loop while True: if uartRtsReq == 1 and uartRtsBusy == 1: uartComRcv() # PicoMain からの受信処理 uartRtsBusy = 0 # UART COM RECIVE 実行終了 uartRtsReq = 0 # PicoMain からの送信要求 クリア else: # MIDI 受信・演奏をここに ・・・ midiPlayYmf825() # uartRtsReq == 1: PicoMain からの送信要求 あれば、処理を中断し、 # uartRtsBusy = 1: UART COM RECIVE 実行開始 をセットして戻る。 # 中断がない限り戻ってこない。 ############################### # 20240705 Change # midiPlayYmf825() # 20240927 Change midiPlayChaimeModeYmf825() ############################### # midiKeyboardPlayYmf825()

PicoM_Lib

(以下のプログラムは予めPicoMainに書き込んでおいてください)
mfrc522.py

mfrc522.py

from machine import Pin, SPI from os import uname class MFRC522: DEBUG = False OK = 0 NOTAGERR = 1 ERR = 2 REQIDL = 0x26 REQALL = 0x52 AUTHENT1A = 0x60 AUTHENT1B = 0x61 PICC_ANTICOLL1 = 0x93 PICC_ANTICOLL2 = 0x95 PICC_ANTICOLL3 = 0x97 def __init__(self, sck, mosi, miso, rst, cs,baudrate=1000000,spi_id=0): self.sck = Pin(sck, Pin.OUT) self.mosi = Pin(mosi, Pin.OUT) self.miso = Pin(miso) self.rst = Pin(rst, Pin.OUT) self.cs = Pin(cs, Pin.OUT) self.rst.value(0) self.cs.value(1) board = uname()[0] if board == 'WiPy' or board == 'LoPy' or board == 'FiPy': self.spi = SPI(0) self.spi.init(SPI.MASTER, baudrate=1000000, pins=(self.sck, self.mosi, self.miso)) elif (board == 'esp8266') or (board == 'esp32'): self.spi = SPI(baudrate=100000, polarity=0, phase=0, sck=self.sck, mosi=self.mosi, miso=self.miso) self.spi.init() elif board == 'rp2': self.spi = SPI(spi_id,baudrate=baudrate,sck=self.sck, mosi= self.mosi, miso= self.miso) else: raise RuntimeError("Unsupported platform") self.rst.value(1) self.init() def _wreg(self, reg, val): self.cs.value(0) self.spi.write(b'%c' % int(0xff & ((reg << 1) & 0x7e))) self.spi.write(b'%c' % int(0xff & val)) self.cs.value(1) def _rreg(self, reg): self.cs.value(0) self.spi.write(b'%c' % int(0xff & (((reg << 1) & 0x7e) | 0x80))) val = self.spi.read(1) self.cs.value(1) return val[0] def _sflags(self, reg, mask): self._wreg(reg, self._rreg(reg) | mask) def _cflags(self, reg, mask): self._wreg(reg, self._rreg(reg) & (~mask)) def _tocard(self, cmd, send): recv = [] bits = irq_en = wait_irq = n = 0 stat = self.ERR if cmd == 0x0E: irq_en = 0x12 wait_irq = 0x10 elif cmd == 0x0C: irq_en = 0x77 wait_irq = 0x30 self._wreg(0x02, irq_en | 0x80) self._cflags(0x04, 0x80) self._sflags(0x0A, 0x80) self._wreg(0x01, 0x00) for c in send: self._wreg(0x09, c) self._wreg(0x01, cmd) if cmd == 0x0C: self._sflags(0x0D, 0x80) i = 2000 while True: n = self._rreg(0x04) i -= 1 if ~((i != 0) and ~(n & 0x01) and ~(n & wait_irq)): break self._cflags(0x0D, 0x80) if i: if (self._rreg(0x06) & 0x1B) == 0x00: stat = self.OK if n & irq_en & 0x01: stat = self.NOTAGERR elif cmd == 0x0C: n = self._rreg(0x0A) lbits = self._rreg(0x0C) & 0x07 if lbits != 0: bits = (n - 1) * 8 + lbits else: bits = n * 8 if n == 0: n = 1 elif n > 16: n = 16 for _ in range(n): recv.append(self._rreg(0x09)) else: stat = self.ERR return stat, recv, bits def _crc(self, data): self._cflags(0x05, 0x04) self._sflags(0x0A, 0x80) for c in data: self._wreg(0x09, c) self._wreg(0x01, 0x03) i = 0xFF while True: n = self._rreg(0x05) i -= 1 if not ((i != 0) and not (n & 0x04)): break return [self._rreg(0x22), self._rreg(0x21)] def init(self): self.reset() self._wreg(0x2A, 0x8D) self._wreg(0x2B, 0x3E) self._wreg(0x2D, 30) self._wreg(0x2C, 0) self._wreg(0x15, 0x40) self._wreg(0x11, 0x3D) self.antenna_on() def reset(self): self._wreg(0x01, 0x0F) def antenna_on(self, on=True): if on and ~(self._rreg(0x14) & 0x03): self._sflags(0x14, 0x03) else: self._cflags(0x14, 0x03) def request(self, mode): self._wreg(0x0D, 0x07) (stat, recv, bits) = self._tocard(0x0C, [mode]) if (stat != self.OK) | (bits != 0x10): stat = self.ERR return stat, bits def anticoll(self,anticolN): ser_chk = 0 ser = [anticolN, 0x20] self._wreg(0x0D, 0x00) (stat, recv, bits) = self._tocard(0x0C, ser) if stat == self.OK: if len(recv) == 5: for i in range(4): ser_chk = ser_chk ^ recv[i] if ser_chk != recv[4]: stat = self.ERR else: stat = self.ERR return stat, recv def PcdSelect(self, serNum,anticolN): backData = [] buf = [] buf.append(anticolN) buf.append(0x70) #i = 0 ###xorsum=0; for i in serNum: buf.append(i) #while i<5: # buf.append(serNum[i]) # i = i + 1 pOut = self._crc(buf) buf.append(pOut[0]) buf.append(pOut[1]) (status, backData, backLen) = self._tocard( 0x0C, buf) if (status == self.OK) and (backLen == 0x18): return 1 else: return 0 def SelectTag(self, uid): byte5 = 0 #(status,puid)= self.anticoll(self.PICC_ANTICOLL1) #print("uid",uid,"puid",puid) for i in uid: byte5 = byte5 ^ i puid = uid + [byte5] if self.PcdSelect(puid,self.PICC_ANTICOLL1) == 0: return (self.ERR,[]) return (self.OK , uid) def tohexstring(self,v): s="[" for i in v: if i != v[0]: s = s+ ", " s=s+ "0x{:02X}".format(i) s= s+ "]" return s def SelectTagSN(self): valid_uid=[] (status,uid)= self.anticoll(self.PICC_ANTICOLL1) #print("Select Tag 1:",self.tohexstring(uid)) if status != self.OK: return (self.ERR,[]) if self.DEBUG: print("anticol(1) {}".format(uid)) if self.PcdSelect(uid,self.PICC_ANTICOLL1) == 0: return (self.ERR,[]) if self.DEBUG: print("pcdSelect(1) {}".format(uid)) #check if first byte is 0x88 if uid[0] == 0x88 : #ok we have another type of card valid_uid.extend(uid[1:4]) (status,uid)=self.anticoll(self.PICC_ANTICOLL2) #print("Select Tag 2:",self.tohexstring(uid)) if status != self.OK: return (self.ERR,[]) if self.DEBUG: print("Anticol(2) {}".format(uid)) rtn = self.PcdSelect(uid,self.PICC_ANTICOLL2) if self.DEBUG: print("pcdSelect(2) return={} uid={}".format(rtn,uid)) if rtn == 0: return (self.ERR,[]) if self.DEBUG: print("PcdSelect2() {}".format(uid)) #now check again if uid[0] is 0x88 if uid[0] == 0x88 : valid_uid.extend(uid[1:4]) (status , uid) = self.anticoll(self.PICC_ANTICOLL3) #print("Select Tag 3:",self.tohexstring(uid)) if status != self.OK: return (self.ERR,[]) if self.DEBUG: print("Anticol(3) {}".format(uid)) if self.MFRC522_PcdSelect(uid,self.PICC_ANTICOLL3) == 0: return (self.ERR,[]) if self.DEBUG: print("PcdSelect(3) {}".format(uid)) valid_uid.extend(uid[0:5]) # if we are here than the uid is ok # let's remove the last BYTE whic is the XOR sum return (self.OK , valid_uid[:len(valid_uid)-1]) #return (self.OK , valid_uid) def auth(self, mode, addr, sect, ser): return self._tocard(0x0E, [mode, addr] + sect + ser[:4])[0] def authKeys(self,uid,addr,keyA=None, keyB=None): status = self.ERR if keyA is not None: status = self.auth(self.AUTHENT1A, addr, keyA, uid) elif keyB is not None: status = self.auth(self.AUTHENT1B, addr, keyB, uid) return status def stop_crypto1(self): self._cflags(0x08, 0x08) def read(self, addr): data = [0x30, addr] data += self._crc(data) (stat, recv, _) = self._tocard(0x0C, data) return stat, recv def write(self, addr, data): buf = [0xA0, addr] buf += self._crc(buf) (stat, recv, bits) = self._tocard(0x0C, buf) if not (stat == self.OK) or not (bits == 4) or not ((recv[0] & 0x0F) == 0x0A): stat = self.ERR else: buf = [] for i in range(16): buf.append(data[i]) buf += self._crc(buf) (stat, recv, bits) = self._tocard(0x0C, buf) if not (stat == self.OK) or not (bits == 4) or not ((recv[0] & 0x0F) == 0x0A): stat = self.ERR return stat def writeSectorBlock(self,uid, sector, block, data, keyA=None, keyB = None): absoluteBlock = sector * 4 + (block % 4) if absoluteBlock > 63 : return self.ERR if len(data) != 16: return self.ERR if self.authKeys(uid,absoluteBlock,keyA,keyB) != self.ERR : return self.write(absoluteBlock, data) return self.ERR def readSectorBlock(self,uid ,sector, block, keyA=None, keyB = None): absoluteBlock = sector * 4 + (block % 4) if absoluteBlock > 63 : return self.ERR, None if self.authKeys(uid,absoluteBlock,keyA,keyB) != self.ERR : return self.read(absoluteBlock) return self.ERR, None def MFRC522_DumpClassic1K(self,uid, Start=0, End=64, keyA=None, keyB=None): for absoluteBlock in range(Start,End): status = self.authKeys(uid,absoluteBlock,keyA,keyB) # Check if authenticated print("{:02d} S{:02d} B{:1d}: ".format(absoluteBlock, absoluteBlock//4 , absoluteBlock % 4),end="") if status == self.OK: status, block = self.read(absoluteBlock) if status == self.ERR: break else: for value in block: print("{:02X} ".format(value),end="") print(" ",end="") for value in block: if (value > 0x20) and (value < 0x7f): print(chr(value),end="") else: print('.',end="") print("") else: break if status == self.ERR: print("Authentication error") return self.ERR return self.OK

midipy2.py

midipy2.py

############################################################################### # MIDI i/f Programed by Hyodo # midipy2.py # 2024/04/15:Rfid2Midi-542.py の MIDI i/f 部分を分離して # ライブラリとして利用できるように再構成した。 # 20240603:不要となった st7789,ps/2Mouseなどのコマンドを削除して # テスト デバッグ用に if __name__ == '__main__': を用意した。 # # RFID-RC522 と PicoMain との接続 # RFID-RC522 <<< Pico >>> # RST (7) --紫-- (29) gpio-22 # 3V3 (8) --灰-- 3.3V # MOSI (3) --橙-- (24) gpio-19 MOSI # SCK (2) --赤-- (23) gpio-18 MISO # GND (6) --青-- (22) GND # SDA-CE (1) --茶-- (21) gpio-17 CE # MISO (4) --黄-- (20) gpio-16 MISO # IRQ (5) --緑-- pull-up 10K No Connect # # ############################################################################### from machine import Timer, SPI, UART, Pin import utime from micropython import const import mojipy as moji from mfrc522 import MFRC522 ############################################################################### # Thonnyで 当該ファイルを開いて、Pico直下に書き込む。 # mojipy.py : 文字変換用 # mfrc522.py : RFID-RC522 接続用 ############################################################################### # Pocket Miku 用の文字辞書 # ローマ字からコードに変換する exRmPkMiku = {'a':0, 'i':1, 'u':2, 'e':3, 'o':4, 'ka':5, 'ki':6, 'ku':7, 'ke':8, 'ko':9, 'ga':10, 'gi':11, 'gu':12, 'ge':13, 'go':14, 'kya':15, 'kyu':16, 'kyo':17, 'gya':18, 'gyu':19, 'gyo':20, 'sa':21, 'swi':22, 'su':23, 'se':24, 'so':25, 'za':26, 'zwi':27, 'zu':28, 'ze':29, 'zo':30, 'sya':31, 'si':32, 'syu':33, 'sye':34, 'syo':35, 'ja':36, 'ji':37, 'ju':38, 'je':39, 'jo':40, 'ta':41, 'thi':42, 'twu':43, 'te':44, 'to':45, 'da':46, 'dhi':47, 'dwu':48, 'de':49, 'do':50, 'thu':51, 'dhu':52, 'tya':53, 'ti':54, 'tyu':55, 'tye':56, 'tyo':57, 'tsa':58, 'tsi':59, 'tu':60, 'tse':61, 'tso':62, 'na':63, 'ni':64, 'nu':65, 'ne':66, 'no':67, 'nya':68, 'nyu':69, 'nyo':70, 'ha':71, 'hi':72, 'hu':73, 'he':74, 'ho':75, 'ba':76, 'bi':77, 'bu':78, 'be':79, 'bo':80, 'pa':81, 'pi':82, 'pu':83, 'pe':84, 'po':85, 'hya':86, 'hyu':87, 'hyo':88, 'bya':89, 'byu':90, 'byo':91, 'pya':92, 'pyu':93, 'pyo':94, 'fa':95, 'fi':96, 'fu':97, 'fe':98, 'fo':99, 'ma':100, 'mi':101, 'mu':102, 'me':103, 'mo':104, 'mya':105, 'myu':106, 'myo':107, 'ya':108, 'yu':109, 'yo':110, 'ra':111, 'ri':112, 'ru':113, 're':114, 'ro':115, 'rya':116, 'ryu':117, 'ryo':118, 'wa':119, 'wi':120, 'we':121, 'wo':122, 'nn':123, 'm':124, 'n':125, 'j':126, } TONE_HEADER = const(16) TONE_DATA_ROW_NUM = const(42) TONE_TABLE_ROW_NUM = const(17) TONE_NUM = const(16) TONE_ANL_COL = const(10) RFID_BLOCK_SIZE = const(16) # RFID のブロックサイズ RFID_ADDRESS_MAX = const(45) # RFID の最大アドレス # プログラムチェンジ(音色変更)記録用のポインタ ChTnPt = [None] * TONE_NUM for i in range(TONE_NUM): ChTnPt[i] = 0 # プログラムチェンジ(音色変更)記録用のリスト ChTnList = [None] * TONE_NUM for i in range(TONE_NUM): ChTnList[i] = [None] * TONE_ANL_COL # MFRC522 SPI 設定 # どういう訳か、Pin() ではダメ・・・多分受け入れ側(ライブラリ)の仕組みか? # xxx rfid_13M_reader=MFRC522(spi_id=0,sck=Pin(6),miso=Pin(4),mosi=Pin(7),cs=Pin(5),rst=Pin(22)) rfid_13M_reader = MFRC522(spi_id=0, sck=18, miso=16, mosi=19, cs=17, rst=22) # シリアル uart_midi を生成 # MIDI は、31250bpsである。 MIDI_BAUD_RATE = 31250 uart_midi = UART(0, MIDI_BAUD_RATE,bits=8,parity=None,stop=1,tx=Pin(0),rx=Pin(1)) # 待ち時間タイマー値 NO_VOICE_TIME = const(4000) # 4000ms=4秒以上の無音空白時間で受信終了とみなす NO_TAG_TIME = const(4000) # 4000ms=4秒以上、TAG が置かれないと終了とみなす # 20240417 Change # TAG_BLOCK_TIME = const(250) # TAG BLOCK 読み取り中断を検出する TAG_BLOCK_TIME = const(300) # TAG BLOCK 読み取り中断を検出する MASK_NEXT_READ = const(500) # 次の TAG 読み込み可能までの不感時間 SWITCH_TASK_TIME = const(500) # マルチタスク処理のための時間管理 # Main との通信用(参照し制御する) # 停止状態にセット stop = 1 end = 1 # 時分割処理のためのタイマー # global変数定義。関数で利用するグローバル変数は、関数の定義前に書くのが無難。 midi_frame_timer = 0 # 無音空白時間カウント(減算) midi_ikg_timer = 0 # midi inter key gap time:(積算) midi_ikg_wait = 0 # midi inter key gap wait:(減算) tag_read_timer = 0 # > 0 で TAG 読み取り優先(減算) tag_write_timer = 0 # > 0 で TAG 書き込み優先(減算) tag_play_timer = 0 # > 0 で TAG 再生を優先(減算) tag_block_timer = 0 # == 0 で TAG BLOCK 読み取り中断を検出(減算) wait_next_read = 0 # TAG 先読み禁止時間(減算)MASK_NEXT_READ で設定 wait_midi_write = 0 # TAG 先読み許可時間(減算)SWITCH_TASK_TIME で設定 ps2kb_timer = 0 # = 0 で Key buff を空読み(減算) rfid_interval = 0 # 500ms 毎に、RFIDにタグやカードがあるかチェックする。 # 10ms 毎に割り込みが発生する なぜか引数 (t) は必須である。 def every10ms(t): ########################################################## global midi_frame_timer, midi_ikg_timer, midi_ikg_wait global tag_read_timer, tag_write_timer, rfid_interval global tag_play_timer, tag_block_timer, wait_next_read global wait_midi_write, ps2kb_timer ########################################################## if midi_frame_timer < 10: midi_frame_timer = 0 else: midi_frame_timer -= 10 if midi_ikg_timer < 10000: midi_ikg_timer += 10 if midi_ikg_wait < 10: midi_ikg_wait = 0 else: midi_ikg_wait -= 10 if tag_read_timer < 10: tag_read_timer = 0 else: tag_read_timer -= 10 if tag_write_timer < 10: tag_write_timer = 0 else: tag_write_timer -= 10 if tag_play_timer < 10: tag_play_timer = 0 else: tag_play_timer -= 10 if tag_block_timer < 10: tag_block_timer = 0 else: tag_block_timer -= 10 if wait_next_read < 10: wait_next_read = 0 else: wait_next_read -= 10 if wait_midi_write < 10: wait_midi_write = 0 else: wait_midi_write -= 10 if ps2kb_timer < 10: ps2kb_timer = 0 else: ps2kb_timer -= 10 if rfid_interval < 0: rfid_interval = 0 else: rfid_interval -= 10 # 20240416 10ms # 10ms 毎に every1ms を呼び出す呪文。これ以前に callback関数を定義のこと。 # Timer(period=10, mode=Timer.PERIODIC, callback=every10ms) # mode=Timer.PERIODIC は、指定しなくても大丈夫のようだ。 # Timer(period=10, callback=every10ms) でも動作するが # MicroPython説明サイトに従ったタイマーの初期化 tim10 = Timer(period=10, mode=Timer.PERIODIC, callback=every10ms) tim10.init(period=10, callback=every10ms) #---------------------------------------------------------------------------- # 楽譜音階データを MIDIデータに変換関数 # channel、octave、keycode、velocity から、MIDI信号を構成する関数 # 音符(時間)情報は、別管理が必要 def key2midi(channel, octave, key, velocity): channel = channel & 0x0F # 0~F に制限 ビット演算 and byte_1 = 0x90 | channel # ビット演算 or if octave < 0: octave = 0 elif octave > 8: octave = 8 # # print("key.lower() = " ,key.lower()) # if key.lower() == 'c': byte_2 = (octave + 1) * 12 + 0 elif key.lower() == 'c$': byte_2 = (octave + 1) * 12 + 1 elif key.lower() == 'd': byte_2 = (octave + 1) * 12 + 2 elif key.lower() == 'd$': byte_2 = (octave + 1) * 12 + 3 elif key.lower() == 'e': byte_2 = (octave + 1) * 12 + 4 elif key.lower() == 'f': byte_2 = (octave + 1) * 12 + 5 elif key.lower() == 'f$': byte_2 = (octave + 1) * 12 + 6 elif key.lower() == 'g': byte_2 = (octave + 1) * 12 + 7 elif key.lower() == 'g$': byte_2 = (octave + 1) * 12 + 8 elif key.lower() == 'a': byte_2 = (octave + 1) * 12 + 9 elif key.lower() == 'a$': byte_2 = (octave + 1) * 12 + 10 elif key.lower() == 'b': byte_2 = (octave + 1) * 12 + 11 # 変換範囲外の音を、この高い音にする。・・・いいのか、悪いのか? else: byte_2 = 120 # 上限(0x7F:127)を超えないようにする if byte_2 > 127: byte_2 = 127 # 上限(0x7F:127)を超えないようにする # velocity = 0 は、ノートオフ byte_3 = velocity & 0x7F # 20240814 Change # return list([byte_1, byte_2, byte_3]) (asc_H, asc_L) = moji.int2asc2(byte_1) asc_1 = asc_H + asc_L (asc_H, asc_L) = moji.int2asc2(byte_2) asc_2 = asc_H + asc_L (asc_H, asc_L) = moji.int2asc2(byte_3) asc_3 = asc_H + asc_L # 文字列を返す return asc_1 + asc_2 + asc_3 #---------------------------------------------------------------------------- # プログラムチェンジ(音色変更)記録用のポインタ、および、リストの用意 def init_ChTnList(): global ChTnPt global ChTnList # プログラムチェンジ(音色変更)記録用のポインタ ChTnPt = [None] * TONE_NUM for i in range(TONE_NUM): ChTnPt[i] = 0 # プログラムチェンジ(音色変更)記録用のリスト ChTnList = [None] * TONE_NUM for i in range(TONE_NUM): ChTnList[i] = [None] * TONE_ANL_COL #---------------------------------------------------------------------------- # 受信した MIDIプログラムチェンジ(音色変更)をリストへに追加記録 def add_ChTnList(ch, tn): global ChTnPt global ChTnList # プログラムチェンジ(音色変更)記録用のポインタ if ChTnPt[ch & 0x0F] < TONE_ANL_COL: ChTnList[ch & 0x0F][ChTnPt[ch & 0x0F]] = tn & 0x7F ChTnPt[ch & 0x0F] += 1 #---------------------------------------------------------------------------- # 記録した MIDIプログラムチェンジ(音色変更)リストをシリアルアウト def serialOutChTnList(): global ChTnPt global ChTnList for i in range(TONE_NUM): print('PROGRAM CHANGE ', i, ' ', end='') for j in range(ChTnPt[i]): print(ChTnList[i][j], ' ', end='') print() #---------------------------------------------------------------------------- # MIDI ALL SOUND OFF def midiSoundOff(): for i in range(TONE_NUM): midi_buff = moji.byte2chr2(chr(0xB0 | i)) midi_buff += moji.byte2chr2(chr(0x78)) midi_buff += moji.byte2chr2(chr(0x00)) midi_writeB(midi_buff) #---------------------------------------------------------------------------- # 文字列("CB0B" や "B07B00"など)を MIDI に出力 # リスト形式を出力する場合は play_midiList() を利用してネ def midi_writeB(midi_buff): if len(midi_buff) > 0: midi_num_list = [] print(" >>> midi_sendbuff = ", midi_buff) for i in range(0, len(midi_buff), 2): midi_data = midi_buff[i] midi_data += midi_buff[i+1] midi_data_num = moji.chr2dec(midi_data) midi_num_list.append(midi_data_num) # 整数リストから、バイト文字列に変換してコピー midi_data_str = bytes(midi_num_list) # バイト文字列を UART で MIDI送信 uart_midi.write(midi_data_str) #---------------------------------------------------------------------------- # リスト形式([['90', '3C', '40'], ['90', '3C', '40'],...])を MIDI に出力 # 文字列を出力する場合は midi_writeB(midi_buff) を利用してネ # 20240603 Change # def play_midiList(): def play_midiList(midi_list): if midi_list != None: if len(midi_list) > 0: print("UART MIDI Writing...") midi_sendbuff = "" wait_ms = 0 wait_CMD = 0 for i in range(len(midi_list)): for j in range(len(midi_list[i])): # 1バイトを数値に変換 midi_data = moji.chr2dec(midi_list[i][j]) # ステータスバイトなら if midi_data >= 0x80: # F2 タイマー値なら、待ち時間を入れる if midi_data == 0xF2: wait_CMD = 1 # F0 Miku Voice情報なら、メッセージを補完する。 # F0 43 79 09 11 0A 00 # 発音文字コード(00~7F)と、末尾に F7が追加されて完成する。 elif midi_data == 0xF0: midi_sendbuff = 'F0437909110A00' # 0xF7(ユニバーサル・システム・エクスクルーシブ)なら # midi_sendbuff に追加する(データとして扱う) # elif midi_data == 0xF7 and wait_CMD == 0: elif midi_data == 0xF7: midi_sendbuff += midi_list[i][j] # 0xF0 , 0xF2 , 0xF7 以外の ステータスバイトなら # midi_sendbuff の先頭にデータを入れる else: midi_sendbuff = midi_list[i][j] # タイマー値ならば タイマー値の計算をする elif wait_CMD == 1: wait_ms += wait_ms * 128 + moji.chr2dec(midi_list[i][j]) if wait_ms > 10000: wait_ms = 10000 # ステータスバイトでなく、かつ、タイマー値でもなければ # midi_sendbuff に追加する(データとして扱う) else: midi_sendbuff += midi_list[i][j] # MIDI 送信バッファにデータがあれば、USART に出力 if len(midi_sendbuff) > 0: # 文字列を UART で MIDI に送信する print("midi i = ", i, end="") ###################################### midi_writeB(midi_sendbuff) ###################################### midi_sendbuff = "" # 必要ないかもしれないが・・・ if wait_ms > 0: print("midi i = ", i, end="") print(" >>> wait_time = ", wait_ms) utime.sleep_ms(wait_ms) wait_ms = 0 wait_CMD = 0 else: print("midi_list has no data ! ") else: print("midi_list is None ! ") #---------------------------------------------------------------------------- # 「F4」キーを押してから、4秒(=tag_read_timer)以内にTAG を置くこと # 「F4」キーを押す前に、TAGを置くと、常時 TAG Read が動作する # MFRC522(13.56MHz)RFID にタグやカードが置かれていたら読み込む # タグやカードが4秒以内に置かれなければ、何もせず終了 def tag_read(): global tag_read_timer while tag_read_timer > 0: # MFRC522(13.56MHz)RFID にタグやカードが置かれていたら読み込む rfid_13M_reader.init() (stat, tag_type) = rfid_13M_reader.request(rfid_13M_reader.REQIDL) if stat == rfid_13M_reader.OK: (stat, uid) = rfid_13M_reader.SelectTagSN() if stat == rfid_13M_reader.OK: # Read Data 準備 key_A = [0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF] # tag_rcvbuff = "" tag_address = 0 for tag_address in range(42): # tag_address を sector と block に換算 sector = int(tag_address / 3) + 1 block = tag_address % 3 # TAG を読み込む (stat, block_data_r) = rfid_13M_reader.readSectorBlock(uid ,sector, block, keyA=key_A, keyB = None) if stat == rfid_13M_reader.OK: # print("BLOCK DATA = ", block_data_r) # # 読み込んだ1ブロックを、文字列(tag_rcvbuff)に追加する flag_tag_EOB = 0 # 1:以降のデータを読み取り不要 flag_tag_EOF = 0 # 1:以降のブロックを読み取り不要 for i in range(len(block_data_r)): # スペース、または、Null('00')ならば、 if block_data_r[i] == 0x20 or block_data_r[i] == 0x00: # ブロックの先頭なら、EOF と判断する if i == 0: flag_tag_EOF = 1 break # ブロックの先頭でなければ、EOB ブロック終了 else: flag_tag_EOB = 1 break # スペース、または、Null('00')でなければ、受信バッファに追加する # 整数リストから文字列に変換する else: tag_rcvbuff += chr(block_data_r[i]) # 常時稼働時には、エラー表示が邪魔なので、出さない else: print ("rfid_13M_reader Read Error") # EOF を検出したら、以降のブロックを読まないで終了する。 if flag_tag_EOF == 1: break # tag_rcvbuf を midi_list に収納する midi_list = [] if len(tag_rcvbuff) > 0: print("tag_rcvbuff Size = ", len(tag_rcvbuff)) temp_list = [] temp_buff = "" begin = 1 for i in range(len(tag_rcvbuff)): # 偶数ならば if (i % 2) == 0: # 直前の temp_list を midi_list に追加する # ただし、一番先頭(空データ)は追加しない if ord(tag_rcvbuff[i]) >= 0x38 and ord(tag_rcvbuff[i]) != 0xF7: if begin == 1: begin = 0 else: midi_list.append(temp_list) temp_list = [] # temp_buff に追加する temp_buff += tag_rcvbuff[i] # 奇数ならば、偶数桁と合わせて、temp_buff に追加する else: temp_buff += tag_rcvbuff[i] temp_list.append(temp_buff) temp_buff = "" # temp_list を midi_list にコピーする midi_list.append(temp_list) print("MIDI List = ", midi_list) # return midi_list break # tag_read_timer == 0 ならの else を skip # 常時稼働時には、エラー表示が邪魔なので、出さない # else: # print("No TAG or CARD") # 常時稼働時には、エラー表示が邪魔なので、出さない # else: # print ("rfid_13M_reader is not Raedy") # # tag_read_timer == 0 なら else: print("TAG Read Time Over") #---------------------------------------------------------------------------- # MFRC522(13.56MHz)RFID にタグやカードが置かれているかチェックする def checkOnRfid(): rfid_13M_reader.init() (stat, tag_type) = rfid_13M_reader.request(rfid_13M_reader.REQIDL) # MFRC522(13.56MHz)RFID にタグやカードが置かれていたら取り除かれるまで待つ if stat == rfid_13M_reader.OK: # print('checkOnRfid(): stat == rfid_13M_reader.OK') return True else: # print('checkOnRfid(): stat != rfid_13M_reader.OK') return False ''' #---------------------------------------------------------------------------- # MFRC522(13.56MHz)RFID にタグやカードが置かれていたら取り除かれるまで待つ def waitRemoveRfid(): rfid_13M_reader.init() (stat, tag_type) = rfid_13M_reader.request(rfid_13M_reader.REQIDL) # MFRC522(13.56MHz)RFID にタグやカードが置かれていたら取り除かれるまで待つ while stat == rfid_13M_reader.OK: print('RFID-CAD または RFID-TAG を 取り除いてください。') utime.sleep_ms(1000) (stat, tag_type) = rfid_13M_reader.request(rfid_13M_reader.REQIDL) ''' #---------------------------------------------------------------------------- # MFRC522(13.56MHz)RFID にタグやカードが置かれていたら # headerID と headerName を読み込む # 1回のみ実行する。タグやカードが置かれていなければ、何もせず終了 # rcv_buff は、ASCII 2文字の文字列 def readRfidHeader(): # MFRC522(13.56MHz)RFID にタグやカードが置かれていたら # tag_address = 0(セクター1、ブロック0) の16バイトを読み込む rfid_13M_reader.init() (stat, tag_type) = rfid_13M_reader.request(rfid_13M_reader.REQIDL) if stat == rfid_13M_reader.OK: (stat, uid) = rfid_13M_reader.SelectTagSN() if stat == rfid_13M_reader.OK: # Read Data 準備 key_A = [0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF] tag_rcvbuff = "" tag_address = 0 # tag_address を sector と block に換算 sector = int(tag_address / 3) + 1 block = tag_address % 3 # TAG を読み込む (stat, block_data_r) = rfid_13M_reader.readSectorBlock(uid ,sector, block, keyA=key_A, keyB = None) if stat == rfid_13M_reader.OK: # 読み込んだ1ブロックを、文字列(tag_rcvbuff)に追加する flag_tag_EOF = 0 # 1:以降のブロックを読み取り不要 for i in range(len(block_data_r)): # ブロックの先頭スペース、または、Null('00')ならば、 if block_data_r[i] == 0x20 or block_data_r[i] == 0x00: # ブロックの先頭なら、EOF と判断する if i == 0: flag_tag_EOF = 1 break else: tag_rcvbuff += chr(block_data_r[i]) # スペース、または、Null('00')でなければ、受信バッファに追加する # 整数リストから文字列に変換する else: tag_rcvbuff += chr(block_data_r[i]) # 常時稼働時には、エラー表示が邪魔なので、出さない else: # DEBUG print # print ("readRfidHeader(): rfid_13M_reader Read Error") headerID = 'Error' headerName = 'RFID Read Error' # EOF を検出したら、以降のブロックを読まないで終了する。 if flag_tag_EOF == 1: headerID = 'Null' headerName = 'Null Data' # 読み取り後の処理 elif len(tag_rcvbuff) > 0: headerID = tag_rcvbuff[0] headerName = tag_rcvbuff[1:TONE_HEADER] # DEBUG print # print('readRfidHeader(): headerID = ', headerID) # print('readRfidHeader(): headerName = ', headerName) return headerID, headerName # 常時稼働時には、エラー表示が邪魔なので、出さない # else: # print("No TAG or CARD") # 常時稼働時には、エラー表示が邪魔なので、出さない # else: # print ("rfid_13M_reader is not Raedy") #---------------------------------------------------------------------------- # MFRC522(13.56MHz)RFID にタグやカードが置かれていたら読み込む # タグやカードが4秒以内に置かれなければ、何もせず終了 # rcv_buff は、ASCII 2文字の文字列 # tag_read_timer を設定して呼び出すこと def readRfidTone(): global tag_read_timer while tag_read_timer > 0: # MFRC522(13.56MHz)RFID にタグやカードが置かれていたら読み込む rfid_13M_reader.init() (stat, tag_type) = rfid_13M_reader.request(rfid_13M_reader.REQIDL) if stat == rfid_13M_reader.OK: (stat, uid) = rfid_13M_reader.SelectTagSN() if stat == rfid_13M_reader.OK: # Read Data 準備 key_A = [0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF] # tag_rcvbuff = "" tag_address = 0 # # 最大で、全アドレス(RFID_ADDRESS_MAX:45 未満)を読み込む for tag_address in range(RFID_ADDRESS_MAX): # tag_address を sector と block に換算 sector = int(tag_address / 3) + 1 block = tag_address % 3 # TAG を読み込む (stat, block_data_r) = rfid_13M_reader.readSectorBlock(uid ,sector, block, keyA=key_A, keyB = None) if stat == rfid_13M_reader.OK: print("BLOCK DATA = ", block_data_r) # 読み込んだ1ブロックを、文字列(tag_rcvbuff)に追加する flag_tag_EOF = 0 # 1:以降のブロックを読み取り不要 for i in range(len(block_data_r)): # ブロックの先頭スペース、または、Null('00')ならば、 if block_data_r[i] == 0x20 or block_data_r[i] == 0x00: # ブロックの先頭なら、EOF と判断する if i == 0: flag_tag_EOF = 1 break else: tag_rcvbuff += chr(block_data_r[i]) # スペース、または、Null('00')でなければ、受信バッファに追加する # 整数リストから文字列に変換する else: tag_rcvbuff += chr(block_data_r[i]) # 常時稼働時には、エラー表示が邪魔なので、出さない else: print ("rfid_13M_reader Read Error") # EOF を検出したら、以降のブロックを読まないで終了する。 if flag_tag_EOF == 1: break # 読み取り後の処理 rcv_size = len(tag_rcvbuff) if rcv_size > 0: print("tag_rcvbuff Size = ", rcv_size) print("tag_rcvbuff = ", tag_rcvbuff) headerID = tag_rcvbuff[0] tone_name = tag_rcvbuff[1:TONE_HEADER] print('Recive Data = ', tone_name) # Tone Data if headerID == '0': # 2文字の ASCIIコード(0-9 A-F)を1byteの整数に変換する。 rcv_tone = [] for i in range(TONE_DATA_ROW_NUM): # 42 # rcv_tone.append(moji.asc2int(chr(tag_rcvbuff[TONE_HEADER + i *2]) \ # + chr(tag_rcvbuff[TONE_HEADER + i *2 +1]))) rcv_tone.append(moji.asc2int(tag_rcvbuff[TONE_HEADER + i *2] \ + tag_rcvbuff[TONE_HEADER + i *2 +1])) # DEBUG print print('rcv_tone = ', rcv_tone) return stat, rcv_size, headerID, tone_name, rcv_tone elif headerID == '1': # 2文字の ASCIIコード(0-9 A-F)を1byteの整数に変換する。 rcv_tone = [] for i in range(TONE_TABLE_ROW_NUM -1): # 0-15 : 16 rcv_tone.append(moji.asc2int(chr(tag_rcvbuff[TONE_HEADER + i *2]) \ + chr(tag_rcvbuff[TONE_HEADER + i *2 +1]))) # DEBUG print print('ToneTable = ', rcv_tone) return stat, rcv_size, headerID, tone_name, rcv_tone # 常時稼働時には、エラー表示が邪魔なので、出さない # else: # print("No TAG or CARD") # 常時稼働時には、エラー表示が邪魔なので、出さない # else: # print ("rfid_13M_reader is not Raedy") # tag_read_timer == 0 なら else: print("TAG Read Time Over") # return stat, rcv_size = None, headerID = None, tone_name = None, rcv_tone = None return stat, None, None, None, None #---------------------------------------------------------------------------- # tag_sendbuff を tag_address から算定した TAG のブロックに書き込む # def tag_write2(midi_list) の付属関数 # TAG の 書き込み領域の sector が 15 を超えたら書き込めないので result = 2 # result = 1 は、正常に書き込み完了したとき def tag_block_write(uid, tag_sendbuff, tag_address): # Write Data 準備 key_A = [0xFF, 0xFF, 0xFF, 0xFF, 0xFF, 0xFF] # 末尾にスペースを入れて16桁にする tag_sendbuff = moji.strFixL_n(tag_sendbuff, 16) # Tag 書き込み用に整数化したリストに展開する block_data_w = [] for k in range(len(tag_sendbuff)): block_data_w.append(ord(tag_sendbuff[k])) print("BLOCK DATA = ", block_data_w) # tag_address を sector と block に換算 sector = int(tag_address / 3) + 1 block = tag_address % 3 print("tag_address=", tag_address, " sector=", sector, " block=", block, end="") # TAG 書き込み領域外なら、書き込みせずに終了 if sector > 15: result = 2 else: # Tag に書き込みする stat = rfid_13M_reader.writeSectorBlock(uid ,sector, block, data=block_data_w[:], keyA=key_A, keyB=None) # 正常終了なら if stat == rfid_13M_reader.OK: result = 1 # 異常終了なら else: result = 0 print(" result=", result) return result, sector, block #---------------------------------------------------------------------------- # MFRC522(13.56MHz)RFID にタグやカードが置かれたら、ToneData を書き込む # タグやカードが4秒以内に置かれなければ、何もせず終了 # send_buff は、ASCII 2文字の文字列 # tag_write_timer を設定して呼び出すこと def writeRfidTone(send_buff): global tag_write_timer # 書き込みデータが 0 でなら中断する if len(send_buff) <= 0: err_size = 1 print('len(send_buff) = ', len(send_buff), 'send_buff SIZE = 0') return else: # 読み取り時 EOF 検出用に 16個のスペースを付け加える send_buff += ' ' # space(16) # send_buff のサイズが RFID_BLOCK_SIZE =16 で余りがあるなら +1 if (len(send_buff) % RFID_BLOCK_SIZE) != 0: address_max = int(len(send_buff) /16) +1 else: address_max = int(len(send_buff) /16) # RFID への書き込み address が RFID_ADDRESS_MAX 未満か確認 if address_max >= RFID_ADDRESS_MAX: # 45 err_size = 1 print('len(send_buff) = ', len(send_buff), 'Bytes, Size is Over RFID_ADDRESS_MAX ',RFID_ADDRESS_MAX, n ) return else: # byte 文字列を 通常の文字列に戻す buffB = send_buff.decode('utf-8') # RFID への書き込み範囲が正しい範囲なら # 0 < send_buff < RFID_ADDRESS_MAX *16 while tag_write_timer > 0: # MFRC522(13.56MHz)RFID にタグやカードが置かれていたら書き込む rfid_13M_reader.init() (stat, tag_type) = rfid_13M_reader.request(rfid_13M_reader.REQIDL) if stat == rfid_13M_reader.OK: (stat, uid) = rfid_13M_reader.SelectTagSN() if stat == rfid_13M_reader.OK: for i in range(address_max): tag_address = i tag_sendbuff = buffB[i *RFID_BLOCK_SIZE : i *RFID_BLOCK_SIZE +RFID_BLOCK_SIZE] if len(tag_sendbuff) > 0: print('tag_address = ', i, 'tag_sendbuff = ', tag_sendbuff) ########################################################################## (result, sector, block) = tag_block_write(uid, tag_sendbuff, tag_address) ########################################################################## # RFID 書き込み時にエラー発生 if result != 1: break # 正常終了 if result == 1: print('Performed write Rfid Tone') break # RFIDモジュールでエラー発生 elif result == 0: print("Failed write Rfid Tone") break # RFID ユーザー領域を超えて書き込もうとするエラー発生 elif result == 2: print("Failed write Rfid Tone! TAG outside write area !! sector = ", sector, " block = ", block) break # 常時稼働時には、エラー表示が邪魔なので、出さない # else: # print("No TAG or CARD") # 常時稼働時には、エラー表示が邪魔なので、出さない # else: # print ("rfid_13M_reader is not Raedy") # tag_write_timer == 0 終了なら else: print("Time Over writeRfidTone() ") #---------------------------------------------------------------------------- # 「F5」キーを押してから、4秒(=tag_write_timer)以内にTAG を置くこと # 「F5」キーを押す前に、TAGを置くと、常時 TAG Read が動作する # MFRC522(13.56MHz)RFID にタグやカードが置かれたら書き込む # タグやカードが4秒以内に置かれなければ、何もせず終了 # def tag_write(midi_list): # 20240417 Change def tag_write(): global midi_list global tag_write_timer err_size = 0 result = 0 while tag_write_timer > 0: # MFRC522(13.56MHz)RFID にタグやカードが置かれていたら書き込む rfid_13M_reader.init() (stat, tag_type) = rfid_13M_reader.request(rfid_13M_reader.REQIDL) if stat == rfid_13M_reader.OK: (stat, uid) = rfid_13M_reader.SelectTagSN() if stat == rfid_13M_reader.OK: tag_sendbuff = "" print("tag_address in range(") tag_address = 0 for i in range(len(midi_list)): temp_buff = "" for j in range(len(midi_list[i])): temp_buff += midi_list[i][j] if len(temp_buff) > 16: err_size = 1 break else: err_size = 0 if len(tag_sendbuff + temp_buff) > 16: print("tag_sendbuff=", tag_sendbuff) ########################################################################## (result, sector, block) = tag_block_write(uid, tag_sendbuff, tag_address) ########################################################################## tag_address += 1 tag_sendbuff = "" if result != 1: break tag_sendbuff += temp_buff # if err_size == 1 or result != 1: break # tag_sendbuff に残っていれば if len(tag_sendbuff) > 0: print("tag_sendbuff=", tag_sendbuff) ########################################################################## (result, sector, block) = tag_block_write(uid, tag_sendbuff, tag_address) ########################################################################## tag_address += 1 tag_sendbuff = "" if result != 1: break if sector <= 15 and block < 2: data_EOF = " " print("data_EOF=", data_EOF) ##################################################################### (result, sector, block) = tag_block_write(uid, data_EOF, tag_address) ##################################################################### tag_address += 1 if result != 1: break # ここまで break なしで来るなら正常終了 if result == 1: break # 常時稼働時には、エラー表示が邪魔なので、出さない # else: # print("No TAG or CARD") # 常時稼働時には、エラー表示が邪魔なので、出さない # else: # print ("rfid_13M_reader is not Raedy") # tag_write_timer == 0 終了なら else: print("TAG Write Time Over") # break 終了なら if err_size == 1: print("midi_list has items over 16 bytes !!") if result == 0: print("Failed Tag Write !!") if result == 2: print("TAG outside write area !! sector = ", sector, " block = ", block) #---------------------------------------------------------------------------- # 与えられた文字列を2バイトを1アイテムとして リストを生成し返す # 例:midi_rcvbuff="9nkkvv" → temp_list = ['9n', 'kk', 'vv'] def midiB2list(midi_rcvbuff): temp_list = [] for i in range(len(midi_rcvbuff)): # 偶数ならば if (i % 2) == 0: temp_buff = midi_rcvbuff[i] # 奇数ならば、偶数桁と合わせて、temp_list に追加する else: temp_buff += midi_rcvbuff[i] temp_list.append(temp_buff) return temp_list #---------------------------------------------------------------------------- # def midi_play(): の付随関数 # midi_rcvbuff で midi_listR を書き換える # midi_rcvbuff の2バイトを1アイテムとして midi_listR 記録する # 例:midi_rcvbuff="9nkkvv" → midi_listR = ['9n', 'kk', 'vv'] def update_midi_listR(cnt_listR, vol_listR, midi_rcvbuff): global midi_listR temp_list = midiB2list(midi_rcvbuff) # vol_listR のアイテム数を持つリスト midi_listR の pt_listR の場所を # temp_list で書き換える。 # cnt_listR は、連番で積算しているので vol_listR よりも大きい場合がある。 # 記録場所の計算 pt_listR = cnt_listR % vol_listR # midi_listR 書き換え midi_listR[pt_listR] = temp_list #---------------------------------------------------------------------------- # MIDI-Inデータを MIDI-Outに 出力しながら記録する。Escキーが押されるまで動作する。 # MIDI入力データを、 midi_listR に記録し、最後に midi_list にコピーする。 # midi_listR は、サイズが vol_listR で一周で繋がったエンドレス構造とし、 # MIDI入力データで順次、上書きし、最後に残ったデータを midi_list にコピーする。 # midi_listの先頭に楽器設定情報を末尾にオールノートオフを追加する。プラス2アイテム # データが少ない場合でも、midi_listの先頭側に空のデータが入らないように工夫した。 # pt_listRは次に記録すべきポイントを示す。何度か書き換えられた場合は再生開始位置。 # 待ち時間を、F2 xx xx 形式で挿入する。 def midi_play(swap_ch = 0): global midi_list, midi_listR, midi_channel, midi_tone_data global midi_ikg_timer, midi_frame_timer global stop, end # Main との通信用(参照し制御する) pt_listR = 0 cnt_listR = 0 # 受信データの MIDI Item 数(積算) vol_listR = const(80) temp_list = ['00', '00', '00'] # 空の midi_listR を生成・・・[temp_list] と [] で括ることが重要 midi_listR = [temp_list] * vol_listR midi_rcvbuff = "" time_buff = "" midi_ikg_timer = 0 # 積算タイマー cnt_rcv = 0 esc_break = 0 begin = 1 data_cnt_old = 0 # プログラムチェンジ(音色変更)記録用のポインタ、および、リストの用意 init_ChTnList() # 演奏状態にセット stop = 0 end = 0 # while midi_frame_timer > 0: # 減算タイマー while end == 0: # end == 1 で終了 # MIDI 受信ありなら if uart_midi.any() > 0: try: # uart_midi から、1データ読み込む(decodeなし=bytesデータのまま) midi_data_rx = uart_midi.read(1) except UnicodeError: # 読み込みエラーになるの?なら print("uart_midi.read(1) Error!!") midi_data_rx = "" # 受信した1バイトを2バイトの文字に変換して、シリアル出力する print(moji.byte2chr2(midi_data_rx), end="") cnt_rcv += 1 # 受信文字数カウント ######################################################################### # MIDIステータスが、 0x80 〜 0xCF であり、かつ、チャンネルが、0 であり、かつ、 # 指定されたスワッピング チャンネルが、0 でないなら # 20240626 上限を <= 0xF7 から <= 0xEF に変更 if ord(midi_data_rx) >= 0x80 and ord(midi_data_rx) <= 0xEF \ and swap_ch != 0: # 受信チャンネルが 0 なら、swap_ch に変換して出力する if ord(midi_data_rx) & 0x0F == 0x00: midi_data_rx = chr((ord(midi_data_rx) & 0xF0) | (swap_ch & 0x0F)) # 受信チャンネルが、swap_ch なら、チャンネル 0 に変換して出力する elif ord(midi_data_rx) & 0x0F == (swap_ch & 0x0F): midi_data_rx = chr((ord(midi_data_rx) & 0xF0)) uart_midi.write(midi_data_rx) ######################################################################## # 待ち時間の追加 # (受信したデータの上位4ビットが、8以上なら、ステータスバイト であり # かつ、0xF7(ユニバーサル・システム・エクスクルーシブ末尾)以外であり) # かつ、midi_ikg_timer が 20ms を超えているなら # 待ち時間を、F2 xx xx 形式で挿入する ######################################################################### if ord(midi_data_rx) >= 0x80 and ord(midi_data_rx) != 0xF7: # 最初のデータでは、何もせず midi_buff に保存する if begin == 1: begin = 0 else: # 直前のデータを記録する # midi_rcvbuff で midi_listR を書き換える print(" >>> cnt_listR = ", cnt_listR, end="") print(" >>> midi_rcvbuff = ", midi_rcvbuff) # 20240612 Add # 直前のデータが、プログラムチェンジならば、 if midi_rcvbuff[0] == 'C': # ChTnList に記録する ch = moji.asc2int(midi_rcvbuff[0:2]) % 16 tn = moji.asc2int(midi_rcvbuff[2:4]) ############################################### add_ChTnList(ch, tn) print('PROGRAM CHANGE ', midi_rcvbuff, ch, tn) ############################################### ###################################################### update_midi_listR(cnt_listR, vol_listR, midi_rcvbuff) ###################################################### # ポインタのカウントアップ cnt_listR += 1 #################### midi_rcvbuff = "" #################### # 直前のデータから 20ms を超えていれば、待ち時間を記録する if midi_ikg_timer > 20: waitMSB = int(midi_ikg_timer / 128) waitLSB = midi_ikg_timer % 128 time_buff = "F2" # 「F2」を記録する time_buff += moji.byte2chr2(chr(waitMSB)) time_buff += moji.byte2chr2(chr(waitLSB)) # time_buff で midi_listR を書き換える print(" >>> time_buff = ", time_buff) ##################################################### update_midi_listR(cnt_listR, vol_listR, time_buff) ##################################################### # ポインタのカウントアップ cnt_listR += 1 ################### midi_ikg_timer = 0 ################### # 受信データの一時保存 # 1バイトの受信データを2バイトの文字データにして保存する midi_rcvbuff += moji.byte2chr2(midi_data_rx) elif stop == 1: # Main def every97ms(t): でセットする] end = 1 # ESC キー押下で、終了なら if end == 1: print("ESC Key Break") ########################## midiSoundOff() ########################## # 直前のデータが残っていれば、midi_rcvbuff で midi_listR を書き換える # 通常、9n kk vv なので、6桁以上ある。 if len(midi_rcvbuff) >= 6: print("midi_rcvbuff = ", midi_rcvbuff) ##################################################### update_midi_listR(cnt_listR, vol_listR, midi_rcvbuff) ##################################################### # ポインタのカウントアップ cnt_listR += 1 midi_rcvbuff = "" # MIDI 受信記録があれば、midi_listR を midi_list にコピーする # 先頭に楽器設定コマンド、末尾にオールノートオフ コマンドを追加する if cnt_listR > 0: # 受信状況をシリアルに出力 print("cnt_rcv = ", cnt_rcv, end="") print(" cnt_listR = ", cnt_listR, end="") print(" pt_listR = ", pt_listR, end="") print(" vol_listR = ", vol_listR) midi_list = [] # 20240615 Comment Out # 先頭に、楽器設定コマンドを入れる # midi_tone_data ######################################## # temp_list = midiB2list(midi_tone_data) ######################################## # midi_list.append(temp_list) # 記録したデータ数が vol_listR 未満なら、midi_listR の先頭からコピーする if cnt_listR < vol_listR: for i in range(cnt_listR): midi_list.append(midi_listR[i]) # 記録したデータ数が vol_listR 以上なら(受信データ数が、一巡以上なら) # pt_listR から、一巡して、pt_listR の一つ手前までコピーする else: for i in range(pt_listR, pt_listR + vol_listR): midi_list.append(midi_listR[ i % vol_listR ]) # 末尾に オールノートオフ(0xB07B00)を追加する # 20240615 オールサウンドオフ(0xB07800)に変更する # すべてのチャンネルの発信しなければならないが、ここでは、チャンネル 0 のみ # midi_Noteoff = moji.byte2chr2(chr( 0xB0 | (midi_channel % 16))) midi_Noteoff = moji.byte2chr2(chr( 0xB0)) # midi_Noteoff += moji.byte2chr2(chr( 0x7B)) midi_Noteoff += moji.byte2chr2(chr( 0x78)) midi_Noteoff += moji.byte2chr2(chr( 0x00)) ######################################## temp_list = midiB2list(midi_Noteoff) ######################################## midi_list.append(temp_list) # midi_list をシリアルに出力 for i in range(len(midi_list)): print("midi_list = ", i, midi_list[i]) # 記録した MIDIプログラムチェンジ(音色変更)リストをシリアルアウト serialOutChTnList() #---------------------------------------------------------------------------- # 百人一首用に編集されたリスト HectOneを再生する。 # address, VoiceRome, OctKey, Velocity, onTime, offTime # Title = 先頭のデータ = ['Ad', 'Voi', 'Key', 'Vel', 'oN', 'oF'] # def playHectOne(HectOne): if HectOne != None: # HectOneが存在するなら for i in range(1, len(HectOne)): # [1]番目のデータは、音声データのローマ字である。 try: voice = exRmPkMiku.get(HectOne[i][1]) except: print(f'Error On : voice = exRmPkMiku.get({HectOne[i][1]})') voice = 0 midi_sendbuff = 'F0437909110A00' # voiceを 16進数表記のASCII文字に変換して midi_sendbuff末尾に追加する。 (asc_H, asc_L) = moji.int2asc2(voice) midi_sendbuff += asc_H + asc_L # 末尾に「F7」追加 midi_sendbuff += 'F7' ###################################### midi_writeB(midi_sendbuff) ###################################### # [2]番目のデータは、音階データ(オクターブとキー)である。 # Octave Key #(#は使いないので $で代用) octkey = HectOne[i][2] # ASCII octave = int(octkey[0]) # INT key = octkey[1] # ASCII if len(octkey) > 2: # 2桁超えならば、半音変化あり。半音は #として扱う key += '$' # ASCII # [3]番目のデータは、音量データ(velocity)である。 velocity = HectOne[i][3] # INT # 音階データを生成 [2]番目のデータと[3]番目のデータ channnel = 0 ############################################################ midi_writeB(key2midi(channel, octave, key, velocity)) ############################################################ # [4]番目のデータは、ノートオン持続時間である。 # F2 Msb Lsb の形式のしたがって、128msを1単位としたMSBとして扱う。 time = HectOne[i][4] * 128 # INT utime.sleep_ms(time) # velocityを 0 とした ノートオフを生成 # 他のデータは、直前のデータを活かす velocity = 0 ############################################################ midi_writeB(key2midi(channel, octave, key, velocity)) ############################################################ # [5]番目のデータは、ノートオフ持続時間である。 # F2 Msb Lsb の形式のしたがって、128msを1単位としたMSBとして扱う。 time = HectOne[i][5] * 128 # INT utime.sleep_ms(time) # 念のために MIDI ALL SOUND OFF midiSoundOff() else: print('Error : HectOne = None') # ---------------------------------------------------------------- # MicroPython Main Program # ---------------------------------------------------------------- channel = 0 octave = 4 key = 'A' velocity = 64 # 変数の初期設定 midi_frame_timer = 0 midi_ikg_timer = 10000 rfid_interval = 500 midi_rcvbuff = "" midi_rcvsize = 0 midi_buff_full = 0 # midi_buff > 240 midi_list = [] midi_listR = [] midi_listM1 = [] midi_listM2 = [] midi_channel = 0 tone = 0 # 楽器の番号 データは、0~127(楽器番号は、1-128) midi_tone_data = moji.byte2chr2(chr( 0xC0 | (midi_channel % 16))) midi_tone_data += moji.byte2chr2(chr(( tone % 16) & 0x7F )) midi_Noteoff = moji.byte2chr2(chr( 0xB0 | (midi_channel % 16))) midi_Noteoff += moji.byte2chr2(chr( 0x7B)) midi_Noteoff += moji.byte2chr2(chr( 0x00)) # ---------------------------------------------------------------- # 20240603 Add # テスト デバッグ用 if __name__ == '__main__': ##################################################################### print("tag_read()... Place your TAG within 4 seconds ...") tag_read_timer = NO_TAG_TIME # TAG 読み取り優先時間 4秒 midi_list = tag_read() # TAG を読み込み midi_list に書き込む ##################################################################### ##################################################################### print("play_midiList()... play memory") # 20240603 Change # play_midiList() # midi_list(リスト) を MIDI 演奏 play_midiList(midi_list) # midi_list(リスト) を MIDI 演奏 #####################################################################

mojipy.py

mojipy.py

############################################################################# # mojipy.py # 2024/04/15:文字処理をライブラリ化:Programd by Hyodo # Thonny で Pico直下にコピーしておき # import mojipy as moji として利用する #---------------------------------------------------------------------------- # 整数を 16進数表記のASCII文字(0~9 A~F)2文字に変換する。 # 整数を 10進数表記のASCII文字に変換するには str(num) でよい。 def int2asc2(num): num_H = int(num / 16) num_L = num % 16 if num_H > 9: num_H = num_H - 9 asc_H = chr(0x40 + num_H) else: asc_H = chr(0x30 + num_H) if num_L > 9: num_L = num_L - 9 asc_L = chr(0x40 + num_L) else: asc_L = chr(0x30 + num_L) return asc_H, asc_L #---------------------------------------------------------------------------- # 2バイト以上の16進数表記のASCIIコード(0~9、A~F a~f)を整数に変換する。 # 10進数表記のASCIIコードを整数に変換するには int(s) でよい。 # int2asc2(num)関数の逆操作 def asc2int(s): l = len(s) data_int = 0 # rcv_data は、s の i 番目の要素を示す 「Python入門」p140 参照 for i, rcv_data in enumerate(s): # print("rcv_data[{}] = {} ".format(i, ord(rcv_data))) data_ascii = ord(rcv_data) # print("rcv_ascii[{}] = {} ".format(i, data_ascii), end="") if data_ascii >= 48 and data_ascii <= 57: data_int += (data_ascii - 48) * 16 ** (l - i - 1) elif data_ascii >= 65 and data_ascii <= 70: data_int += ((data_ascii - 65) + 10) * 16 ** (l - i - 1) elif data_ascii >= 97 and data_ascii <= 102: data_int += ((data_ascii - 97) + 10) * 16 ** (l - i - 1) return data_int #---------------------------------------------------------------------------- # 1バイトデータを4ビットx2に分け、0~9 A~F 2文字(2バイト)に変換する。 # chr2dec(s)関数(16進表記の文字列を 0x00 〜 0xFF の数値に変換)の逆操作 def byte2chr2(num): num_H = int(ord(num) / 16) num_L = ord(num) % 16 if num_H > 9: num_H = num_H - 9 str_H = chr(0x40 + num_H) else: str_H = chr(0x30 + num_H) if num_L > 9: num_L = num_L - 9 str_L = chr(0x40 + num_L) else: str_L = chr(0x30 + num_L) return str_H + str_L #---------------------------------------------------------------------------- # 2バイト以上のAsciiコード(0~9、A~F a~f)を数値に変換する。 # byte2chr2(num)関数(1バイトを16進表記2文字に変換)の逆操作 # 2023/11/27:この関数は、次の式と同じであるが、Value Errorを生じさせない。 # return int("0x" + s) def chr2dec(s): l = len(s) data_hex = 0 # rcv_data は、s の i 番目の要素を示す 「Python入門」p140 参照 for i, rcv_data in enumerate(s): # print("rcv_data[{}] = {} ".format(i, ord(rcv_data))) data_ascii = ord(rcv_data) # print("rcv_ascii[{}] = {} ".format(i, data_ascii), end="") if data_ascii >= 48 and data_ascii <= 57: data_hex += (data_ascii - 48) * 16 ** (l - i - 1) elif data_ascii >= 65 and data_ascii <= 70: data_hex += ((data_ascii - 65) + 10) * 16 ** (l - i - 1) elif data_ascii >= 97 and data_ascii <= 102: data_hex += ((data_ascii - 97) + 10) * 16 ** (l - i - 1) return data_hex #---------------------------------------------------------------------------- # 数値を指定桁の数字文字列に変換・・・数字を右詰め、先頭の余白は、スペース # 20240328 Name Change def int2str_n(num, n): # int num, n:桁数 sp_n = ' ' * n # 最大桁数分 スペース # sp_n = '0' * n とする方法あり str_num = sp_n + str(num) # 先頭側に スペース return str_num[-n:] # 末尾の n桁 #---------------------------------------------------------------------------- # 文字列項目を固定桁数に編集・・・文字を左詰め、末尾の余白は、スペース # 20240328 Name Change def strFixL_n(str_x, n): # n:桁数 sp_n = ' ' * n # 最大桁数分 スペース str_x += sp_n # 末尾側に スペース return str_x[:n] # 先頭の n桁 #---------------------------------------------------------------------------- # 文字列項目を固定桁数に編集・・・文字を左詰め、末尾の余白は、0 # 20240328 Name Change def strFixLZ_n(str_x, n): # n:桁数 sp_n = '0' * n # 最大桁数分 0 str_x += sp_n # 末尾側に 0 return str_num[:n] # 先頭の n桁 #---------------------------------------------------------------------------- # 文字列項目を固定桁数に編集・・・文字を右詰め、先頭の余白は、スペース # 20240328 Name Change def strFixR_n(str_x, n): # n:桁数 sp_n = ' ' * n # 最大桁数分 スペース sp_n += str_x # 先頭側に スペース return sp_n[-n:] # 末尾の n桁 #----------------------------------------------------------------------------

ps2kb3.py

ps2kb3.py

############################################################################### # Pico に PS2 Keyboard を接続する # ps2kb.py # Auto Push ではなく、1文字受信毎に Push にしないと、うまくいかなかった。 # PS2Keyboard からの読み込みを文字列からリストに変更 # 2023/10/13 通常入力には耐えられる # # Pico <==> レベル変換 <==> PS2キーボードの接続 # 6-pin Mini-DIN 端子では、4Pin:+5V、3Pin:GND、1Pin:Data、5Pin:Clockである。 # マウスもキーボードと同じPin接続である。 # USB 端子では、1Pin:+5V、2Pin:D-:(Data)、3Pin:D+:(Clock)、4Pin:GND である。 # USB端子と レベル変換 バッファ Pico との接続は次のようにした。 # # ps2KeyBoard と Pico との接続 ps2KeyBoard # Pico レベル変換 USB-A USB-A # (メス) (オス) # 3V 5V GND (4) | | # gpio-13 (17)---clock -- 青------------- LV1 HV1 --- D+ --(3)--| | # gpio-12 (16)---data -- 緑------------- LV2 HV2 --- D- --(2)--| | # +5V (1) |___| # (TopView) # # 2024/05/22:ps2 Mouse 両用するため、 # gpio27(Clock) ==> gpio13(Clock) # gpio26(Data) ==> gpio12(Data) # sm4 ==> sm1 Mouse sm4 起動後に、KeyBoard sm1 を起動する # に 変更した。 # DEBUG用のLEDを3端子用意できないので、gpio-14:LED-R、gpio-15:LED-G のみとした。 # # PS2 キーボードインターフェース # PS2 Keyboard は、+5V、GND、KB_Data、KB_Clock のラインがある。 # KB_Data、KBClock は、オープンコレクタで接続されている。 # 双方向通信であるが、PS2 Keyboard から Pico(PC側)へのデータ送信では、 # KB_Clock の立下りで判断し、KB_Data が Lowなら0、Highなら、1であり # 10ビットで1つのキャラクタを構成する。 # 1ビットのスタートビット(1bit:0) # 8ビットのデータビット (8bit:data) # 1ビットのパリティビット(Parity:ODD) # 1ビットのストップビット(1bit:1) # # Ver:2023/10/14 Program by Hyodo # Ver:2023/10/16 SKB-KG3WN Shift + Arrow # Ver:2023/11/01 簡潔にした # Ver:2024/06/18 # 複数のモジュールから同時利用できるよう、クラス変数を導入した。 # # ############################################################################### # from rp2 import PIO, StateMachine, asm_pio # from micropython import const # # Debug Print True or False # DEBUG_PRINT = True DEBUG_PRINT = False # ############################################################################### # LSB First なので、右シフト:PIO RX-FIFO 上位に受信データがセットされる # 指定のビット数(BIT_CNT)を受信したら、自動的にPIO RX-FIFO にpush するよう # autopush を設定するとダメだった。 # MicroPython 側では、右シフトしてから取り出すこと ############################################################################### @asm_pio(in_shiftdir=PIO.SHIFT_RIGHT) def ps2kb_recive(): # BIT_CNT = 9 # 1文字の受信ビット数(data 8bit, parity 1bit) wrap_target() # 次の2つの命令は、PIO起動時1回のみ実行する。 pull() # osr = timeout value # label("start_frame") # <start_frame> mov(x, osr) # x = timeout value タイマー値をセット # label("start_1chr") # <start_1chr> set(y, BIT_CNT) # y = 9 bit (data 8bit, parity 1bit) jmp(y_dec, "wait_start_bit_L") # y != 0 なら y -= 1 して wait_start_bit_L へ # 回数(ビット数)処理調整のため減算、次へ label("wait_start_bit_L") # <wait_start_bit_L> jmp(pin, "chk_timeout") # ① jmp_pin = kb_clock が 1 なら chk_timeout へ # jmp_pin = kb_clock が 0 の場合は、次へ # label("wait_start_bit_H") # <wait_start_bit_H> wait(1, pin, 1) # in_base(kb_data) +1 = kb_clock が 1 を待つ # start bit end, next recive 1chr label("recive_1chr") # <recive_1chr> wait(0, pin, 1) # in_base(kb_data) +1 = kb_clock が 0 を待つ mov(x, osr) # x = timeout value タイマー値を再度セット in_(pins, 1) # in_base = kb_data から 1bit 読み込み # ISR へ書き込む (LSB First) jmp(y_dec, "wait_start_bit_H") # y != 0 なら y -= 1 して wait_start_bit_H へ # y = 0 なら 次へ (8bit + parity recived) label("chk_stop_bit") # <chk_stop_bit> wait(1, pin, 1) # in_base(kb_data) +1 = kb_clock が 1 を待つ wait(0, pin, 1) # in_base(kb_data) +1 = kb_clock が 0 を待つ wait(1, pin, 1) # in_base(kb_data) +1 = kb_clock が 1 を待つ jmp("start_1chr") # 新たな文字受信待ち start_1chr へ label("chk_timeout") # <chk_timeout> timeout = 1 Frame Recived jmp(x_dec, "wait_start_bit_L") # ② x != 0 なら x -= 1 して wait_start_bit_L へ # x = 0 (=timeout) なら 次へ mov(x, isr) # isr(受信データ)を x レジスタに jmp(not_x, "start_frame") # isr = 0 でならば 受信なし start_frame へ # isr != 0 でならば 受信あり 次へ push() # autopush=True, push_thresh=11 を使用しない # 20240522 Change # irq(0) # 割り込み発生(受信データあり) irq(1) # 割り込み発生(受信データあり) jmp("start_frame") # 新たな受信待ち start_frame へ # sm4 StateMachine -------------------------------------------------------------- class PS2KeyRecive: # class 変数初期設定 「cls.」を付加して参照できる # 関数名(self) ではなく @classmethod 関数名(cls) とする必要あり # (self) で参照する場合は、cls.ではなく、クラス名(PS2KeyRecive.)とする。 key_code = "" # キーコードをクリア(文字列) key_code_ret = "" data_rx1_old = 0x00 data_rx2_old = 0x00 data_rx3_old = 0x00 shift = 0 # 0:直前に Shift キーなし 1:直前に Shift キーあり ctrl = 0 # 0:直前に Ctrl キーなし 1:直前に Ctrl キーあり # PS2KeyBoard Map Dictuionary PS2KeyCode:Value # 直前に Shift 押下なし テーブル(辞書)機能キーである(data_rx2 == 0xE0) PS2KeyMapE = {0x1F:'Win',0x14:'Ctrl',0x11:'Alt',0x27:'Win', 0x2F:'Appli', 0x75:'Arw_U',0x72:'Arw_D',0x6B:'Arw_L',0x74:'Arw_R', 0x7D:'PgUp',0x7A:'PgDn',0x6C:'Home',0x69:'End', 0x71:'Del',0x70:'Ins',0x7C:'PrtSc', 0x4A:'/',0x5A:'Enter',0x7E:'ScrLk', } # 直前に Shift 押下あり テーブル(辞書)機能キーである(data_rx2 == 0xE0) PS2KeyMapF = {0x1F:'Win',0x14:'Ctrl',0x11:'Alt',0x27:'Win', 0x2F:'Appli', 0x75:'PgUp',0x72:'PgDn',0x6B:'Home',0x74:'End', 0x7D:'PgUp',0x7A:'PgDn',0x6C:'Home',0x69:'End', 0x71:'Ins',0x70:'Ins',0x7C:'PrtSc', 0x4A:'/',0x5A:'Enter',0x7E:'ScrLk', } # 直前に Shift 押下 なし テーブル(辞書)機能キーでない(data_rx2 != 0xE0) PS2KeyMapD = {0x16:'1',0x1E:'2',0x26:'3',0x25:'4',0x2E:'5', 0x36:'6',0x3D:'7',0x3E:'8',0x46:'9',0x45:'0', 0x1C:'a',0x32:'b',0x21:'c',0x23:'d',0x24:'e', 0x2B:'f',0x34:'g',0x33:'h',0x43:'i',0x3B:'j', 0x42:'k',0x4B:'l',0x3A:'m',0x31:'n',0x44:'o', 0x4D:'p',0x15:'q',0x2D:'r',0x1B:'s',0x2C:'t', 0x3C:'u',0x2A:'v',0x1D:'w',0x22:'x',0x35:'y', 0x1A:'z',0x4E:'-',0x55:'^',0x6A:'Yen',0x54:'@', 0x5B:'[',0x4C:';',0x52:':',0x5D:']',0x41:',', 0x49:'.',0x4A:'/',0x51:'Yen', 0x70:'0',0x69:'1',0x72:'2',0x7A:'3',0x6B:'4', 0x73:'5',0x74:'6',0x6C:'7',0x75:'8',0x7D:'9', 0x71:'.',0x7C:'*',0x7B:'-',0x79:'+',0x77:'NumLk', 0x05:'F1',0x06:'F2',0x04:'F3',0x0C:'F4',0x03:'F5', 0x0B:'F6',0x83:'F7',0x0A:'F8',0x01:'F9',0x09:'F10', 0x78:'F11',0x07:'F12', 0x66:'BS',0x5A:'Enter',0x29:'Space',0x76:'Esc', 0x0E:'kanji',0x0D:'Tab',0x58:'Caps',0x12:'Shift', 0x14:'Ctrl',0x11:'Alt',0x59:'Shift',0x67:'NoChange', 0x64:'Change',0x13:'Rome',0x7E:'ScrLk', } # 直前に Shift 押下 あり テーブル(辞書)機能キーでない(data_rx2 != 0xE0) PS2KeyMapU = {0x16:'!',0x1E:'"',0x26:'Sharp',0x25:'$',0x2E:'%', 0x36:'&',0x3D:"'",0x3E:'(',0x46:')',0x45:'*', 0x1C:'A',0x32:'B',0x21:'C',0x23:'D',0x24:'E', 0x2B:'F',0x34:'G',0x33:'H',0x43:'I',0x3B:'J', 0x42:'K',0x4B:'L',0x3A:'M',0x31:'N',0x44:'O', 0x4D:'P',0x15:'Q',0x2D:'R',0x1B:'S',0x2C:'T', 0x3C:'U',0x2A:'V',0x1D:'W',0x22:'X',0x35:'Y', 0x1A:'Z',0x4E:'=',0x55:'~',0x6A:'|',0x54:'`', 0x5B:'{',0x4C:'+',0x52:'*',0x5D:'}',0x41:'<', 0x49:'>',0x4A:'?',0x51:'_', 0x70:'0',0x69:'1',0x72:'2',0x7A:'3',0x6B:'4', 0x73:'5',0x74:'6',0x6C:'7',0x75:'8',0x7D:'9', 0x71:'.',0x7C:'*',0x7B:'-',0x79:'+',0x77:'ScrLk', 0x05:'F1',0x06:'F2',0x04:'F3',0x0C:'F4',0x03:'F5', 0x0B:'F6',0x83:'F7',0x0A:'F8',0x01:'F9',0x09:'F10', 0x78:'F11',0x07:'F12', 0x66:'BS',0x5A:'Enter',0x29:'Space',0x76:'Esc', 0x0E:'kanji',0x0D:'Shift_Tab',0x58:'Caps',0x12:'Shift', 0x14:'Ctrl',0x11:'Alt',0x59:'Shift',0x67:'NoChange', 0x64:'Change',0x13:'Rome',0x7E:'ScrLk', } # 直前に Ctrl 押下 あり テーブル(辞書)機能キーでない(data_rx2 != 0xE0) PS2KeyMapC = {0x1C:'Ctrl_A',0x32:'Ctrl_B',0x21:'Ctrl_C',0x23:'Ctrl_D', 0x24:'Ctrl_E',0x2B:'Ctrl_F',0x34:'Ctrl_G',0x33:'Ctrl_H', 0x43:'Ctrl_I',0x3B:'Ctrl_J',0x42:'Ctrl_K',0x4B:'Ctrl_L', 0x3A:'Ctrl_M',0x31:'Ctrl_N',0x44:'Ctrl_O',0x4D:'Ctrl_P', 0x15:'Ctrl_Q',0x2D:'Ctrl_R',0x1B:'Ctrl_S',0x2C:'Ctrl_T', 0x3C:'Ctrl_U',0x2A:'Ctrl_V',0x1D:'Ctrl_W',0x22:'Ctrl_X', 0x35:'Ctrl_Y',0x1A:'Ctrl_Z', } def __init__(self, pio_freq, kb_data, kb_clock): print("PS2KeyRecive:init") self.sm = StateMachine(1, ps2kb_recive, freq=pio_freq, in_base=kb_data, jmp_pin=kb_clock) self.sm.irq(self._handler) # コールバック関数の登録 def set_active(self): # StateMachine 起動 self.sm.active(1) def set_inactive(self): # StateMachine 停止 self.sm.active(0) def put_timer(self, time_out): # time_out 値を PIO OSR にセット self.sm.put(time_out) # 受信サイズ取得 @classmethod def rcv_size(cls): return len(cls.key_code_ret) # 受信データサイズを返す # 受信末尾バッファ取得 @classmethod def get_code(cls): key_ret = cls.key_code_ret # 受信データ読み取りでクリア cls.key_code_ret = "" return key_ret # 取得キーコードを返す # StateMachine から割り込み発生でここにコールバック # 受信データの蓄積 def _handler(self, ev): # PIO からの割り込み 1 Char 受信 # Debug # print("PS2KeyRecive:Irq") # data_rx = self.sm.get() # PIO RX-FIFO からデータを取得 # Parity をチェックする場合、次の部分に工夫が必要 # Paritybit は、上位ビット 9ビット目にある。 data_rx1 = data_rx >> (5) & 0xFF data_rx2 = data_rx >> (14) & 0xFF data_rx3 = data_rx >> (24-1) & 0xFF # # Debug print if DEBUG_PRINT: print("IRQ data_rx = ", data_rx1, data_rx2, data_rx3) # # Shift キーの状態チェック if PS2KeyRecive.data_rx2_old == 0x00 and (PS2KeyRecive.data_rx3_old == 0x12 or PS2KeyRecive.data_rx3_old == 0x59): PS2KeyRecive.shift = 1 # 1:Shift キー押されている elif PS2KeyRecive.data_rx2_old == 0xF0 and (PS2KeyRecive.data_rx3_old == 0x12 or PS2KeyRecive.data_rx3_old == 0x59): PS2KeyRecive.shift = 0 # 0:Shift キー押されてない # Ctrl キーの状態チェック if (PS2KeyRecive.data_rx2_old == 0x00 and PS2KeyRecive.data_rx3_old == 0x14) or (PS2KeyRecive.data_rx2_old == 0xE0 and PS2KeyRecive.data_rx3_old == 0x14): PS2KeyRecive.ctrl = 1 # 1:Ctrl キー押されている elif (PS2KeyRecive.data_rx2_old == 0xF0 and PS2KeyRecive.data_rx3_old == 0x14) or (PS2KeyRecive.data_rx1_old == 0xE0 and PS2KeyRecive.data_rx2_old == 0xF0 and PS2KeyRecive.data_rx3_old == 0x14): PS2KeyRecive.ctrl = 0 # 0:Ctrl キー押されてない # Pause キー 対応 if data_rx1 == 0x14 and data_rx2 == 0xF0 and data_rx3 == 0x77: PS2KeyRecive.key_code = 'Pause' # テーブル(辞書)機能キーである(data_rx2 == 0xE0) elif data_rx2 == 0xE0: # Shift キーの状態チェック if PS2KeyRecive.shift == 1: # Shift 押下 あり テーブル(辞書)を利用 PS2KeyRecive.key_code = PS2KeyRecive.PS2KeyMapF.get(data_rx3) else: # Shift 押下 なし テーブル(辞書)を利用 PS2KeyRecive.key_code = PS2KeyRecive.PS2KeyMapE.get(data_rx3) # 本来は if data_rx2 == 0x00: (キーが押された時)で良いのだが # SANWA SKB-KG3WN OLD では 2,w,s,x ラインが 0x00 ではなく 0x22 が出るので対応 # 20231020:出ていない??? elif data_rx2 == 0x00: if PS2KeyRecive.shift == 1: # Shift 押下 あり テーブル(辞書)を利用 PS2KeyRecive.key_code = PS2KeyRecive.PS2KeyMapU.get(data_rx3) elif PS2KeyRecive.ctrl == 1: # Ctrl 押下 あり テーブル(辞書)を利用 PS2KeyRecive.key_code = PS2KeyRecive.PS2KeyMapC.get(data_rx3) # 多くのキーは、ここになる else: # Shift 押下 なし Ctrl 押下 なし テーブル(辞書)を利用 PS2KeyRecive.key_code = PS2KeyRecive.PS2KeyMapD.get(data_rx3) elif data_rx2 == 0xF0: PS2KeyRecive.key_code = "" # None または Shift キー または Ctrl キー なら出力しない if PS2KeyRecive.key_code == None or PS2KeyRecive.key_code == 'Shift' or PS2KeyRecive.key_code == 'Ctrl': PS2KeyRecive.key_code_ret = "" # Yen Sharp 対応 elif len(PS2KeyRecive.key_code) > 0: if PS2KeyRecive.key_code == 'Yen': # \ Pythonでは、辞書に表現できない PS2KeyRecive.key_code_ret = chr(0x5C) elif PS2KeyRecive.key_code == 'Sharp': # # Pythonでは、辞書に表現できない PS2KeyRecive.key_code_ret = chr(0x23) else: PS2KeyRecive.key_code_ret = PS2KeyRecive.key_code else: PS2KeyRecive.key_code_ret = "" # # Debug print if DEBUG_PRINT: if len(PS2KeyRecive.key_code_ret) > 0: print("Key data = ", PS2KeyRecive.key_code_ret) # PS2KeyRecive.data_rx1_old = data_rx1 PS2KeyRecive.data_rx2_old = data_rx2 PS2KeyRecive.data_rx3_old = data_rx3 #

ps2mus.py

ps2mus.py

############################################################################### # 目標:Pico に PS/2 Mouse を接続する # ps2mus_20240601.py Program by Hyodo # # PS2マウス と Picoの接続 # 6-pin Mini-DIN 端子では、4Pin:+5V、3Pin:GND、1Pin:Data、5Pin:Clockである。 # (2Pin:Data、6Pin:Clock は、間違いであり、キーボードと同じ接続である。) # USB 端子では、1Pin:+5V、2Pin:D-:(Data)、3Pin:D+:(Clock)、4Pin:GND である。 # Pico --- オープンドレインバッファ --- レベル変換 --- USBとの接続は次の通り # # Pico レベル変換 USBコネクタAメス ps/2 Mouse # GND GND GND 4pin # gpio26(Clock) +++++++++++++++++ <<+++++>> +++ D+ 3pin # gpio27(Data) --------- + --- <<----->> --- D- 2pin # 3V + - 3V 5V +5V 1pin # + - ( TopView ) # OpenDrainBuff + - # gpio20 ++ 11pin 10pin ++ - # gpio21 -- 13pin 12pin ----- # ( 74HC07 ) # # Data と Clock は、マウスのオープンコレクタトランジスタに接続されている。 # 双方向通信であるが、PS/2 Mouse から Pico(PC側)へのデータ送信では、Pico は、 # マウスからの Clock の立下りで判断し、Data が Lowなら0、Highなら、1であり # 11ビットで1つのキャラクタを構成する。 # 1ビットのスタートビット(1bit:0) # 8ビットのデータビット (8bit:data) # 1ビットのパリティビット(1bit:ODD Parity、Parityを含めて奇数にする) # 1ビットのストップビット(1bit:1) # # PS2 マウスを動作状態にするには、Pico(PC側)から、リセット・コマンド(0xFF)と # ストリーミング・モード開始コマンド(0xF4)を、マウスに送信しなければならない。 # その制御は やや複雑で、DATA 信号が high レベル状態で、Clockを 100us Low にし、 # 続いて、Dataを Low(=スタートビット送出)すると、マウスがクロックを発生する。 # マウスからの Clockの立下りに合わせて LSB から 順にコマンド・データを出力する。 # データ 8ビット送出し、パリティ・ビットを送出し、 # ストップビットを送出するという流れになる。 # # PS2 スクロール・ホイール対応 # Pico(CPU)側から、マウスに対して、リセット・コマンド(0xFF)を出した後、 # 先に、サンプリング・レイト設定(0xF3, xx)で、 # 10進数で、xx = 200, 100, 80 を続けて出した後に # ストリーミング・モード開始コマンド(0xF4)を出すと、 # スクロール・ホイール機能を有効化できる。 # # マウスの初期設定を PicoMain から、次の4つのコマンドで実行できるようにした。 # ps2mus_cmd.init_ps2cmd_1() # ps2mus_rcv.init_ps2rcv_1(MUS_PIO_FRQ) # ps2mus_cmd.init_ps2cmd_2() # ps2mus_rcv.init_ps2rcv_2(MUS_PIO_FRQ) # # ############################################################################### # from rp2 import PIO, StateMachine, asm_pio from machine import Timer from micropython import const import utime # Debug Print Use = 1 Debug Print NonUse = 0 # DEBUG_PRINT = const(1) DEBUG_PRINT = const(0) ############################################################################### # sm2 Pio -------------------------------------------------------------- # PS/2 Mouse との通信は、LSB First である。 # PIOは、Pythonがセットした TX-FIFOのデータの下位から取り出すので、右シフトにする # MicroPython側は、TX-FIFOの下位側にコマンドデータをセットすること。 # LSB側から start_bit,data8bit,oddparity_bit,stop_bitの順に11ビットに編集する # PIOからマウスへのデータは、out 命令を利用し、上記、11ビットを出力する。 # PIOから出力する clock は、set 命令で出力する。 ############################################################################### # Command Send from Pico to Mouse @asm_pio(set_init=PIO.OUT_HIGH, out_init=PIO.OUT_HIGH, \ out_shiftdir=PIO.SHIFT_RIGHT, in_shiftdir=PIO.SHIFT_RIGHT) def ps2mus_send(): SEND_BITS = 9 # start_bit, data_8bit, parity_bit, stop_bit RECIVE_BITS = 10 # start_bit, data_8bit, parity_bit, stop_bit WAIT_CNT = 30 # 待ち時間用 ERROR_BITS = 7 # エラー応答用 wrap_target() # Block with TX deasserted until data available 送信データがセットされるのを待つ pull() # Python 側からコマンドが到着したら # Pico から マウス側に コマンド送信要求を送出する(clock:0, Data:1 を 120us) set(y, WAIT_CNT) # 待ち時間をセット label("wait") # <wait> set(pins, 0) [2] # send_clock:0 を送出 send_data は初期状態:1 jmp(y_dec, "wait") # y != 0 なら wait へ PIO_freq=1MHzなら120us # y = 0 なら 次へ set(y, SEND_BITS) # ここまでは、Pico から マウスへの一方的な処理=コマンド送信要求である # start_bit 送出し、send_clock:1 を送出 label("tx_start") # <tx_start> out(pins, 1) # ⓪ (OSR で受信した)start_bit 「0」を送出 nop() [3] # 微妙なタイミングを想定して待ち時間挿入 set(pins, 1) # send_clock:1 を送出 # Mouse側が制御する rcv_clock 立下りを待って、data_8bit,parity_bit,stop_bit 送出 # 待ち時間タイマーを仕掛け、永遠ループを避ける set(x, WAIT_CNT) # 待ち時間をセット label("tx_loop") # <tx_loop> ① ~ ⑩ jmp(pin, "chk_timeout") # jmp_pin = rcv_clock が 1 なら chk_timeout へ # rcv_clock が 0 なら次へ set(x, WAIT_CNT) # 待ち時間を再度セット out(pins, 1) # data_8bit,parity_bit,stop_bit 送出 LSB First # Mouse 側からの rcv_clock 立上りを待つ wait(1, pin, 1) # in_base +1 = rcv_clock が 1 になるのを待つ # data_8bit, parity_bit, stop_bit 送出 途中なら tx_loop へ jmp(y_dec, "tx_loop") # y != 0 なら y -= 1 して tx_loop へ # data_8bit,parity_bit,stop_bit送出 終了なら 次へ # Mouse 側からの rcv_clock 立下りを待ち、アクノリッジを受信する wait(0, pin, 1) # in_base +1 = rcv_clock が 0 になるのを待つ in_(pins, 1) # ⑪ in_base = rcv_data から 1bit 読み込み # ISR へ書き込む (LSB First) # Mouse 側からの rcv_clock 立上りを待つ=終了 wait(1, pin, 1) # in_base +1 = rcv_clock が 1 になるのを待つ # 続けてマウスからの応答メッセージを受信する # スタートビットも ストップビットも in_ で isr に取り込む set(y, RECIVE_BITS) # start_bit, data_8bit, parity_bit, stop_bit の 11bit を受信する # この部分に、待ち時間タイマーを仕掛けたいが、プログラム容量なし label("rx_loop") # <rx_loop> ⓪ ~ ⑩ jmp(pin, "rx_loop") # jmp_pin = rcv_clock が 1 なら rx_loop へ # rcv_clock が 0 なら次へ in_(pins, 1) # in_base = rcv_data から 1bit 読み込み # ISR へ書き込む(LSB First) # Mouse 側からの rcv_clock 立上りを待つ wait(1, pin, 1) # in_base +1 = rcv_clock が 1 になるのを待つ # データ+パリティ受信 途中なら rx_loop へ jmp(y_dec, "rx_loop") # y != 0 なら y -= 1 して rx_loop へ # データ+パリティ受信 終了なら 次へ jmp("result_out") # result_out へ label("chk_timeout") # <chk_timeout> jmp(x_dec, "tx_loop") [3] # x != 0 なら y -= 1 して tx_loop へ 5us x 30 # x = 0(rcv_clock が 150us来ない)なら 次へ label("timeout") set(x, ERROR_BITS) # x = 0b111 mov(osr, x) # Data ラインに出力するため osr にコピー out(pins, 1) # Data ラインを Highにする mov(isr, osr) # osr に 残った 0b11 を isr に転送 # 次の命令で Python側に知らせる label("result_out") # <result_out> 結果をセットしPython側に知らせる push() # autopush=True, push_thresh を使用しない irq(2) # 割り込み発生 # アクノリッジデータと応答メッセージをセット # アクノリッジデータは、0:正常、1:異常 wrap() ############################################################################### # sm4 Pio -------------------------------------------------------------- # PS/2 Keyboard や Mouse との通信は、LSB First である。 # PIO RX-FIFOを右シフトで利用するので、上位に受信データがセットされる # MicroPython 側では、右シフトしてから取り出すこと ############################################################################### # Keyboard & Mouse Data Recive @asm_pio(in_shiftdir=PIO.SHIFT_RIGHT) def ps2mus_recive(): BIT_CNT = 7 # 受信ビット数(data 8bit) wrap_target() # 次の命令は、PIO起動時1回のみ実行する。 # Block with TX deasserted until data available 送信データがセットされるのを待つ pull() # osr = timeout value label("start_frame") # <start_frame> wait(1, pin, 1) # in_base(kb_data) +1 = kb_clock が 1 を待つ label("start_1chr") # <start_1chr> mov(x, osr) # x = timeout value タイマー値をセット set(y, BIT_CNT) # y = 7(data 8bit) label("wait_start_bit") # <wait_start_bit> # wait(0, pin, 1) # in_base(kb_data) +1 = kb_clock が 0 を待つ jmp(pin, "chk_timeout1") # ① jmp_pin = kb_clock が 1 なら chk_timeout へ # jmp_pin = kb_clock が 0 の場合は、次へ mov(x, osr) # x = timeout value タイマー値をセット label("wait_clock_H1") # <wait_clock_H1> # wait(1, pin, 1) # in_base(kb_data) +1 = kb_clock が 1 を待つ jmp(pin, "recive_1bit") # ① jmp_pin = kb_clock が 1 なら recive_1bit へ # jmp_pin = kb_clock が 0 の場合は、次へ jmp(x_dec, "wait_clock_H1") # ② x != 0 なら x -= 1 して wait_clock_H1 へ # x = 0 (=timeout) なら 次へ jmp("start_frame") # 中断して、新たな受信待ち start_frame へ # start bit end, recive 1chr label("recive_1bit") # <recive_1bit> # wait(0, pin, 1) # in_base(kb_data) +1 = kb_clock が 0 を待つ jmp(pin, "chk_timeout2") # ① jmp_pin = kb_clock が 1 なら chk_timeout2 へ # jmp_pin = kb_clock が 0 の場合は、次へ in_(pins, 1) # in_base = kb_data から 1bit 読み込み # ISR へ書き込む (LSB First) jmp(y_dec, "wait_clock_H1") # y != 0 なら y -= 1 して wait_clock_H1 へ # y = 0 なら 次へ (8bit recived) # wait parity and stop bit mov(x, osr) # x = timeout value タイマー値をセット set(y, 1) # 2回実行 label("wait_clock_H2") # <wait_clock_H2> # wait(1, pin, 1) # in_base(kb_data) +1 = kb_clock が 1 を待つ jmp(pin, "wait_clock_L2") # ① jmp_pin = kb_clock が 1 なら wait_clock_L2 へ # jmp_pin = kb_clock が 0 の場合は、次へ jmp(x_dec, "wait_clock_H2") # ② x != 0 なら x -= 1 して wait_clock_H2 へ # x = 0 (=timeout) なら 次へ jmp("start_frame") # 中断して、受信待ち start_frame へ label("wait_clock_L2") # <wait_clock_L2> # wait(0, pin, 1) # in_base(kb_data) +1 = kb_clock が 0 を待つ jmp(pin, "chk_timeout3") # ① jmp_pin = kb_clock が 1 なら chk_timeout3 へ # jmp_pin = kb_clock が 0 の場合は、次へ jmp(y_dec, "wait_clock_H2") # y != 0 なら y -= 1 して wait_clock_H2 へ # y = 0 なら 次へ (parity and stop bit recived) label("wait_clock_H3") # <wait_clock_H3> # wait(1, pin, 1) # in_base(kb_data) +1 = kb_clock が 1 を待つ jmp(pin, "start_1chr") # ① jmp_pin = kb_clock が 1 なら start_1chr へ # jmp_pin = kb_clock が 0 の場合は、次へ jmp(x_dec, "wait_clock_H3") # ② x != 0 なら x -= 1 して wait_clock_H3 へ # x = 0 (=timeout) なら 次へ jmp("start_frame") # 中断して、受信待ち start_frame へ label("chk_timeout3") # <chk_timeout3> timeout = clock time over jmp(x_dec, "wait_clock_L2") # ② x != 0 なら x -= 1 して wait_clock_L2 へ # x = 0 (=timeout) なら 次へ jmp("start_frame") # 中断して、受信待ち start_frame へ label("chk_timeout2") # <chk_timeout2> timeout = clock time over jmp(x_dec, "recive_1bit") # ② x != 0 なら x -= 1 して recive_1bit へ # x = 0 (=timeout) なら 次へ jmp("start_frame") # 中断して、受信待ち start_frame へ label("chk_timeout1") # <chk_timeout1> timeout = 1 Frame Recived jmp(x_dec, "wait_start_bit") # ② x != 0 なら x -= 1 して wait_start_bit へ # x = 0 (=timeout) なら 次へ # ステップ数が不足するので以下のチェックは # Python側で処理することにした。 mov(x, isr) # isr(受信データ)を x レジスタに jmp(not_x, "start_frame") # isr = 0 でならば 受信なし start_frame へ # isr != 0 でならば 受信あり 次へ push() # autopush=True, push_thresh=11 を使用しない irq(0) # 割り込み発生(受信データあり) jmp("start_frame") # 1 Frame 受信完了。受信待ち start_frame へ ################################################################################ # sm0 StateMachine ------------------------------------------------------------- class PS2MouseCommand: def __init__(self, pio_freq, send_data, send_clock, rcv_data, rcv_clock): print("PS2MouseSend:init") self.sm = StateMachine(2, ps2mus_send, freq=pio_freq, \ out_base=send_data, set_base=send_clock, in_base=rcv_data, \ jmp_pin=rcv_clock) self.sm.irq(self._handler) # コールバック関数の登録 def set_active(self): # StateMachine 起動 self.sm.active(1) def set_inactive(self): # StateMachine 停止 self.sm.active(0) # マウスへのコマンド送信 # PS/2 Mouse への Command を PIO OSR にセット # start_bit, oddparity, stop_bit を加え、11bit に編集 def put_command(self, cmd): cmd_2 = cmd oddparity = 0 for i in range(8): if cmd_2 & 0x01 == 1: # LSB 1bit が「1」を数える oddparity += 1 cmd_2 >>= 1 # 1bit 右シフト if (oddparity % 2) == 0: # 奇数パリティ oddparity = 1 else: oddparity = 0 msb_data = 0x02 | oddparity # 0X02:stop_bit cmd_data = (msb_data << 8 | cmd) << 1 # 末尾の左シフトが start_bit:0 # Debug print if DEBUG_PRINT: print("PS2 Command = ", cmd_data) # self.sm.put(cmd_data) # 受信サイズ取得・・・必要ないか? def rcv_size(self): # self.key_ret = self.key_code_ret return len(data_resp) # 受信データサイズを返す # レスポンスを取得 def get_response(self): # 受信データ読み取りでクリア self.response = data_resp return self.response # 取得したレスポンスを返す # StateMachine から割り込み発生でここにコールバック # 受信データの蓄積 def _handler(self, ev): # PIO からの割り込み受信 data_rx = self.sm.get() # PIO RX-FIFO からデータを取得 # Parity をチェックする場合、次の部分に工夫が必要 # Paritybit は、データの上位ビット 9ビット目にある。(Parity,MSB...LSB) # if (data_rx & 0xFF) == 0x03: # PIO ERROR SET == 0x03 data_ack = 1 data_resp = 1 else: # stop, Parity, data8bit, start, ACK # Command 直後のアクノレッジ 0 が正常 data_ack = data_rx >> 20 & 0x01 # Command に対するレスポンス 0xFA が正常 data_resp = data_rx >> 22 & 0xFF # # Debug print if DEBUG_PRINT: print("PS2 Mouse IRQ CMD Response = ", data_ack, data_resp) # # 20240531 Add def init_ps2cmd_1(self): print("ps2mus_cmd.put_command(0xFF)") # PS2KeyRecive ドライバの起動 self.set_active() self.put_command(0xFF) utime.sleep_ms(10) # PS2KeyRecive ドライバの停止 self.set_inactive() # 念のために、PS/2 Data & Clock のラインを High にする----------------- # send_data.value(1) # PS2 Mouse Send Data # send_clock.value(1) # PS2 Mouse Send Clock # 20240531 Add def init_ps2cmd_2(self): # マウス・ホイール機能を有効にする 1 -> 3 # PS2KeyRecive ドライバの起動 self.set_active() # マウス ホイール機能を有効にする−1 print("ps2mus_cmd.put_command(0xF3)") self.put_command(0xF3) utime.sleep_ms(10) print("ps2mus_cmd.put_command(200)") self.put_command(200) utime.sleep_ms(10) # マウス・ホイール機能を有効にする−2 print("ps2mus_cmd.put_command(0xF3)") self.put_command(0xF3) utime.sleep_ms(10) print("ps2mus_cmd.put_command(100)") self.put_command(100) utime.sleep_ms(10) # マウス・ホイール機能を有効にする−3 print("ps2mus_cmd.put_command(0xF3)") self.put_command(0xF3) utime.sleep_ms(10) print("ps2mus_cmd.put_command(80)") self.put_command(80) utime.sleep_ms(10) # マウス・ホイール機能を有効にする 1 -> 3 終了 # マウス・ストリーミング・モード開始 print("ps2mus_cmd.put_command(0xF4)") # PS2KeyRecive ドライバの起動 self.set_active() self.put_command(0xF4) utime.sleep_ms(10) # PS2KeyRecive ドライバの停止 self.set_inactive() # 念のために、PS/2 Data & Clock のラインを High にする----------------- # send_data.value(1) # PS2 Mouse Send Data # send_clock.value(1) # PS2 Mouse Send Clock ################################################################################ # sm4 StateMachine ------------------------------------------------------------- class PS2MouseRecive: # class 変数初期設定 「cls.」を付加して参照できる # 関数名(self) ではなく @classmethod 関数名(cls) とする必要あり # (self) で参照する場合は、cls.ではなく、クラス名(PS2MouseRecive.)とする。 btn_l = 0 # 左クリック btn_r = 0 # 右クリック btn_c = 0 # スクロールボタンクリック cnt_x = 0 # マウス移動量:X方向:横方向 cnt_y = 0 # マウス移動量:Y方向:縦方向 cnt_s = 0 # マウス移動量:スクロールホイール data_cnt = 0 # マウスから受信したら カウントアップ # data_rx1_old = 0 # 第1バイト(ボタンクリック) data_rx2_old = 0 # 第2バイト(X方向:横方向) data_rx3_old = 0 # 第3バイト(Y方向:縦方向) data_rx4=old = 0 # 第4バイト(スクロールホイール) def __init__(self, pio_freq, rcv_data, rcv_clock): # print("PS2MouseRecive:init") self.sm = StateMachine(4, ps2mus_recive, freq=pio_freq, \ in_base=rcv_data, jmp_pin=rcv_clock) # self.sm.irq(self._handler) # コールバック関数の登録 def set_active(self): # StateMachine 起動 self.sm.active(1) def set_inactive(self): # StateMachine 停止 self.sm.active(0) def put_timer(self, time_out): # time_out 値を PIO OSR にセット self.sm.put(time_out) # @classmethod を指定することでクラス変数を利用する。 # 関数名(self) ではなく 関数名(cls) とし、変数の先頭に cls. を付ける。 # 受信データ取得とボタン情報のクリア @classmethod def get_data(cls): btn_l = cls.btn_l btn_r = cls.btn_r btn_c = cls.btn_c cls.btn_l = 0 cls.btn_r = 0 cls.btn_c = 0 return (cls.data_cnt, btn_l, btn_r, btn_c, cls.cnt_x, cls.cnt_y, cls.cnt_s) # 座標のセットと積算カウンタのリセット @classmethod def set_cnt(cls, x, y, s): cls.cnt_x = x # マウス移動量:X方向:横方向 cls.cnt_y = y # マウス移動量:Y方向:縦方向 cls.cnt_s = s # マウス移動量:スクロールホイール cls.data_cnt = 0 # StateMachine から割り込み発生でここにコールバック # クラス変数は、先頭に「PS2MouseRecive.」を付加する # 受信データの蓄積 def _handler(self, ev): # PIO からの割り込み受信 data_rx = self.sm.get() # PIO RX-FIFO からデータを取得 # if data_rx != 0: PS2MouseRecive.data_cnt += 1 data_rx1 = data_rx & 0xFF # 第1バイト(ボタンクリック) data_rx2 = data_rx >> 8 & 0xFF # 第2バイト(X方向:横方向) data_rx3 = data_rx >> 16 & 0xFF # 第3バイト(Y方向:縦方向) data_rx4 = data_rx >> 24 & 0xFF # 第4バイト(スクロールホイール) # スクロールの移動量を積算 # 積算値は、set_cnt で設定できる。通常は、0にする? if data_rx2 >= 128: # X方向:横方向の積算 PS2MouseRecive.cnt_x -= (256 - data_rx2) else: PS2MouseRecive.cnt_x += data_rx2 if data_rx3 >= 128: # Y方向:横方向の積算 PS2MouseRecive.cnt_y -= (256 - data_rx3) else: PS2MouseRecive.cnt_y += data_rx3 if data_rx4 >= 128: # スクロールホイールの積算 PS2MouseRecive.cnt_s -= (256 - data_rx4) else: PS2MouseRecive.cnt_s += data_rx4 # Button if data_rx1 & 0x01 != 0: # 左クリック押されているなら 1 PS2MouseRecive.btn_l = 1 #else: # PS2MouseRecive.btn_l = 0 # if data_rx1 & 0x02 != 0: # 右クリック押されているなら 1 PS2MouseRecive.btn_r = 1 #else: # PS2MouseRecive.btn_r = 0 # if data_rx1 & 0x04 != 0: # スクロールボタン押されているなら 1 PS2MouseRecive.btn_c = 1 #else: # PS2MouseRecive.btn_c = 0 # # Debug print if (PS2MouseRecive.data_rx1_old != data_rx1 or PS2MouseRecive.data_rx2_old != data_rx2 or PS2MouseRecive.data_rx3_old != data_rx3 or PS2MouseRecive.data_rx4_old != data_rx4) and DEBUG_PRINT: PS2MouseRecive.data_rx1_old = data_rx1 PS2MouseRecive.data_rx2_old = data_rx2 PS2MouseRecive.data_rx3_old = data_rx3 PS2MouseRecive.data_rx4_old = data_rx4 print("PS2 IRQ data_rx = ", PS2MouseRecive.data_cnt, PS2MouseRecive.btn_l, PS2MouseRecive.btn_r, PS2MouseRecive.btn_c, PS2MouseRecive.cnt_x, PS2MouseRecive.cnt_y, PS2MouseRecive.cnt_s) # 20240531 Add def init_ps2rcv_1(self, mus_pio_frq): # PS2MouseRecive ドライバの起動とタイマー設定 self.set_active() # 1Frame Time の設定 # マウスを連続して動かすと、12ms周期くらいでデータ送信してくる。 # 何もない時間が 4ms続けば区切りとして、インターラプトを発生させる。 # 4ms:(1MHz*1ms*1000us)/PIOクロック数 2 # time_out = int((MUS_PIO_FRQ * 4) / (1000 * 2)) time_out = int((mus_pio_frq * 4) / (1000 * 2)) self.put_timer(time_out) print("ps2mus_rcv.put_timer(time_out) = ", time_out) ########################################################################## # リセット・コマンドに対するマウスのセルフテスト完了の応答は、 # 400ms後くらいになるようだ。 utime.sleep_ms(1000) ########################################################################## # PS2MouseRecive ドライバの停止 self.set_inactive() # 20240531 Add def init_ps2rcv_2(self, mus_pio_frq): # PS2MouseRecive ドライバの起動 self.set_active() # 1Frame Time の設定 # マウスを連続して動かすと、12ms周期くらいでデータ送信してくる。 # 何もない時間が 4ms続けば区切りとして、インターラプトを発生させる。 # 4ms:(1MHz*1ms*1000us)/PIOクロック数 2 # time_out = int((MUS_PIO_FRQ * 4) / (1000 * 2)) time_out = int((mus_pio_frq * 4) / (1000 * 2)) self.put_timer(time_out) print("ps2mus_rcv.put_timer(time_out) = ", time_out) utime.sleep_ms(100)

st7789py.py

st7789py.py

""" st7789 tft driver in MicroPython based on devbis' st7789py_mpy module from https://github.com/devbis/st7789py_mpy. I added support for display rotation, scrolling and drawing text using 8 and 16 bit wide bitmap fonts with heights that are multiples of 8. Included are 12 bitmap fonts derived from classic pc text mode fonts. """ import time from micropython import const import ustruct as struct # commands ST7789_NOP = const(0x00) ST7789_SWRESET = const(0x01) ST7789_RDDID = const(0x04) ST7789_RDDST = const(0x09) ST7789_SLPIN = const(0x10) ST7789_SLPOUT = const(0x11) ST7789_PTLON = const(0x12) ST7789_NORON = const(0x13) ST7789_INVOFF = const(0x20) ST7789_INVON = const(0x21) ST7789_DISPOFF = const(0x28) ST7789_DISPON = const(0x29) ST7789_CASET = const(0x2A) ST7789_RASET = const(0x2B) ST7789_RAMWR = const(0x2C) ST7789_RAMRD = const(0x2E) ST7789_PTLAR = const(0x30) ST7789_VSCRDEF = const(0x33) ST7789_COLMOD = const(0x3A) ST7789_MADCTL = const(0x36) ST7789_VSCSAD = const(0x37) ST7789_MADCTL_MY = const(0x80) ST7789_MADCTL_MX = const(0x40) ST7789_MADCTL_MV = const(0x20) ST7789_MADCTL_ML = const(0x10) ST7789_MADCTL_BGR = const(0x08) ST7789_MADCTL_MH = const(0x04) ST7789_MADCTL_RGB = const(0x00) ST7789_RDID1 = const(0xDA) ST7789_RDID2 = const(0xDB) ST7789_RDID3 = const(0xDC) ST7789_RDID4 = const(0xDD) COLOR_MODE_65K = const(0x50) COLOR_MODE_262K = const(0x60) COLOR_MODE_12BIT = const(0x03) COLOR_MODE_16BIT = const(0x05) COLOR_MODE_18BIT = const(0x06) COLOR_MODE_16M = const(0x07) # Color definitions BLACK = const(0x0000) BLUE = const(0x001F) RED = const(0xF800) GREEN = const(0x07E0) CYAN = const(0x07FF) MAGENTA = const(0xF81F) YELLOW = const(0xFFE0) WHITE = const(0xFFFF) _ENCODE_PIXEL = ">H" _ENCODE_POS = ">HH" _DECODE_PIXEL = ">BBB" _BUFFER_SIZE = const(256) _BIT7 = const(0x80) _BIT6 = const(0x40) _BIT5 = const(0x20) _BIT4 = const(0x10) _BIT3 = const(0x08) _BIT2 = const(0x04) _BIT1 = const(0x02) _BIT0 = const(0x01) def color565(red, green=0, blue=0): """ Convert red, green and blue values (0-255) into a 16-bit 565 encoding. """ try: red, green, blue = red # see if the first var is a tuple/list except TypeError: pass return (red & 0xf8) << 8 | (green & 0xfc) << 3 | blue >> 3 def _encode_pos(x, y): """Encode a postion into bytes.""" return struct.pack(_ENCODE_POS, x, y) def _encode_pixel(color): """Encode a pixel color into bytes.""" return struct.pack(_ENCODE_PIXEL, color) class ST7789(): """ ST7789 driver class Args: spi (spi): spi object width (int): display width height (int): display height reset (pin): reset pin dc (pin): dc pin cs (pin): cs pin backlight(pin): backlight pin xstart (int): display xstart offset ystart (int): display ystart offset rotation (int): display rotation """ def __init__(self, spi, width, height, reset, dc, cs=None, backlight=None, xstart=-1, ystart=-1, rotation=0): """ Initialize display. """ if (width, height) != (240, 240) and (width, height) != (135, 240): raise ValueError( "Unsupported display. Only 240x240 and 135x240 are supported." ) self._display_width = self.width = width self._display_height = self.height = height self.spi = spi self.reset = reset self.dc = dc self.cs = cs self.backlight = backlight self._rotation = rotation % 4 self.xstart = xstart self.ystart = ystart self.hard_reset() self.soft_reset() self.sleep_mode(False) self._set_color_mode(COLOR_MODE_65K|COLOR_MODE_16BIT) time.sleep_ms(50) self.rotation(self._rotation) self.inversion_mode(True) time.sleep_ms(10) self.write(ST7789_NORON) time.sleep_ms(10) if backlight is not None: backlight.value(1) self.fill(0) self.write(ST7789_DISPON) time.sleep_ms(500) def write(self, command=None, data=None): """SPI write to the device: commands and data.""" if self.cs: self.cs.off() if command is not None: self.dc.off() self.spi.write(bytes([command])) if data is not None: self.dc.on() self.spi.write(data) if self.cs: self.cs.on() def hard_reset(self): """ Hard reset display. """ if self.cs: self.cs.off() if self.reset: self.reset.on() time.sleep_ms(50) # 20240320 Before 50 if self.reset: self.reset.off() time.sleep_ms(50) # 20240320 Before 50 if self.reset: self.reset.on() time.sleep_ms(150) # 20240320 Before 150 if self.cs: self.cs.on() def soft_reset(self): """ Soft reset display. """ self.write(ST7789_SWRESET) time.sleep_ms(150) def sleep_mode(self, value): """ Enable or disable display sleep mode. Args: value (bool): if True enable sleep mode. if False disable sleep mode """ if value: self.write(ST7789_SLPIN) else: self.write(ST7789_SLPOUT) def inversion_mode(self, value): """ Enable or disable display inversion mode. Args: value (bool): if True enable inversion mode. if False disable inversion mode """ if value: self.write(ST7789_INVON) else: self.write(ST7789_INVOFF) def _set_color_mode(self, mode): """ Set display color mode. Args: mode (int): color mode COLOR_MODE_65K, COLOR_MODE_262K, COLOR_MODE_12BIT, COLOR_MODE_16BIT, COLOR_MODE_18BIT, COLOR_MODE_16M """ self.write(ST7789_COLMOD, bytes([mode & 0x77])) def rotation(self, rotation): """ Set display rotation. Args: rotation (int): 0-Portrait, 1-Landscape, 2-Inverted Portrait, 3-Inverted Landscape """ self._rotation = rotation % 4 if self._rotation == 0: # Portrait madctl = ST7789_MADCTL_RGB self.width = self._display_width self.height = self._display_height if self._display_width == 135: self.xstart = 52 self.ystart = 40 elif self._rotation == 1: # Landscape madctl = ST7789_MADCTL_MX | ST7789_MADCTL_MV | ST7789_MADCTL_RGB self.width = self._display_height self.height = self._display_width if self._display_width == 135: self.xstart = 40 self.ystart = 53 elif self._rotation == 2: # Inverted Portrait madctl = ST7789_MADCTL_MX | ST7789_MADCTL_MY | ST7789_MADCTL_RGB self.width = self._display_width self.height = self._display_height if self._display_width == 135: self.xstart = 53 self.ystart = 40 else: # Inverted Landscape madctl = ST7789_MADCTL_MV | ST7789_MADCTL_MY | ST7789_MADCTL_RGB self.width = self._display_height self.height = self._display_width if self._display_width == 135: self.xstart = 40 self.ystart = 52 self.write(ST7789_MADCTL, bytes([madctl])) def _set_columns(self, start, end): """ Send CASET (column address set) command to display. Args: start (int): column start address end (int): column end address """ if start <= end <= self.width: self.write(ST7789_CASET, _encode_pos( start+self.xstart, end + self.xstart)) def _set_rows(self, start, end): """ Send RASET (row address set) command to display. Args: start (int): row start address end (int): row end address """ if start <= end <= self.height: self.write(ST7789_RASET, _encode_pos( start+self.ystart, end+self.ystart)) def set_window(self, x0, y0, x1, y1): """ Set window to column and row address. Args: x0 (int): column start address y0 (int): row start address x1 (int): column end address y1 (int): row end address """ self._set_columns(x0, x1) self._set_rows(y0, y1) self.write(ST7789_RAMWR) def vline(self, x, y, length, color): """ Draw vertical line at the given location and color. Args: x (int): x coordinate Y (int): y coordinate length (int): length of line color (int): 565 encoded color """ self.fill_rect(x, y, 1, length, color) def hline(self, x, y, length, color): """ Draw horizontal line at the given location and color. Args: x (int): x coordinate Y (int): y coordinate length (int): length of line color (int): 565 encoded color """ self.fill_rect(x, y, length, 1, color) def pixel(self, x, y, color): """ Draw a pixel at the given location and color. Args: x (int): x coordinate Y (int): y coordinate color (int): 565 encoded color """ self.set_window(x, y, x, y) self.write(None, _encode_pixel(color)) def blit_buffer(self, buffer, x, y, width, height): """ Copy buffer to display at the given location. Args: buffer (bytes): Data to copy to display x (int): Top left corner x coordinate Y (int): Top left corner y coordinate width (int): Width height (int): Height """ self.set_window(x, y, x + width - 1, y + height - 1) self.write(None, buffer) def rect(self, x, y, w, h, color): """ Draw a rectangle at the given location, size and color. Args: x (int): Top left corner x coordinate y (int): Top left corner y coordinate width (int): Width in pixels height (int): Height in pixels color (int): 565 encoded color """ self.hline(x, y, w, color) self.vline(x, y, h, color) self.vline(x + w - 1, y, h, color) self.hline(x, y + h - 1, w, color) def fill_rect(self, x, y, width, height, color): """ Draw a rectangle at the given location, size and filled with color. Args: x (int): Top left corner x coordinate y (int): Top left corner y coordinate width (int): Width in pixels height (int): Height in pixels color (int): 565 encoded color """ self.set_window(x, y, x + width - 1, y + height - 1) chunks, rest = divmod(width * height, _BUFFER_SIZE) pixel = _encode_pixel(color) self.dc.on() if chunks: data = pixel * _BUFFER_SIZE for _ in range(chunks): self.write(None, data) if rest: self.write(None, pixel * rest) def fill(self, color): """ Fill the entire FrameBuffer with the specified color. Args: color (int): 565 encoded color """ self.fill_rect(0, 0, self.width, self.height, color) def line(self, x0, y0, x1, y1, color): """ Draw a single pixel wide line starting at x0, y0 and ending at x1, y1. Args: x0 (int): Start point x coordinate y0 (int): Start point y coordinate x1 (int): End point x coordinate y1 (int): End point y coordinate color (int): 565 encoded color """ steep = abs(y1 - y0) > abs(x1 - x0) if steep: x0, y0 = y0, x0 x1, y1 = y1, x1 if x0 > x1: x0, x1 = x1, x0 y0, y1 = y1, y0 dx = x1 - x0 dy = abs(y1 - y0) err = dx // 2 if y0 < y1: ystep = 1 else: ystep = -1 while x0 <= x1: if steep: self.pixel(y0, x0, color) else: self.pixel(x0, y0, color) err -= dy if err < 0: y0 += ystep err += dx x0 += 1 def vscrdef(self, tfa, vsa, bfa): """ Set Vertical Scrolling Definition. To scroll a 135x240 display these values should be 40, 240, 40. There are 40 lines above the display that are not shown followed by 240 lines that are shown followed by 40 more lines that are not shown. You could write to these areas off display and scroll them into view by changing the TFA, VSA and BFA values. Args: tfa (int): Top Fixed Area vsa (int): Vertical Scrolling Area bfa (int): Bottom Fixed Area """ struct.pack(">HHH", tfa, vsa, bfa) self.write(ST7789_VSCRDEF, struct.pack(">HHH", tfa, vsa, bfa)) def vscsad(self, vssa): """ Set Vertical Scroll Start Address of RAM. Defines which line in the Frame Memory will be written as the first line after the last line of the Top Fixed Area on the display Example: for line in range(40, 280, 1): tft.vscsad(line) utime.sleep(0.01) Args: vssa (int): Vertical Scrolling Start Address """ self.write(ST7789_VSCSAD, struct.pack(">H", vssa)) def _text8(self, font, text, x0, y0, color=WHITE, background=BLACK): """ Internal method to write characters with width of 8 and heights of 8 or 16. Args: font (module): font module to use text (str): text to write x0 (int): column to start drawing at y0 (int): row to start drawing at color (int): 565 encoded color to use for characters background (int): 565 encoded color to use for background """ for char in text: ch = ord(char) if (font.FIRST <= ch < font.LAST and x0+font.WIDTH <= self.width and y0+font.HEIGHT <= self.height): if font.HEIGHT == 8: passes = 1 size = 8 each = 0 else: passes = 2 size = 16 each = 8 for line in range(passes): idx = (ch-font.FIRST)*size+(each*line) buffer = struct.pack('>64H', color if font.FONT[idx] & _BIT7 else background, color if font.FONT[idx] & _BIT6 else background, color if font.FONT[idx] & _BIT5 else background, color if font.FONT[idx] & _BIT4 else background, color if font.FONT[idx] & _BIT3 else background, color if font.FONT[idx] & _BIT2 else background, color if font.FONT[idx] & _BIT1 else background, color if font.FONT[idx] & _BIT0 else background, color if font.FONT[idx+1] & _BIT7 else background, color if font.FONT[idx+1] & _BIT6 else background, color if font.FONT[idx+1] & _BIT5 else background, color if font.FONT[idx+1] & _BIT4 else background, color if font.FONT[idx+1] & _BIT3 else background, color if font.FONT[idx+1] & _BIT2 else background, color if font.FONT[idx+1] & _BIT1 else background, color if font.FONT[idx+1] & _BIT0 else background, color if font.FONT[idx+2] & _BIT7 else background, color if font.FONT[idx+2] & _BIT6 else background, color if font.FONT[idx+2] & _BIT5 else background, color if font.FONT[idx+2] & _BIT4 else background, color if font.FONT[idx+2] & _BIT3 else background, color if font.FONT[idx+2] & _BIT2 else background, color if font.FONT[idx+2] & _BIT1 else background, color if font.FONT[idx+2] & _BIT0 else background, color if font.FONT[idx+3] & _BIT7 else background, color if font.FONT[idx+3] & _BIT6 else background, color if font.FONT[idx+3] & _BIT5 else background, color if font.FONT[idx+3] & _BIT4 else background, color if font.FONT[idx+3] & _BIT3 else background, color if font.FONT[idx+3] & _BIT2 else background, color if font.FONT[idx+3] & _BIT1 else background, color if font.FONT[idx+3] & _BIT0 else background, color if font.FONT[idx+4] & _BIT7 else background, color if font.FONT[idx+4] & _BIT6 else background, color if font.FONT[idx+4] & _BIT5 else background, color if font.FONT[idx+4] & _BIT4 else background, color if font.FONT[idx+4] & _BIT3 else background, color if font.FONT[idx+4] & _BIT2 else background, color if font.FONT[idx+4] & _BIT1 else background, color if font.FONT[idx+4] & _BIT0 else background, color if font.FONT[idx+5] & _BIT7 else background, color if font.FONT[idx+5] & _BIT6 else background, color if font.FONT[idx+5] & _BIT5 else background, color if font.FONT[idx+5] & _BIT4 else background, color if font.FONT[idx+5] & _BIT3 else background, color if font.FONT[idx+5] & _BIT2 else background, color if font.FONT[idx+5] & _BIT1 else background, color if font.FONT[idx+5] & _BIT0 else background, color if font.FONT[idx+6] & _BIT7 else background, color if font.FONT[idx+6] & _BIT6 else background, color if font.FONT[idx+6] & _BIT5 else background, color if font.FONT[idx+6] & _BIT4 else background, color if font.FONT[idx+6] & _BIT3 else background, color if font.FONT[idx+6] & _BIT2 else background, color if font.FONT[idx+6] & _BIT1 else background, color if font.FONT[idx+6] & _BIT0 else background, color if font.FONT[idx+7] & _BIT7 else background, color if font.FONT[idx+7] & _BIT6 else background, color if font.FONT[idx+7] & _BIT5 else background, color if font.FONT[idx+7] & _BIT4 else background, color if font.FONT[idx+7] & _BIT3 else background, color if font.FONT[idx+7] & _BIT2 else background, color if font.FONT[idx+7] & _BIT1 else background, color if font.FONT[idx+7] & _BIT0 else background ) self.blit_buffer(buffer, x0, y0+8*line, 8, 8) x0 += 8 def _text16(self, font, text, x0, y0, color=WHITE, background=BLACK): """ Internal method to draw characters with width of 16 and heights of 16 or 32. Args: font (module): font module to use text (str): text to write x0 (int): column to start drawing at y0 (int): row to start drawing at color (int): 565 encoded color to use for characters background (int): 565 encoded color to use for background """ for char in text: ch = ord(char) if (font.FIRST <= ch < font.LAST and x0+font.WIDTH <= self.width and y0+font.HEIGHT <= self.height): if font.HEIGHT == 16: passes = 2 size = 32 each = 16 else: passes = 4 size = 64 each = 16 for line in range(passes): idx = (ch-font.FIRST)*size+(each*line) buffer = struct.pack('>128H', color if font.FONT[idx] & _BIT7 else background, color if font.FONT[idx] & _BIT6 else background, color if font.FONT[idx] & _BIT5 else background, color if font.FONT[idx] & _BIT4 else background, color if font.FONT[idx] & _BIT3 else background, color if font.FONT[idx] & _BIT2 else background, color if font.FONT[idx] & _BIT1 else background, color if font.FONT[idx] & _BIT0 else background, color if font.FONT[idx+1] & _BIT7 else background, color if font.FONT[idx+1] & _BIT6 else background, color if font.FONT[idx+1] & _BIT5 else background, color if font.FONT[idx+1] & _BIT4 else background, color if font.FONT[idx+1] & _BIT3 else background, color if font.FONT[idx+1] & _BIT2 else background, color if font.FONT[idx+1] & _BIT1 else background, color if font.FONT[idx+1] & _BIT0 else background, color if font.FONT[idx+2] & _BIT7 else background, color if font.FONT[idx+2] & _BIT6 else background, color if font.FONT[idx+2] & _BIT5 else background, color if font.FONT[idx+2] & _BIT4 else background, color if font.FONT[idx+2] & _BIT3 else background, color if font.FONT[idx+2] & _BIT2 else background, color if font.FONT[idx+2] & _BIT1 else background, color if font.FONT[idx+2] & _BIT0 else background, color if font.FONT[idx+3] & _BIT7 else background, color if font.FONT[idx+3] & _BIT6 else background, color if font.FONT[idx+3] & _BIT5 else background, color if font.FONT[idx+3] & _BIT4 else background, color if font.FONT[idx+3] & _BIT3 else background, color if font.FONT[idx+3] & _BIT2 else background, color if font.FONT[idx+3] & _BIT1 else background, color if font.FONT[idx+3] & _BIT0 else background, color if font.FONT[idx+4] & _BIT7 else background, color if font.FONT[idx+4] & _BIT6 else background, color if font.FONT[idx+4] & _BIT5 else background, color if font.FONT[idx+4] & _BIT4 else background, color if font.FONT[idx+4] & _BIT3 else background, color if font.FONT[idx+4] & _BIT2 else background, color if font.FONT[idx+4] & _BIT1 else background, color if font.FONT[idx+4] & _BIT0 else background, color if font.FONT[idx+5] & _BIT7 else background, color if font.FONT[idx+5] & _BIT6 else background, color if font.FONT[idx+5] & _BIT5 else background, color if font.FONT[idx+5] & _BIT4 else background, color if font.FONT[idx+5] & _BIT3 else background, color if font.FONT[idx+5] & _BIT2 else background, color if font.FONT[idx+5] & _BIT1 else background, color if font.FONT[idx+5] & _BIT0 else background, color if font.FONT[idx+6] & _BIT7 else background, color if font.FONT[idx+6] & _BIT6 else background, color if font.FONT[idx+6] & _BIT5 else background, color if font.FONT[idx+6] & _BIT4 else background, color if font.FONT[idx+6] & _BIT3 else background, color if font.FONT[idx+6] & _BIT2 else background, color if font.FONT[idx+6] & _BIT1 else background, color if font.FONT[idx+6] & _BIT0 else background, color if font.FONT[idx+7] & _BIT7 else background, color if font.FONT[idx+7] & _BIT6 else background, color if font.FONT[idx+7] & _BIT5 else background, color if font.FONT[idx+7] & _BIT4 else background, color if font.FONT[idx+7] & _BIT3 else background, color if font.FONT[idx+7] & _BIT2 else background, color if font.FONT[idx+7] & _BIT1 else background, color if font.FONT[idx+7] & _BIT0 else background, color if font.FONT[idx+8] & _BIT7 else background, color if font.FONT[idx+8] & _BIT6 else background, color if font.FONT[idx+8] & _BIT5 else background, color if font.FONT[idx+8] & _BIT4 else background, color if font.FONT[idx+8] & _BIT3 else background, color if font.FONT[idx+8] & _BIT2 else background, color if font.FONT[idx+8] & _BIT1 else background, color if font.FONT[idx+8] & _BIT0 else background, color if font.FONT[idx+9] & _BIT7 else background, color if font.FONT[idx+9] & _BIT6 else background, color if font.FONT[idx+9] & _BIT5 else background, color if font.FONT[idx+9] & _BIT4 else background, color if font.FONT[idx+9] & _BIT3 else background, color if font.FONT[idx+9] & _BIT2 else background, color if font.FONT[idx+9] & _BIT1 else background, color if font.FONT[idx+9] & _BIT0 else background, color if font.FONT[idx+10] & _BIT7 else background, color if font.FONT[idx+10] & _BIT6 else background, color if font.FONT[idx+10] & _BIT5 else background, color if font.FONT[idx+10] & _BIT4 else background, color if font.FONT[idx+10] & _BIT3 else background, color if font.FONT[idx+10] & _BIT2 else background, color if font.FONT[idx+10] & _BIT1 else background, color if font.FONT[idx+10] & _BIT0 else background, color if font.FONT[idx+11] & _BIT7 else background, color if font.FONT[idx+11] & _BIT6 else background, color if font.FONT[idx+11] & _BIT5 else background, color if font.FONT[idx+11] & _BIT4 else background, color if font.FONT[idx+11] & _BIT3 else background, color if font.FONT[idx+11] & _BIT2 else background, color if font.FONT[idx+11] & _BIT1 else background, color if font.FONT[idx+11] & _BIT0 else background, color if font.FONT[idx+12] & _BIT7 else background, color if font.FONT[idx+12] & _BIT6 else background, color if font.FONT[idx+12] & _BIT5 else background, color if font.FONT[idx+12] & _BIT4 else background, color if font.FONT[idx+12] & _BIT3 else background, color if font.FONT[idx+12] & _BIT2 else background, color if font.FONT[idx+12] & _BIT1 else background, color if font.FONT[idx+12] & _BIT0 else background, color if font.FONT[idx+13] & _BIT7 else background, color if font.FONT[idx+13] & _BIT6 else background, color if font.FONT[idx+13] & _BIT5 else background, color if font.FONT[idx+13] & _BIT4 else background, color if font.FONT[idx+13] & _BIT3 else background, color if font.FONT[idx+13] & _BIT2 else background, color if font.FONT[idx+13] & _BIT1 else background, color if font.FONT[idx+13] & _BIT0 else background, color if font.FONT[idx+14] & _BIT7 else background, color if font.FONT[idx+14] & _BIT6 else background, color if font.FONT[idx+14] & _BIT5 else background, color if font.FONT[idx+14] & _BIT4 else background, color if font.FONT[idx+14] & _BIT3 else background, color if font.FONT[idx+14] & _BIT2 else background, color if font.FONT[idx+14] & _BIT1 else background, color if font.FONT[idx+14] & _BIT0 else background, color if font.FONT[idx+15] & _BIT7 else background, color if font.FONT[idx+15] & _BIT6 else background, color if font.FONT[idx+15] & _BIT5 else background, color if font.FONT[idx+15] & _BIT4 else background, color if font.FONT[idx+15] & _BIT3 else background, color if font.FONT[idx+15] & _BIT2 else background, color if font.FONT[idx+15] & _BIT1 else background, color if font.FONT[idx+15] & _BIT0 else background ) self.blit_buffer(buffer, x0, y0+8*line, 16, 8) x0 += font.WIDTH def text(self, font, text, x0, y0, color=WHITE, background=BLACK): """ Draw text on display in specified font and colors. 8 and 16 bit wide fonts are supported. Args: font (module): font module to use. text (str): text to write x0 (int): column to start drawing at y0 (int): row to start drawing at color (int): 565 encoded color to use for characters background (int): 565 encoded color to use for background """ if font.WIDTH == 8: self._text8(font, text, x0, y0, color, background) else: self._text16(font, text, x0, y0, color, background)

ymf825py.py

ymf825py.py

############################################################################# # YMF825 synthesizer with Raspberry Pi PICO W. # # Hardware Information: # YMF825 bottom view # 1 2 3 4 5 6 7 8 9 # -+-+-+-+-+-+-+-+-+- # | S M M C G V R | # | S O I L N C S | # | S S K D C T | # | I O | # | Audio Jack # ---------------+--- # | # ---Amp # # PICO GPIO(pin) YMF825 name(pin) LED(n/a) # 5V (40) VCC (6) # GND (38) GND (5) # SPI1 MOSI 19(25) SPI MOSI (2) # SPI1 MISO 16(21) SPI MISO (3) # SPI1 CLK 18( 24) SPI CLK (4) # SPI1 CS 17(22) SS (1) # RESET 22(29) RESET (7) #n/a LED 28(34) Anode---1K--->|---Cathode---GND # # Copyright (c) by Shunsuke Ohira # 00.100 2023/08/05: Play Do-Re-Mi # # 20240326 Change Voice_Op_Orig & VoiceZ_Op_Edit # # 20241014 Note On Velocity 対応 ・・・ 固定値(0x54)から指定値(vol)対応へ # ############################################################################# from machine import SPI, UART, Pin import time import utime import uos from micropython import const import mojipy as moji ############################################################################### # 上記インポートファイルは、予め、Picoに書き込んでおくこと # Thonnyで 当該ファイルを開いて、Picoに書き込む。 # 直下に # mojipy.py : 文字処理用 ############################################################################### # YMF825 FM_OP_NUM = const(4) # number of FM operators OP_PARAM_NUM = const(10) # number of operater parameters TONE_NUM = const(16) # number of tones TONE_OP_NUM = const(4) # number of TONE operators TONE_PARAM_NUM = const(17) # number of tone parameters TONE_HEADER = const(16) # ToneID(1) + ToneName(15) ToneName = [None] * (TONE_NUM +1) ToneName[0] = 'GRAND_PIANO' ToneName[1] = 'E_PIANO' ToneName[2] = 'TENOR_SAX' ToneName[3] = 'PICK_BASS' ToneName[4] = 'TNKL_BELL' ToneName[5] = 'NEW_AGE_PD' ToneName[6] = 'BRIGHT_PIANO' ToneName[7] = 'VIBES' ToneName[8] = 'CHURCH_ORGAN' ToneName[9] = 'FLUTE' ToneName[10] = 'ROCK_ORGAN' ToneName[11] = 'NYLON_GUITER' ToneName[12] = 'SQUARE_LEAD' ToneName[13] = 'SAW_LEAD' ToneName[14] = 'HARPSICHORD' ToneName[15] = 'HARMONICA' ToneName[16] = 'No Tone' TONE_MENU_COL_NUM = const(2) # number of tones menu TONE_MENU_ROW_NUM = const(17) # number of tones menu # ToneMenuTable の生成 ToneMenuTable = [None] * TONE_MENU_ROW_NUM # 17 for i in range(TONE_MENU_ROW_NUM): ToneMenuTable[i] = [None] * TONE_MENU_COL_NUM # 2 ToneMenuTable[0][1] = ' Select Tone' for i in range(1, TONE_MENU_ROW_NUM): ToneMenuTable[i][0] = i -1 ToneMenuTable[i][1] = ToneName[i -1] TONE_TABLE_COL_NUM = const(2) # number of tones menu TONE_TABLE_ROW_NUM = const(17) # number of tones menu # ToneTavle の生成 ToneTable = [None] * TONE_TABLE_ROW_NUM # 17 for i in range(TONE_TABLE_ROW_NUM): ToneTable[i] = [None] * TONE_TABLE_COL_NUM # 2 ToneTable[0][1] = ' Select Tone' for i in range(1, TONE_TABLE_ROW_NUM): ToneTable[i][0] = i -1 # ToneTable[i][1] = 16 # 16:No Tone ToneTable[i][1] = 0 # 0:GRAND PIANO # Original Tone Data by YAMAHA # | 00 | 01 | 02 | 03 | 04 | 05 | 06 | 07 | Tone_Com_Orig = [0x0b,0x0d,0x0d,0x0b,0x0d,0x0d,0x0b,0x0d, # | 08 | 09 | 10 | 11 | 12 | 13 | 14 | 15 | 0x0f,0x0d,0x17,0x15,0x15,0x15,0x0e,0x14] Tone_Op_Orig = [ # 00:GrandPiano [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x01,0x0f,0x07,0x00,0x06,0x0f,0x27,0x00,0x01,0x08], # op2 [0x07,0x0e,0x03,0x02,0x03,0x02,0x28,0x00,0x05,0x00], # op3 [0x00,0x0d,0x01,0x01,0x04,0x03,0x22,0x01,0x01,0x00], # op4 [0x06,0x0d,0x02,0x02,0x06,0x04,0x00,0x01,0x01,0x00] ], # 01:E.Piano [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x54,0x0f,0x04,0x05,0x0c,0x0b,0x23,0x44,0x07,0x12], # op2 [0x02,0x0f,0x02,0x01,0x08,0x0f,0x04,0x45,0x01,0x00], # op3 [0x25,0x0f,0x00,0x01,0x0b,0x01,0x12,0x44,0x01,0x00], # op4 [0x04,0x0f,0x02,0x01,0x07,0x0f,0x04,0x41,0x01,0x00] ], # 02:TenorSax [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x36,0x07,0x03,0x00,0x00,0x00,0x05,0x44,0x01,0x01], # op2 [0x00,0x07,0x02,0x00,0x09,0x00,0x0f,0x43,0x01,0x08], # op3 [0x36,0x07,0x03,0x00,0x00,0x00,0x08,0x44,0x01,0x09], # op4 [0x02,0x07,0x02,0x00,0x09,0x00,0x0d,0x43,0x01,0x00] ], # 03:PickBass [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x56,0x0f,0x07,0x02,0x03,0x01,0x13,0x44,0x01,0x00], # op2 [0x04,0x0c,0x0b,0x04,0x06,0x07,0x15,0x44,0x07,0x00], # op3 [0x06,0x0f,0x09,0x02,0x06,0x02,0x17,0x44,0x02,0x00], # op4 [0x04,0x0b,0x02,0x06,0x08,0x06,0x00,0x44,0x01,0x00] ], # 04:TnklBell [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x31,0x0f,0x06,0x03,0x04,0x05,0x10,0x44,0x0e,0x00], # op2 [0x02,0x0c,0x06,0x07,0x06,0x0e,0x0b,0x44,0x02,0x00], # op3 [0x00,0x0c,0x06,0x02,0x02,0x05,0x1e,0x44,0x77,0x01], # op4 [0x00,0x0f,0x05,0x04,0x05,0x0d,0x01,0x54,0x06,0x00] ], # 05:NewAgePd [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x54,0x0f,0x0f,0x03,0x03,0x00,0x26,0x44,0x07,0x01], # op2 [0x02,0x0f,0x07,0x04,0x04,0x00,0x0b,0x44,0x05,0x00], # op3 [0x62,0x06,0x01,0x00,0x01,0x00,0x18,0x03,0x71,0x01], # op4 [0x02,0x08,0x01,0x00,0x05,0x01,0x00,0x03,0x01,0x00] ], # 06:BrightPiano [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x06,0x0f,0x02,0x02,0x02,0x05,0x27,0x00,0x01,0x00], # op2 [0x02,0x0f,0x02,0x02,0x03,0x0f,0x1c,0x00,0x05,0x00], # op3 [0x06,0x0f,0x02,0x02,0x02,0x0d,0x19,0x01,0x01,0x00], # op4 [0x04,0x0f,0x02,0x01,0x05,0x04,0x0a,0x01,0x01,0x00] ], # 07:Vibes [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x00,0x0c,0x04,0x02,0x04,0x02,0x17,0x41,0x07,0x00], # op2 [0x06,0x0d,0x09,0x02,0x05,0x06,0x07,0x51,0x04,0x00], # op3 [0x04,0x0c,0x04,0x02,0x03,0x02,0x1e,0x51,0x08,0x00], # op4 [0x00,0x0d,0x02,0x03,0x04,0x0f,0x07,0x31,0x01,0x00] ], # 08:ChurchOrgan [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x02,0x09,0x0f,0x00,0x05,0x00,0x13,0x00,0x03,0x00], # op2 [0x00,0x0b,0x07,0x00,0x02,0x02,0x1d,0x00,0x07,0x00], # op3 [0x02,0x08,0x0f,0x00,0x05,0x00,0x04,0x00,0x01,0x00], # op4 [0x02,0x08,0x07,0x00,0x05,0x00,0x04,0x00,0x00,0x05] ], # 09:Flute [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x70,0x0d,0x0a,0x00,0x01,0x01,0x07,0x10,0x03,0x00], # op2 [0x00,0x07,0x08,0x00,0x0b,0x03,0x25,0x01,0x03,0x00], # op3 [0x70,0x0e,0x08,0x00,0x09,0x00,0x27,0x31,0x01,0x00], # op4 [0x00,0x06,0x05,0x00,0x0a,0x00,0x01,0x11,0x01,0x10] ], # 10:RockOrgan [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x41,0x0f,0x0f,0x00,0x0d,0x00,0x09,0x70,0x31,0x00], # op2 [0x01,0x0b,0x0f,0x00,0x0a,0x01,0x05,0x50,0x01,0x15], # op3 [0x00,0x0f,0x0f,0x00,0x0e,0x00,0x09,0x13,0x62,0x00], # op4 [0x00,0x0f,0x0f,0x00,0x0e,0x00,0x09,0x33,0x70,0x11] ], # 11:NylonGuiter [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x65,0x0e,0x01,0x01,0x04,0x08,0x15,0x00,0x01,0x00], # op2 [0x00,0x0f,0x03,0x03,0x07,0x0f,0x00,0x01,0x01,0x00], # op3 [0x04,0x0b,0x05,0x05,0x05,0x04,0x0e,0x00,0x03,0x00], # op4 [0x02,0x0d,0x04,0x04,0x09,0x0f,0x0d,0x03,0x01,0x00] ], # 12:SquareLead [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x03,0x0f,0x0f,0x00,0x07,0x04,0x2e,0x00,0x01,0x08], # op2 [0x02,0x0a,0x0f,0x00,0x0a,0x00,0x03,0x03,0x01,0x08], # op3 [0x13,0x0f,0x0f,0x00,0x02,0x03,0x26,0x10,0x02,0x08], # op4 [0x02,0x0a,0x0f,0x00,0x0a,0x00,0x03,0x01,0x01,0x08] ], # 13:SawLead [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x70,0x0f,0x00,0x00,0x07,0x00,0x1a,0x41,0x71,0x00], # op2 [0x00,0x0d,0x0f,0x00,0x07,0x00,0x0a,0x41,0x71,0x08], # op3 [0x71,0x0f,0x0f,0x00,0x03,0x00,0x14,0x41,0x01,0x1c], # op4 [0x00,0x0e,0x0f,0x00,0x08,0x00,0x0a,0x41,0x01,0x08] ], # 14:Harpsichord [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x45,0x0e,0x02,0x02,0x05,0x00,0x00,0x00,0x01,0x04], # op2 [0x07,0x0f,0x02,0x00,0x05,0x03,0x14,0x00,0x06,0x03], # op3 [0x06,0x0f,0x03,0x00,0x01,0x06,0x1c,0x00,0x07,0x04], # op4 [0x04,0x0e,0x02,0x02,0x07,0x0f,0x04,0x03,0x01,0x05] ], # 15:Harmonica [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x03,0x0f,0x0f,0x00,0x09,0x00,0x2c,0x07,0x0e,0x00], # op2 [0x00,0x0f,0x0f,0x00,0x08,0x00,0x29,0x11,0x0a,0x00], # op3 [0x00,0x0f,0x0f,0x00,0x08,0x00,0x24,0x03,0x01,0x00], # op4 [0x02,0x06,0x0f,0x00,0x08,0x00,0x03,0x01,0x02,0x00] ] ] # | 00 | 01 | 02 | 03 | 04 | 05 | 06 | 07 | Tone_Com_Edit = [0x0b,0x0d,0x0d,0x0b,0x0d,0x0d,0x0b,0x0d, # | 08 | 09 | 10 | 11 | 12 | 13 | 14 | 15 | 0x0f,0x0d,0x17,0x15,0x15,0x15,0x0e,0x14] Tone_Op_Edit = [ # 00:GrandPiano [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x01,0x0f,0x07,0x00,0x06,0x0f,0x27,0x00,0x01,0x08], # op2 [0x07,0x0e,0x03,0x02,0x03,0x02,0x28,0x00,0x05,0x00], # op3 [0x00,0x0d,0x01,0x01,0x04,0x03,0x22,0x01,0x01,0x00], # op4 [0x06,0x0d,0x02,0x02,0x06,0x04,0x00,0x01,0x01,0x00] ], # 01:E.Piano [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x54,0x0f,0x04,0x05,0x0c,0x0b,0x23,0x44,0x07,0x12], # op2 [0x02,0x0f,0x02,0x01,0x08,0x0f,0x04,0x45,0x01,0x00], # op3 [0x25,0x0f,0x00,0x01,0x0b,0x01,0x12,0x44,0x01,0x00], # op4 [0x04,0x0f,0x02,0x01,0x07,0x0f,0x04,0x41,0x01,0x00] ], # 02:TenorSax [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x36,0x07,0x03,0x00,0x00,0x00,0x05,0x44,0x01,0x01], # op2 [0x00,0x07,0x02,0x00,0x09,0x00,0x0f,0x43,0x01,0x08], # op3 [0x36,0x07,0x03,0x00,0x00,0x00,0x08,0x44,0x01,0x09], # op4 [0x02,0x07,0x02,0x00,0x09,0x00,0x0d,0x43,0x01,0x00] ], # 03:PickBass [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x56,0x0f,0x07,0x02,0x03,0x01,0x13,0x44,0x01,0x00], # op2 [0x04,0x0c,0x0b,0x04,0x06,0x07,0x15,0x44,0x07,0x00], # op3 [0x06,0x0f,0x09,0x02,0x06,0x02,0x17,0x44,0x02,0x00], # op4 [0x04,0x0b,0x02,0x06,0x08,0x06,0x00,0x44,0x01,0x00] ], # 04:TnklBell [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x31,0x0f,0x06,0x03,0x04,0x05,0x10,0x44,0x0e,0x00], # op2 [0x02,0x0c,0x06,0x07,0x06,0x0e,0x0b,0x44,0x02,0x00], # op3 [0x00,0x0c,0x06,0x02,0x02,0x05,0x1e,0x44,0x77,0x01], # op4 [0x00,0x0f,0x05,0x04,0x05,0x0d,0x01,0x54,0x06,0x00] ], # 05:NewAgePd [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x54,0x0f,0x0f,0x03,0x03,0x00,0x26,0x44,0x07,0x01], # op2 [0x02,0x0f,0x07,0x04,0x04,0x00,0x0b,0x44,0x05,0x00], # op3 [0x62,0x06,0x01,0x00,0x01,0x00,0x18,0x03,0x71,0x01], # op4 [0x02,0x08,0x01,0x00,0x05,0x01,0x00,0x03,0x01,0x00] ], # 06:BrightPiano [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x06,0x0f,0x02,0x02,0x02,0x05,0x27,0x00,0x01,0x00], # op2 [0x02,0x0f,0x02,0x02,0x03,0x0f,0x1c,0x00,0x05,0x00], # op3 [0x06,0x0f,0x02,0x02,0x02,0x0d,0x19,0x01,0x01,0x00], # op4 [0x04,0x0f,0x02,0x01,0x05,0x04,0x0a,0x01,0x01,0x00] ], # 07:Vibes [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x00,0x0c,0x04,0x02,0x04,0x02,0x17,0x41,0x07,0x00], # op2 [0x06,0x0d,0x09,0x02,0x05,0x06,0x07,0x51,0x04,0x00], # op3 [0x04,0x0c,0x04,0x02,0x03,0x02,0x1e,0x51,0x08,0x00], # op4 [0x00,0x0d,0x02,0x03,0x04,0x0f,0x07,0x31,0x01,0x00] ], # 08:ChurchOrgan [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x02,0x09,0x0f,0x00,0x05,0x00,0x13,0x00,0x03,0x00], # op2 [0x00,0x0b,0x07,0x00,0x02,0x02,0x1d,0x00,0x07,0x00], # op3 [0x02,0x08,0x0f,0x00,0x05,0x00,0x04,0x00,0x01,0x00], # op4 [0x02,0x08,0x07,0x00,0x05,0x00,0x04,0x00,0x00,0x05] ], # 09:Flute [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x70,0x0d,0x0a,0x00,0x01,0x01,0x07,0x10,0x03,0x00], # op2 [0x00,0x07,0x08,0x00,0x0b,0x03,0x25,0x01,0x03,0x00], # op3 [0x70,0x0e,0x08,0x00,0x09,0x00,0x27,0x31,0x01,0x00], # op4 [0x00,0x06,0x05,0x00,0x0a,0x00,0x01,0x11,0x01,0x10] ], # 10:RockOrgan [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x41,0x0f,0x0f,0x00,0x0d,0x00,0x09,0x70,0x31,0x00], # op2 [0x01,0x0b,0x0f,0x00,0x0a,0x01,0x05,0x50,0x01,0x15], # op3 [0x00,0x0f,0x0f,0x00,0x0e,0x00,0x09,0x13,0x62,0x00], # op4 [0x00,0x0f,0x0f,0x00,0x0e,0x00,0x09,0x33,0x70,0x11] ], # 11:NylonGuiter [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x65,0x0e,0x01,0x01,0x04,0x08,0x15,0x00,0x01,0x00], # op2 [0x00,0x0f,0x03,0x03,0x07,0x0f,0x00,0x01,0x01,0x00], # op3 [0x04,0x0b,0x05,0x05,0x05,0x04,0x0e,0x00,0x03,0x00], # op4 [0x02,0x0d,0x04,0x04,0x09,0x0f,0x0d,0x03,0x01,0x00] ], # 12:SquareLead [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x03,0x0f,0x0f,0x00,0x07,0x04,0x2e,0x00,0x01,0x08], # op2 [0x02,0x0a,0x0f,0x00,0x0a,0x00,0x03,0x03,0x01,0x08], # op3 [0x13,0x0f,0x0f,0x00,0x02,0x03,0x26,0x10,0x02,0x08], # op4 [0x02,0x0a,0x0f,0x00,0x0a,0x00,0x03,0x01,0x01,0x08] ], # 13:SawLead [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x70,0x0f,0x00,0x00,0x07,0x00,0x1a,0x41,0x71,0x00], # op2 [0x00,0x0d,0x0f,0x00,0x07,0x00,0x0a,0x41,0x71,0x08], # op3 [0x71,0x0f,0x0f,0x00,0x03,0x00,0x14,0x41,0x01,0x1c], # op4 [0x00,0x0e,0x0f,0x00,0x08,0x00,0x0a,0x41,0x01,0x08] ], # 14:Harpsichord [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x45,0x0e,0x02,0x02,0x05,0x00,0x00,0x00,0x01,0x04], # op2 [0x07,0x0f,0x02,0x00,0x05,0x03,0x14,0x00,0x06,0x03], # op3 [0x06,0x0f,0x03,0x00,0x01,0x06,0x1c,0x00,0x07,0x04], # op4 [0x04,0x0e,0x02,0x02,0x07,0x0f,0x04,0x03,0x01,0x05] ], # 15:Harmonica [ # KC | AR | DR | SR | RR | SL | TL | VB | PT | WS # op1 [0x03,0x0f,0x0f,0x00,0x09,0x00,0x2c,0x07,0x0e,0x00], # op2 [0x00,0x0f,0x0f,0x00,0x08,0x00,0x29,0x11,0x0a,0x00], # op3 [0x00,0x0f,0x0f,0x00,0x08,0x00,0x24,0x03,0x01,0x00], # op4 [0x02,0x06,0x0f,0x00,0x08,0x00,0x03,0x01,0x02,0x00] ] ] # Tone Data 20240516 Change tone_data = bytearray(485) # 20240518 Add send_tone = [] rcv_tone = [] UART_TONE_NUM = const(42) # 整数の送信データサイズ UART_BUFF_SIZE = const(100) # UART ASCII の送信データサイズ send_buff = bytearray(UART_BUFF_SIZE) # ASCII 2文字のバッファ rcv_buff = bytearray(UART_BUFF_SIZE) # ASCII 2文字のバッファ # 20240619 Add uartT_buff = '' # uart ASCII 2文字の送信バッファ uartR_buff = '' # uart ASCII 2文字の受信バッファ # VOICE COMMON PARAMETER ROW s Number COM = const(0) BO = const(1) LFO = const(2) ALG = const(3) VOICE_COM_COL_NUM = const(4) # VOICE COMMON PARAMETER COLUMON s Number CM0 = const(0) # Oparator Name CM1 = const(1) # Edit CM2 = const(2) # Orig CM3 = const(3) # Max VOICE_COM_ROW_NUM = const(4) # 20240326 Change # VOICE OPTIONAL PARAMETER ROW s Number OPT = const(0) SR = const(1) XOF = const(2) KSR = const(3) RR = const(4) DR = const(5) AR = const(6) SL = const(7) TL = const(8) KSL = const(9) DAM = const(10) EAM = const(11) DVB = const(12) EVB = const(13) MUL = const(14) DT = const(15) WS = const(16) FB = const(17) VOICE_OP_ROW_NUM = const(18) # VOICE OPTIONAL PARAMETER COLUMN s Number OP0 = const(0) # Oparator Name OP1 = const(1) # Edit 1 OP2 = const(2) # Edit 2 OP3 = const(3) # Edit 3 OP4 = const(4) # Edit 4 OP5 = const(5) # Orig 1 OP6 = const(6) # Orig 2 OP7 = const(7) # Orig 3 OP8 = const(8) # Orig 4 OP9 = const(9) # Max VOICE_OP_COL_NUM = const(10) # 編集用テーブルの生成 # Voice_Com_Edit テーブルの生成 Voice_Com_Edit = [None] * VOICE_COM_ROW_NUM # 4 for i in range(VOICE_COM_ROW_NUM): Voice_Com_Edit[i] = [None] * VOICE_COM_COL_NUM # 4 # VOICE COMMON PARAMETER TABLE ROW s NAME # Voice_Com_Edit[COM][CM0] = 'Common' Voice_Com_Edit[COM][CM1] = 'Edt' Voice_Com_Edit[COM][CM2] = 'Org' Voice_Com_Edit[COM][CM3] = 'Max' # VOICE COMMON PARAMETER ROW s NAME VoiceComName = [None] * VOICE_COM_ROW_NUM VoiceComName[COM] = 'Common' VoiceComName[BO] = 'BO' VoiceComName[LFO] = 'LFO' VoiceComName[ALG] = 'ALG' # VOICE COMMON PARAMETER TABLE COLUMN s NAME for i in range(VOICE_COM_ROW_NUM): # i : 0 - 3 Voice_Com_Edit[i][CM0] = VoiceComName[i] # Voice_Op_Edit テーブルの生成 Voice_Op_Edit = [None] * VOICE_OP_ROW_NUM # 18 for i in range(VOICE_OP_ROW_NUM): Voice_Op_Edit[i] = [None] * VOICE_OP_COL_NUM # 10 # VOICE OPARATOR PARAMETER TABLE ROW s NAME # Voice_Op_Edit[OPT][OP0] = 'Optional' Voice_Op_Edit[OPT][OP1] = 'e1' Voice_Op_Edit[OPT][OP2] = 'e2' Voice_Op_Edit[OPT][OP3] = 'e3' Voice_Op_Edit[OPT][OP4] = 'e4' Voice_Op_Edit[OPT][OP5] = 'o1' Voice_Op_Edit[OPT][OP6] = 'o2' Voice_Op_Edit[OPT][OP7] = 'o3' Voice_Op_Edit[OPT][OP8] = 'o4' Voice_Op_Edit[OPT][OP9] = 'Mx' # VOICE OPARATOR PARAMETER COLUMN s NAME VoiceOpName = [None] * VOICE_OP_ROW_NUM VoiceOpName[OPT] = 'Operator' VoiceOpName[SR] = 'SR' VoiceOpName[XOF] = 'XOF' VoiceOpName[KSR] = 'KSR' VoiceOpName[RR] = 'RR' VoiceOpName[DR] = 'DR' VoiceOpName[AR] = 'AR' VoiceOpName[SL] = 'SL' VoiceOpName[TL] = 'TL' VoiceOpName[KSL] = 'KSL' VoiceOpName[DAM] = 'DAM' VoiceOpName[EAM] = 'EAM' VoiceOpName[DVB] = 'DVB' VoiceOpName[EVB] = 'EVB' VoiceOpName[MUL] = 'MUL' VoiceOpName[DT] = 'DT' VoiceOpName[WS] = 'WS' VoiceOpName[FB] = 'FB' # VOICE OPTIONAL PARAMETER TABLE ROW s NAME for i in range(VOICE_OP_ROW_NUM): # i : 0 - 18 Voice_Op_Edit[i][OP0] = VoiceOpName[i] # Common BO, LFO, ALG の最大値 Voice_Com_Max = [3, 3, 7] # Common SR, XOF, KSR, RR, DR, AR, SL, TL, KSL, DAM, EAM, DVB, EVB, MUL, DT,WS, FB の最大値 Voice_Op_Max = [15, 1, 1, 15, 15, 15, 15, 63, 3, 3, 1, 3, 1, 15, 7, 31, 7] # Voice_Com_Edit TABLE s Max # VOICE COMMON PARAMETER TABLE s Max for i in range(1, VOICE_COM_ROW_NUM): # i : 1 - 3 Voice_Com_Edit[i][CM3] = Voice_Com_Max[i -1] # Voice_Op_Edit TABLE s Max # VOICE OPTIONAL PARAMETER TABLE s Max for i in range(1, VOICE_OP_ROW_NUM): # i : 1 - 17 Voice_Op_Edit[i][OP9] = Voice_Op_Max[i -1] # MIDI GM to YMF825 Tone Change # テーブル(辞書) exGmYmf825 = {00:00, # ピアノ系 01:06, # 02:Bright Acostic Piano:06:Bright Piano 02:01, # 03:Electric Grand Piano:01:E.Piano 03:01, # 04:Honky-Tonk Piano:01:E.Piano 04:01, # 05:Electric Piano 1:01:E.Piano 05:01, # 06:Electric Piano 2:01:E.Piano 06:00, 07:00, # クロマチック・パーカッション系:04:TiklBell 08:04, 09:04, 10:04, 11:07, # 12:Vibraphone:ヴィヴラフォン:大型鉄琴:07:Vibes 12:04, 13:04, 14:04, 15:04, 16:04, 17:04, # オルガン系 18:10, 19:08, # 20:Church Organ:パイプ・オルガン:08:Church Organ 20:10, 21:15, # 22:Accordion:アコーディオン:15:Harmonica 22:10, 23:10, # ギター系 24:11, # 25:Acostic Guitar (nylon):11:NulonGuiter 25:11, 26:11, 27:11, 28:11, 29:11, 30:11, 31:11, # ベース系 32:03, # 33:Acosic Bass 通常のアコースティック:03:PickBass 33:03, # 34:Electric Bass (finger):03:PickBass 34:03, # 35:Electric Bass (pick):03:PickBass 35:03, 36:03, 37:03, 38:03, 39:03, # ストリングス系 40:12, # 41:Violin ヴァイオリン:12:SpuareLead 41:14, 42:14, 43:14, 44:14, 45:14, 46:14, # 47:Orchestral Harp:ハープ:14:Harrpsichord 47:14, # アンサンブル系 48:10, # 49:String Ensamble-1:ストリングス・アンサンブル:10:RockOrgan 49:10, # 50:String Ensamble 2:ストリンクス・アンサンブル:10:RockOrgan 50:05, # 51:Synth Strings 1:05:NewAgePd 51:05, # 52:Synth Strings 2:05:NewAgePd 52:05, 53:05, 54:05, 55:05, # ブラス系 56:00, 57:00, 58:00, 59:00, 60:00, 61:00, 62:00, 63:00, # リード系 64:02, 65:02, 66:02, 67:02, 68:02, 69:02, 70:02, 71:02, # 72:Clarinet:クラリネット:02:TenorSax # パイプ系 72:09, 73:09, # 74:Flute:フルート:09:Flute 74:13, # 75:Recorder:リコーダー:13:SawLead 75:13, 76:13, 77:13, 78:13, 79:13, # シンセ・リード系 80:00, 81:00, 82:00, 83:00, 84:00, 85:00, 86:00, 87:00, # シンセ・パッド系 88:00, 89:00, 90:00, 91:00, 92:00, 93:00, 94:00, 95:00, 96:00, # シンセ・エフェクト系 97:00, 98:00, 99:00,100:00,101:00,102:00,103:00, # エスニック系 104:00,105:00,106:00,107:00,108:00,109:00,110:00,111:00, # パーカッシヴ系 112:00,113:00,114:00,115:00,116:00,117:00,118:00,119:00, # 効果音 120:00,121:00,122:00,123:00,124:00,125:00,126:00,127:00,} GmName = [None] * (128 +1) GmName[0] = 'MIDI GM Tone Name' # ピアノ系 GmName[1] = 'Acostic Grand Piano' GmName[2] = 'Bright Acostic Piano' GmName[3] = 'Electric Grand Piano' GmName[4] = 'Honky-Tonk Piano' GmName[5] = 'Electric Piano 1' GmName[6] = 'Electric Piano 2' GmName[7] = 'Harpsicord' GmName[8] = 'Clavi' # クロマチック・パーカッション系 GmName[9] = 'Celesta' GmName[10] = 'Glockenspiel' GmName[11] = 'Music Box' GmName[12] = 'Vibraphone' GmName[13] = 'Marimba' GmName[14] = 'Xylophone' GmName[15] = 'Tubular Bells' GmName[16] = 'Dulcimer' # オルガン系 GmName[17] = 'Drawber Organ' GmName[18] = 'Percussive Organ' GmName[19] = 'Rock Organ' GmName[20] = 'Church Organ' GmName[21] = 'Reed Organ' GmName[22] = 'Accordion' GmName[23] = 'Harmonica' GmName[24] = 'Tango Accordion' # ギター系 GmName[25] = 'Acostic Guitar (nylon)' GmName[26] = 'Acostic Guitar (steel)' GmName[27] = 'Electric Guitar (jazz)' GmName[28] = 'Electric Guitar (clean)' GmName[29] = 'Electric Guitar (muted)' GmName[30] = 'Overdriven Guitar' GmName[31] = 'Distortion Guitar' GmName[32] = 'Guitar Harmonics' # ベース系 GmName[33] = 'Acosic Bass' GmName[34] = 'Electric Bass (finger)' GmName[35] = 'Electric Bass (pick)' GmName[36] = 'Fretless Bass' GmName[37] = 'Slap Bass 1' GmName[38] = 'Slap Bass 2' GmName[39] = 'Synth Bass 1' GmName[40] = 'Synth Bass 2' # ストリングス系 GmName[41] = 'Violin' GmName[42] = 'Viola' GmName[43] = 'Cello' GmName[44] = 'Contrabass' GmName[45] = 'Tremoro Strings' GmName[46] = 'Pizzicato Strings' GmName[47] = 'Orchestral Harp' GmName[48] = 'Timpani' # アンサンブル系 GmName[49] = 'String Ensamble 1' GmName[50] = 'String Ensamble 2' GmName[51] = 'Synth Strings 1' GmName[52] = 'Synth Strings 2' GmName[53] = 'Choir Aahs' GmName[54] = 'Voice Oohs' GmName[55] = 'Synth Voice' GmName[56] = 'Orchestra Hit' # ブラス系 GmName[57] = 'Trumpet' GmName[58] = 'Trombone' GmName[59] = 'Tuba' GmName[60] = 'Muted Trumpet' GmName[61] = 'French Horn' GmName[62] = 'Brass Section' GmName[63] = 'Synth Brass 1' GmName[64] = 'Synth Brass 2' # リード系 GmName[65] = 'Soprano Sax' GmName[66] = 'Alto Sax' GmName[67] = 'Tenor Sax' GmName[68] = 'Baritone Sax' GmName[69] = 'Oboe' GmName[70] = 'English Horn' GmName[71] = 'Bassoon' GmName[72] = 'Clarinet' # パイプ系 GmName[73] = 'Piccolo' GmName[74] = 'Flute' GmName[75] = 'Recorder' GmName[76] = 'Pan Flute' GmName[77] = 'Bottle Blow' GmName[78] = 'Shakuhachi' GmName[79] = 'Whistle' GmName[80] = 'Ocarina' # シンセ・リード系 GmName[81] = 'Lead 1 (square)' GmName[82] = 'Lead 2 (sawtooth)' GmName[83] = 'Lead 3 (caliope)' GmName[84] = 'Lead 4 (chiff)' GmName[85] = 'Lead 5 (charang)' GmName[86] = 'Lead 6 (voice)' GmName[87] = 'Lead 7 (fifth)' GmName[88] = 'Lead 8 (bass+lead)' # シンセ・パッド系 GmName[89] = 'Pad 1 (new age)' GmName[90] = 'Pad 2 (warm)' GmName[91] = 'Pad 3 (polysynth)' GmName[92] = 'Pad 4 (choir)' GmName[93] = 'Pad 5 (bowed)' GmName[94] = 'Pad 6 (metalic)' GmName[95] = 'Pad 7 (halo)' GmName[96] = 'Pad 8 (sweep)' # シンセ・エフェクト系 GmName[97] = 'FX (rain)' GmName[98] = 'FX (soundtrack)' GmName[99] = 'FX (crystal)' GmName[100] = 'FX (atmosphere)' GmName[101] = 'FX (brightness)' GmName[102] = 'FX (goblins)' GmName[103] = 'FX (echoes)' GmName[104] = 'FX (sci-fi)' # エスニック系 GmName[105] = 'Sitar' GmName[106] = 'Banjo' GmName[107] = 'Shamisen' GmName[108] = 'Koto' GmName[109] = 'Kalimba' GmName[110] = 'Bagpipe' GmName[111] = 'Fiddle' GmName[112] = 'Shanai' # パーカッシヴ系 GmName[113] = 'Tinkle Bell' GmName[114] = 'Agogo' GmName[115] = 'Steel Drums' GmName[116] = 'Woodblock' GmName[117] = 'Taiko Drum' GmName[118] = 'Melodic Tom' GmName[119] = 'Synth Drum' GmName[120] = 'Reverse Cymbal' # 効果音 GmName[121] = 'Guitar Fret Noise' GmName[122] = 'Breath Noise' GmName[123] = 'Seashore' GmName[124] = 'Bird Tweet' GmName[125] = 'Telephone Ring' GmName[126] = 'Helicopter' GmName[127] = 'Applause' GmName[128] = 'Gun Shot' # YMF825 # key name KEY_C = const(0) KEY_C_SHARP = const(1) KEY_D = const(2) KEY_D_SHARP = const(3) KEY_E = const(4) KEY_F = const(5) KEY_F_SHARP = const(6) KEY_G = const(7) KEY_G_SHARP = const(8) KEY_A = const(9) KEY_A_SHARP = const(10) KEY_B = const(11) # I/O voltage IOVDD_5V = const(0) IOVDD_3V3 = const(1) # number of channels CH_MAX = const(16) # FNUM table # C C# D D# E F F# G G# A A# B FNUM = [357, 378, 401, 425, 450, 477, 505, 535, 567, 601, 637, 674] FNUM_MAX = 12 # FNUM table (high byte) calcurated in SimpleYMF825::begin() uint8_t FNUMH[12]; FNUMH = [None] * FNUM_MAX # FNUM table (low byte) calcurated in SimpleYMF825::begin() uint8_t FNUML[12]; # uint8_t FNUML[12]; FNUML = [None] * FNUM_MAX # init_fnum() で FNUMH FNUML を定義する for i in range(FNUM_MAX): FNUMH[i] = (FNUM[i] & 0x0380) >> 4 FNUML[i] = (FNUM[i] & 0x007F) # PicoMain では、使用しない ・・・ ここから # SPI SPI_CH = const(0) SPI_MAX_SPEED_HZ = const(1000000) #GPIO assignment # Pin No. SPIPORT_MOSI = const(19) # pin25 SPIPORT_MISO = const(16) # pin21 SPIPORT_CLK = const(18) # pin24 SPIPORT_CE = const(17) # pin22 YMF825_RESET = const(22) # pin29 GPIO_LED = const(28) # pin34 # N/A # SPI0 YMF825 spi_cs = Pin(SPIPORT_CE, Pin.OUT) spi0 = SPI(SPI_CH, sck=Pin(SPIPORT_CLK), mosi=Pin(SPIPORT_MOSI), miso=Pin(SPIPORT_MISO), baudrate=SPI_MAX_SPEED_HZ, firstbit=SPI.MSB, polarity=0, phase=0) print('spi0 YMF825 in YMF825py = ', spi0) print() # YMF825 RESET pin YMF825_reset = Pin(YMF825_RESET, Pin.OUT) # LED (CURRENTLY NOT AVAILABLE) led_indicator = Pin(GPIO_LED, Pin.OUT) # PicoMain では、使用しない ・・・ ここまで # 20240501 Add # UASRT 設定 UART_BAUD = 115200 # 20240517 Change # uart = UART(1, UART_BAUD, bits=8, parity=None, stop=1, tx=Pin(4), rx=Pin(5)) # # uart = UART(1, UART_BAUD, bits=8, parity=None, stop=1, tx=Pin(4), rx=Pin(5), \ # timeout=500, timeout_char=500) # # 20240810 Change uart = UART(1, UART_BAUD, bits=8, parity=None, stop=1, tx=Pin(4), rx=Pin(5), \ timeout=50, timeout_char=50) #LED indicator # onoff:: True:turn on, False: turn off def led_turn(onoff): led_indicator.value(1 if onoff else 0) #Wait timer # msec:: Waite time in milli-seconds def delay(msec): time.sleep(msec/1000) #Set SPI Slave Select Pin (CE0) # PicoMain では、使用しない。 # pinv:: GPIO.HIGH: not-select, GPIO.LOW: select def chip_select(sel): spi_cs.value(0 if sel else 1) ################################################################# # 20240316 Add & Change in use # PicoMain では、使用しない。 # SPI write (multi bytes) def spi_write_multi(addr, multi_data): data_array = bytearray([addr]) data_array += multi_data chip_select(True) spi0.write(data_array) chip_select(False) # 20240316 Add in use # PicoMain では、使用しない。 # SPI write (single byte) def spi_write_single(addr, single_data): data_array = bytearray([addr, single_data]) chip_select(True) spi0.write(data_array) chip_select(False) ################################################################# #Set Chanel # PicoMain では、使用しない。 def set_chanel(): spi_write_single(0x0F,0x30) spi_write_single(0x10,0x71) spi_write_single(0x11,0x00) spi_write_single(0x12,0x08) spi_write_single(0x13,0x00) # 20240314 ADD in use # Set Mater Volume # vol: volume (0...63) # 1...63; -50dB... +12dB def set_MasterVolume(vol): spi_write_single(25,vol << 2) # 20240315 Add in use # PicoMain では、使用しない。 # ch:0-15 octave: vol: def keyon4(ch, octave, key, vol): global FNUMH global FNUML # global sund_param spi_write_single(11,ch & 0x0F) spi_write_single(12,vol & 0x7F) spi_write_single(13,FNUMH[key] | (octave - 1)) spi_write_single(14,FNUML[key]) spi_write_single(15,0x40 | 0) # Last 0:tone led_turn(True) # 20240315 Add in use # PicoMain では、使用しない。 # ch:0-15 octave: vol: def keyon5(ch, octave, key, vol, tone): global FNUMH global FNUML # global sund_param spi_write_single(11,ch & 0x0F) spi_write_single(12,vol & 0x7F) spi_write_single(13,FNUMH[key] | (octave - 1)) spi_write_single(14,FNUML[key]) spi_write_single(15,0x40 | tone & 0x0F) led_turn(True) # 20240315 Add in use # PicoMain では、使用しない。 def keyoff(ch): spi_write_single(11, ch & 0x0F) spi_write_single(0x0F, 0) led_turn(False) # 20240622 Add in use # PicoMain では、使用しない。 # tone は、ToneTableを参照し、16以上ならば何もしない def note_on(ch, octave, key, vol): global FNUMH global FNUML global Tonetable # ToneTable の値が 16未満なら if ToneTable[(ch & 0x0F) +1][1] <16: spi_write_single(11,ch & 0x0F) spi_write_single(12,vol & 0x7F) spi_write_single(13,FNUMH[key] | (octave - 1)) spi_write_single(14,FNUML[key]) spi_write_single(15,0x40 | ToneTable[(ch & 0x0F) +1][1]) led_turn(True) # 20240705 Add in use # PicoMain では、使用しない。 # tone は、1つを指定する。 def note_onMonoTone(ch, octave, key, vol, tone): global FNUMH global FNUML global Tonetable # ToneTable の値が 16未満なら if ToneTable[(ch & 0x0F) +1][1] <16: spi_write_single(11,ch & 0x0F) spi_write_single(12,vol & 0x7F) spi_write_single(13,FNUMH[key] | (octave - 1)) spi_write_single(14,FNUML[key]) spi_write_single(15,0x40 | tone & 0x0F) led_turn(True) # 20240622 Add in use # PicoMain では、使用しない。 def note_off(ch): spi_write_single(11, ch & 0x0F) spi_write_single(0x0F, 0) led_turn(False) ################################################################# # 20240501 Add def uart_write_multi(multi_data): uart.write(multi_data) # 20240618 Add def uart_read_multi(): global rcv_buff rcv_size = uart.readinto(rcv_buff) return rcv_size # 20240701 Add def uart_blank_read(): while uart.any() > 0: try: data_rx = uart.read(1) except UnicodeError: print('uart_blank_read() UnicodeError') except: print('uart_blank_read() Error') ################################################################# # 20240518 Add def uartOutToneData(): global send_buff print("Write tone data out to UART") uart_write_multi(send_buff) # 20240619 Add def uartOutToneTable(): global uartT_buff print("Write Tone Table out to UART") uart_write_multi(uartT_buff) # 20240518 Add # 20240619 Change ==> headerID が、0:ToneData、1:ToneTable、両方に対応 def uartInTone(): global rcv_buff # ASCII 2文字リスト global rcv_tone # 整数リスト print("Read tone data in from UART") rcv_size = uart_read_multi() if rcv_size == None: # 受信データなし print() print('rcv_size = None') # rcv_size, headerID, tone_name return None, None, None else: # 受信データあり # DEBUG print print('rcv_size = ', rcv_size) # print(rcv_buff) headerID = chr(rcv_buff[0]) if headerID == '0': tone_name = rcv_buff[1:TONE_HEADER].decode('utf-8') # tone_name print('Recive Data = ', tone_name) # 2文字の ASCIIコード(0-9 A-F)を1byteの整数に変換する。 rcv_tone = [] for i in range(UART_TONE_NUM): # 42 rcv_tone.append(moji.asc2int(chr(rcv_buff[TONE_HEADER + i *2]) \ + chr(rcv_buff[TONE_HEADER + i *2 +1]))) # DEBUG print print('rcv_tone = ', rcv_tone) return rcv_size, headerID, tone_name elif headerID == '1': tone_name = rcv_buff[1:TONE_HEADER].decode('utf-8') # 'Tone Table' print('Recive Data = ', tone_name) # 2文字の ASCIIコード(0-9 A-F)を1byteの整数に変換する。 rcv_tone = [] for i in range(TONE_TABLE_ROW_NUM -1): # 0-15 : 16 rcv_tone.append(moji.asc2int(chr(rcv_buff[TONE_HEADER + i *2]) \ + chr(rcv_buff[TONE_HEADER + i *2 +1]))) # DEBUG print print('ToneTable = ', rcv_tone) return rcv_size, headerID, tone_name elif headerID == '%': # PrintDFP command print('rcv_buff = ', rcv_buff) PrintDFPcommand = rcv_buff[:rcv_size] # DEBUG print print('PrintDFPcommand = ', PrintDFPcommand) # return rcv_size, headerID, tone_name return rcv_size, headerID, PrintDFPcommand ################################################################# # 20240518 Add # tone_num で指定された Tone\Com_edit および Tone_Op_Edit の情報を # UART送信で使用する ASCIIコードの2文字に変換する。 def takeOutToneData(num): tone_num = num & 0x0F # 念の為 0-15 に限定 global send_tone global send_buff global Tone_Com_Edit global Tone_Op_Edit send_buff[0] = ord('0') # [0]:ZERO:ToneID # ToneName send_buff[1:TONE_HEADER] = bytes(moji.strFixL_n(ToneName[tone_num], TONE_HEADER -1), 'utf8') send_tone = [] # 整数 send_tone.append(tone_num) send_tone.append(Tone_Com_Edit[tone_num]) for i in range(FM_OP_NUM): # 4 for j in range(OP_PARAM_NUM): # 10 send_tone.append(Tone_Op_Edit[tone_num][i][j]) # DEBUG print # print('send ToneData = ', send_tone) # 1byte の整数を ASCIIコード(0-9 A-F)の2文字に変換する。 for i in range(UART_TONE_NUM): # 42 (asc_H, asc_L) = moji.int2asc2(send_tone[i]) send_buff[TONE_HEADER + i *2] = ord(asc_H) send_buff[TONE_HEADER + i *2 +1] = ord(asc_L) ################################################################# # 20240619 Add # Tone Table の情報を UART送信で使用する ASCIIコードの2文字に変換する。 def takeOutToneTable(): global ToneTable global send_tone global send_buff global uartT_buff send_buff[0] = ord('1') # [1]:ONE:Tone Table # Tone Table send_buff[1:TONE_HEADER] = bytes(moji.strFixL_n('Tone Table', TONE_HEADER -1), 'utf8') # ToneTable の中身だけ抽出 ・・・ ch 0-15 に対応する ToneNo 0-16 16:NoTone send_tone = [] # 整数リスト for i in range(1, TONE_TABLE_ROW_NUM): # 1-17 : 16 send_tone.append(ToneTable[i][1]) # DEBUG print # print('send ToneTable = ', send_tone) # 1byte の整数を ASCIIコード(0-9 A-F)の2文字に変換する。 for i in range(TONE_TABLE_ROW_NUM -1): # 0-15 : 16 (asc_H, asc_L) = moji.int2asc2(send_tone[i]) send_buff[TONE_HEADER + i *2] = ord(asc_H) send_buff[TONE_HEADER + i *2 +1] = ord(asc_L) # TONE_HEADER + (TONE_TABLE_ROW_NUM) *2) = 48 bytes uartT_buff = send_buff[:(TONE_HEADER + (TONE_TABLE_ROW_NUM -1) *2)] ################################################################# # 20240518 Add # 受信した rcv_tone のデータを tone_num で指定された番号の # Tone_Com_edit および Tone_Op_Edit に格納する。 def takeInToneData(): global rcv_tone # 整数リスト global Tone_Com_Edit global Tone_Op_Edit if len(rcv_tone) > 0: tone_num = rcv_tone[0] & 0x0F # 念の為、0-15 に限定 Tone_Com_Edit[tone_num] = rcv_tone[1] # rcv_tone は、2文字の ASCIIコード(0-9 A-F)から1byteの整数に変換したリスト # rcv_tone データを Tone_Op_Edit に格納 for i in range(FM_OP_NUM): # 4 for j in range(OP_PARAM_NUM): # 10 Tone_Op_Edit[tone_num][i][j] = rcv_tone[2 + i *OP_PARAM_NUM + j] else: tone_num = None return tone_num ################################################################# # 20240620 Add # 受信した rcv_tone のデータを ToneTable に格納する。 def takeInToneTable(): global rcv_tone # 整数リスト global ToneTable if len(rcv_tone) > 0: if len(rcv_tone) == (TONE_TABLE_ROW_NUM -1): # 16 # rcv_tone は、2文字の ASCIIコード(0-9 A-F)から1byteの整数に変換したリスト # rcv_tone データを ToneTable に格納 for i in range(1, TONE_TABLE_ROW_NUM): # 1-17 : 16 ToneTable[i][1] = rcv_tone[i -1] # rcv_tone[i] 0-16 print('ToneTable =', ToneTable) else: print('rcv_tone size is not 16') print('ToneTable size is 16') else: print('rcv_tone is None') ################################################################# # 20240318 Add in use # YMF825 に tone_dataを設定する # PicoMain では、使用しない。 def init_tone(): global tone_data global Tone_Com_Edit global Tone_Op_Edit global ToneName # tone parameter data size COMMON_PARAM_SIZE = 2 OP_PARAM_SIZE = 7 TONE_PARAM_SIZE = COMMON_PARAM_SIZE + FM_OP_NUM * OP_PARAM_SIZE; MESSAGE_SIZE = TONE_PARAM_SIZE * TONE_NUM + 1 + 4 # 485 # print() # print("YMF825 Tone Parameter Setting Data") # header tone_data[0] = 0x80 + TONE_NUM # print(f"Header = {tone_data[0]:4}") # register data for i in range(TONE_NUM): # print() # print(f'Voice Number = {i} : {ToneName[i]}') # common parameter 1 tone_data[i* TONE_PARAM_SIZE +1] = (Tone_Com_Edit[i] & 0x60) >> 5 # print(f"common parameter 1 = {tone_data[i* TONE_PARAM_SIZE +1]:4}") # common parameter 2 tone_data[i* TONE_PARAM_SIZE +2] = ((Tone_Com_Edit[i] & 0x18) << 3) | (Tone_Com_Edit[i] & 0x07) # print(f"common parameter 2 = {tone_data[i* TONE_PARAM_SIZE +2]:4}") # operator parameter 7 x 4 for j in range(FM_OP_NUM): # print(f"operator parameter [{j}] = ", end = "") tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +3] = \ (Tone_Op_Edit[i][j][3] << 4) | \ (Tone_Op_Edit[i][j][0] & 0x08) | \ ((Tone_Op_Edit[i][j][0] & 0x04) >> 2) # print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +3]:4},", end = "") tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +4] = \ (Tone_Op_Edit[i][j][4] << 4) | Tone_Op_Edit[i][j][2] # print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +4]:4},", end = "") tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +5] = \ (Tone_Op_Edit[i][j][1] << 4) | Tone_Op_Edit[i][j][5] # print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +5]:4},", end = "") tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +6] = \ (Tone_Op_Edit[i][j][6] << 2) | (Tone_Op_Edit[i][j][0] & 0x03) # print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +6]:4},", end = "") tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +7] = \ Tone_Op_Edit[i][j][7] # print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +7]:4},", end = "") tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +8] = \ ((Tone_Op_Edit[i][j][8] & 0x0f) << 4) | \ ((Tone_Op_Edit[i][j][8] & 0xf0) >> 4) # print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +8]:4},", end = "") tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +9] = \ (Tone_Op_Edit[i][j][9] << 3) | \ ((Tone_Op_Edit[i][j][0] & 0x70) >> 4) # print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +9]:4}") # footer tone_data[TONE_PARAM_SIZE * TONE_NUM +0] = 0x80 # print(f"footer 1 = {tone_data[TONE_PARAM_SIZE * TONE_NUM +0]}") tone_data[TONE_PARAM_SIZE * TONE_NUM +1] = 0x03 # print(f"footer 2 = {tone_data[TONE_PARAM_SIZE * TONE_NUM +1]}") tone_data[TONE_PARAM_SIZE * TONE_NUM +2] = 0x81 # print(f"footer 3 = {tone_data[TONE_PARAM_SIZE * TONE_NUM +2]}") tone_data[TONE_PARAM_SIZE * TONE_NUM +3] = 0x80 # print(f"footer 4 = {tone_data[TONE_PARAM_SIZE * TONE_NUM +3]}") #Burst write mode # print("YMF85 Burst write mode.") spi_write_single(0x08, 0xF6) # delay(20) delay(10) # Before 1000 spi_write_single(0x08, 0x00) #Write tone data to YMF825 FIFO print("Write tone data to YMF825.") spi_write_multi(0x07, tone_data) def copyTone2_Voice(num): tone_num = num & 0x0F global Tone_Com_Orig global Tone_Op_Orig global Tone_Com_Edit global Tone_Op_Edit global Voice_Com_Edit global Voice_Op_Edit # common parameter Voice_Com_Edit[BO][CM2] = (Tone_Com_Orig[tone_num] & 0x60) >> 5 Voice_Com_Edit[LFO][CM2] = (Tone_Com_Orig[tone_num] & 0x18) >> 3 Voice_Com_Edit[ALG][CM2] = Tone_Com_Orig[tone_num] & 0x07; Voice_Com_Edit[BO][CM1] = (Tone_Com_Edit[tone_num] & 0x60) >> 5 Voice_Com_Edit[LFO][CM1] = (Tone_Com_Edit[tone_num] & 0x18) >> 3 Voice_Com_Edit[ALG][CM1] = Tone_Com_Edit[tone_num] & 0x07 # operator s parameter for j in range(FM_OP_NUM): # j : 0 - 3 Voice_Op_Edit[SR][j +5] = (Tone_Op_Orig[tone_num][j][3] & 0x0F) Voice_Op_Edit[SR][j +1] = (Tone_Op_Edit[tone_num][j][3] & 0x0F) Voice_Op_Edit[XOF][j +5] = ((Tone_Op_Orig[tone_num][j][0] & 0x08) >> 3) Voice_Op_Edit[XOF][j +1] = ((Tone_Op_Edit[tone_num][j][0] & 0x08) >> 3) Voice_Op_Edit[KSR][j +5] = ((Tone_Op_Orig[tone_num][j][0] & 0x04) >> 2) Voice_Op_Edit[KSR][j +1] = ((Tone_Op_Edit[tone_num][j][0] & 0x04) >> 2) Voice_Op_Edit[RR][j +5] = (Tone_Op_Orig[tone_num][j][4] & 0x0F) Voice_Op_Edit[RR][j +1] = (Tone_Op_Edit[tone_num][j][4] & 0x0F) Voice_Op_Edit[DR][j +5] = (Tone_Op_Orig[tone_num][j][2] & 0x0f) Voice_Op_Edit[DR][j +1] = (Tone_Op_Edit[tone_num][j][2] & 0x0f) Voice_Op_Edit[AR][j +5] = (Tone_Op_Orig[tone_num][j][1] & 0x0F) Voice_Op_Edit[AR][j +1] = (Tone_Op_Edit[tone_num][j][1] & 0x0F) Voice_Op_Edit[SL][j +5] = (Tone_Op_Orig[tone_num][j][5] & 0x0F) Voice_Op_Edit[SL][j +1] = (Tone_Op_Edit[tone_num][j][5] & 0x0F) Voice_Op_Edit[TL][j +5] = (Tone_Op_Orig[tone_num][j][6] & 0x3F) Voice_Op_Edit[TL][j +1] = (Tone_Op_Edit[tone_num][j][6] & 0x3F) Voice_Op_Edit[KSL][j +5] = (Tone_Op_Orig[tone_num][j][0] & 0x03) Voice_Op_Edit[KSL][j +1] = (Tone_Op_Edit[tone_num][j][0] & 0x03) Voice_Op_Edit[DAM][j +5] = ((Tone_Op_Orig[tone_num][j][7] & 0x60) >> 5) Voice_Op_Edit[DAM][j +1] = ((Tone_Op_Edit[tone_num][j][7] & 0x60) >> 5) Voice_Op_Edit[EAM][j +5] = ((Tone_Op_Orig[tone_num][j][7] & 0x10) >> 4) Voice_Op_Edit[EAM][j +1] = ((Tone_Op_Edit[tone_num][j][7] & 0x10) >> 4) Voice_Op_Edit[DVB][j +5] = ((Tone_Op_Orig[tone_num][j][7] & 0x06) >> 1) Voice_Op_Edit[DVB][j +1] = ((Tone_Op_Edit[tone_num][j][7] & 0x06) >> 1) Voice_Op_Edit[EVB][j +5] = (Tone_Op_Orig[tone_num][j][7] & 0x01) Voice_Op_Edit[EVB][j +1] = (Tone_Op_Edit[tone_num][j][7] & 0x01) Voice_Op_Edit[MUL][j +5] = (Tone_Op_Orig[tone_num][j][8] & 0x0f) Voice_Op_Edit[MUL][j +1] = (Tone_Op_Edit[tone_num][j][8] & 0x0f) Voice_Op_Edit[DT][j +5] = ((Tone_Op_Orig[tone_num][j][8] & 0xf0) >> 4) Voice_Op_Edit[DT][j +1] = ((Tone_Op_Edit[tone_num][j][8] & 0xf0) >> 4) Voice_Op_Edit[WS][j +5] = (Tone_Op_Orig[tone_num][j][9] & 0x1F) Voice_Op_Edit[WS][j +1] = (Tone_Op_Edit[tone_num][j][9] & 0x1F) Voice_Op_Edit[FB][j +5] = ((Tone_Op_Orig[tone_num][j][0] & 0x70) >> 4) Voice_Op_Edit[FB][j +1] = ((Tone_Op_Edit[tone_num][j][0] & 0x70) >> 4) def copyVoice2_Tone(num): tone_num = num & 0x0F global Tone_Com_Edit global Tone_Op_Edit global Voice_Com_Edit global Voice_Op_Edit Tone_Com_Edit[tone_num] = ((Voice_Com_Edit[BO][CM1] << 5) | \ (Voice_Com_Edit[LFO][CM1] << 3) | \ Voice_Com_Edit[ALG][CM1]) # operator s parameter for j in range(FM_OP_NUM): # j : 0 - 3 Tone_Op_Edit[tone_num][j][0] = ((Voice_Op_Edit[FB][j +1] << 4) | \ (Voice_Op_Edit[XOF][j +1] << 3) | \ (Voice_Op_Edit[KSR][j +1] << 2) | \ (Voice_Op_Edit[KSL][j +1])) Tone_Op_Edit[tone_num][j][1] = Voice_Op_Edit[AR][j +1] Tone_Op_Edit[tone_num][j][2] = Voice_Op_Edit[DR][j +1] Tone_Op_Edit[tone_num][j][3] = Voice_Op_Edit[SR][j +1] Tone_Op_Edit[tone_num][j][4] = Voice_Op_Edit[RR][j +1] Tone_Op_Edit[tone_num][j][5] = Voice_Op_Edit[SL][j +1] Tone_Op_Edit[tone_num][j][6] = Voice_Op_Edit[TL][j +1] Tone_Op_Edit[tone_num][j][7] = ((Voice_Op_Edit[DAM][j +1] << 5) | \ (Voice_Op_Edit[EAM][j +1] << 4) | \ (Voice_Op_Edit[DVB][j +1] << 1) | \ (Voice_Op_Edit[EVB][j +1])) Tone_Op_Edit[tone_num][j][8] = ((Voice_Op_Edit[DT][j +1] << 4) | \ (Voice_Op_Edit[MUL][j +1])) Tone_Op_Edit[tone_num][j][9] = Voice_Op_Edit[WS][j +1] def serialOutVoiceData(num): tone_num = num & 0x0F global Voice_Com_Edit global Voice_Op_Edit print() print(f'Voice Number = {tone_num} : {ToneName[tone_num]}') # Voice Edit Common Parameter print(f'{Voice_Com_Edit[COM][CM0]:10}:', end = '') # Header for j in range(1, VOICE_COM_COL_NUM): print(f'{Voice_Com_Edit[COM][j]:> 5}|', end = '') print() # Body for i in range(1, VOICE_COM_ROW_NUM): # print(f'|{moji.strFixL_n(Voice_Com_Edit[i][CM0], 5)} ', ' :', end = '') print(f'{Voice_Com_Edit[i][CM0]:10}:', end = '') print(f'{Voice_Com_Edit[i][CM1]: 5}|', end = '') print(f'{Voice_Com_Edit[i][CM2]: 5}|', end = '') print(f'{Voice_Com_Edit[i][CM3]: 5}|') print() # Vice Edit Oparetor Parameter print(f'{Voice_Op_Edit[OPT][OP0]:10}:', end = '') # Header for j in range(1, VOICE_OP_COL_NUM): print(f'{Voice_Op_Edit[OPT][j]:> 3}|', end = '') print() # Body for i in range(1, VOICE_OP_ROW_NUM): # print(f'|{moji.strFixL_n(Voice_Com_Edit[i][CM0], 5)} ', ' :', end = '') print(f'{Voice_Op_Edit[i][OP0]:10}:', end = '') print(f'{Voice_Op_Edit[i][OP1]: 3}|', end = '') print(f'{Voice_Op_Edit[i][OP2]: 3}|', end = '') print(f'{Voice_Op_Edit[i][OP3]: 3}|', end = '') print(f'{Voice_Op_Edit[i][OP4]: 3}|', end = '') print(f'{Voice_Op_Edit[i][OP5]: 3}|', end = '') print(f'{Voice_Op_Edit[i][OP6]: 3}|', end = '') print(f'{Voice_Op_Edit[i][OP7]: 3}|', end = '') print(f'{Voice_Op_Edit[i][OP8]: 3}|', end = '') print(f'{Voice_Op_Edit[i][OP9]: 3}|') print() def serialOutToneData(num): tone_num = num & 0x0F global tone_data global Tone_Com_Edit global Tone_Op_Edit global ToneName # tone parameter data size COMMON_PARAM_SIZE = 2 OP_PARAM_SIZE = 7 TONE_PARAM_SIZE = COMMON_PARAM_SIZE + FM_OP_NUM * OP_PARAM_SIZE; MESSAGE_SIZE = TONE_PARAM_SIZE * TONE_NUM + 1 + 4 # 485 print() print("YMF825 Tone Parameter Setting Data") print(f'Voice Number = {tone_num} : {ToneName[tone_num]}') tone_data[tone_num * TONE_PARAM_SIZE +1] = (Tone_Com_Edit[i] & 0x60) >> 5 print(f"common parameter 1 = {tone_data[tone_num * TONE_PARAM_SIZE +1]:4}") # common parameter 2 tone_data[tone_num * TONE_PARAM_SIZE +2] = ((Tone_Com_Edit[i] & 0x18) << 3) | (Tone_Com_Edit[i] & 0x07) print(f"common parameter 2 = {tone_data[tone_num * TONE_PARAM_SIZE +2]:4}") # operator parameter 7 x 4 for j in range(FM_OP_NUM): print(f"operator parameter [{j}] = ", end = "") tone_data[tone_num * TONE_PARAM_SIZE + j* OP_PARAM_SIZE +3] = \ (Tone_Op_Edit[i][j][3] << 4) | \ (Tone_Op_Edit[i][j][0] & 0x08) | \ ((Tone_Op_Edit[i][j][0] & 0x04) >> 2) print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +3]:4},", end = "") tone_data[tone_num * TONE_PARAM_SIZE + j* OP_PARAM_SIZE +4] = \ (Tone_Op_Edit[i][j][4] << 4) | Tone_Op_Edit[i][j][2] print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +4]:4},", end = "") tone_data[tone_num * TONE_PARAM_SIZE + j* OP_PARAM_SIZE +5] = \ (Tone_Op_Edit[i][j][1] << 4) | Tone_Op_Edit[i][j][5] print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +5]:4},", end = "") tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +6] = \ (Tone_Op_Edit[i][j][6] << 2) | (Tone_Op_Edit[i][j][0] & 0x03) print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +6]:4},", end = "") tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +7] = \ Tone_Op_Edit[i][j][7] print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +7]:4},", end = "") tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +8] = \ ((Tone_Op_Edit[i][j][8] & 0x0f) << 4) | \ ((Tone_Op_Edit[i][j][8] & 0xf0) >> 4) print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +8]:4},", end = "") tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +9] = \ (Tone_Op_Edit[i][j][9] << 3) | \ ((Tone_Op_Edit[i][j][0] & 0x70) >> 4) print(f"{tone_data[i* TONE_PARAM_SIZE + j* OP_PARAM_SIZE +9]:4}") # footer tone_data[TONE_PARAM_SIZE * TONE_NUM +0] = 0x80 print(f"footer 1 = {tone_data[TONE_PARAM_SIZE * TONE_NUM +0]}") tone_data[TONE_PARAM_SIZE * TONE_NUM +1] = 0x03 print(f"footer 2 = {tone_data[TONE_PARAM_SIZE * TONE_NUM +1]}") tone_data[TONE_PARAM_SIZE * TONE_NUM +2] = 0x81 print(f"footer 3 = {tone_data[TONE_PARAM_SIZE * TONE_NUM +2]}") tone_data[TONE_PARAM_SIZE * TONE_NUM +3] = 0x80 print(f"footer 4 = {tone_data[TONE_PARAM_SIZE * TONE_NUM +3]}") #Reset and Initialize YMF825 # PicoMain では、使用しない。 def init_YMF825(): YMF825_reset.low() # print("RESET LOW") # C-out delay(100) # Before 1000 YMF825_reset.high() delay(100) # Before 1000 # print("Reset YMF825.") # C-out spi_write_single(0x1D,0x00) spi_write_single(0x02,0x0E) delay(20) spi_write_single(0x00,0x01) spi_write_single(0x01,0x00) spi_write_single(0x1A,0xA3) delay(20) spi_write_single(0x1A,0x00) delay(40) spi_write_single(0x02,0x04) delay(20) spi_write_single(0x02,0x00) # add spi_write_single(0x19,0xFF) spi_write_single(0x1B,0x3F) spi_write_single(0x14,0x00) spi_write_single(0x03,0x01) spi_write_single(0x08,0xF6) delay(40) spi_write_single(0x08,0x00) spi_write_single(0x09,0xF8) spi_write_single(0x0A,0x00) spi_write_single(0x17,0x40) spi_write_single(0x18,0x00) print("YMF825 initialized.") # YMF825 Set up hardware # PicoMain では、使用しない。 def setup(): led_turn(True) chip_select(False) print("Set up YMF825") init_YMF825() set_chanel() print("Finished setting up.") led_turn(False)

「fonts」フォルダを作りその下に次のファイルをコピーしてください

vga1_16x32.py

vga1_16x32.py

WIDTH = 16 HEIGHT = 32 FIRST = 0x20 LAST = 0x7f _FONT = \ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x7f\xfe\x7f\xfe\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x7f\xfe\x7f\xfe\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x01\x80\x01\x80\x0f\xf0\x0f\xf0\x39\x9c\x39\x9c\x71\x8e\x71\x8e\x71\x80\x71\x80\x39\x80\x39\x80\x0f\xf0\x0f\xf0\x01\x9c\x01\x9c\x01\x8e\x01\x8e\x71\x8e\x71\x8e\x39\x9c\x39\x9c\x0f\xf0\x0f\xf0\x01\x80\x01\x80\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x1e\x1c\x1e\x1c\x1e\x38\x1e\x38\x00\x70\x00\x70\x00\xe0\x00\xe0\x01\xc0\x01\xc0\x03\x80\x03\x80\x07\x00\x07\x00\x0e\x3c\x0e\x3c\x1c\x3c\x1c\x3c\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\xc0\x07\xc0\x1c\x70\x1c\x70\x38\x38\x38\x38\x1c\x70\x1c\x70\x07\xc0\x07\xc0\x0f\xce\x0f\xce\x38\xfc\x38\xfc\x70\x78\x70\x78\x70\x78\x70\x78\x38\xfc\x38\xfc\x0f\xce\x0f\xce\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\xe0\x00\xe0\x01\xc0\x01\xc0\x03\x80\x03\x80\x07\x00\x07\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\xe0\x00\xe0\x01\xc0\x01\xc0\x03\x80\x03\x80\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x03\x80\x03\x80\x01\xc0\x01\xc0\x00\xe0\x00\xe0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\x00\x07\x00\x03\x80\x03\x80\x01\xc0\x01\xc0\x00\xe0\x00\xe0\x00\xe0\x00\xe0\x00\xe0\x00\xe0\x00\xe0\x00\xe0\x00\xe0\x00\xe0\x01\xc0\x01\xc0\x03\x80\x03\x80\x07\x00\x07\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x0e\x38\x0e\x38\x03\xe0\x03\xe0\x3f\xfe\x3f\xfe\x03\xe0\x03\xe0\x0e\x38\x0e\x38\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x3f\xfe\x3f\xfe\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x03\xc0\x03\xc0\x03\xc0\x03\xc0\x03\x80\x03\x80\x07\x00\x07\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xfe\x3f\xfe\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x03\xc0\x03\xc0\x03\xc0\x03\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x1c\x00\x1c\x00\x38\x00\x38\x00\x70\x00\x70\x00\xe0\x00\xe0\x01\xc0\x01\xc0\x03\x80\x03\x80\x07\x00\x07\x00\x0e\x00\x0e\x00\x1c\x00\x1c\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\xe0\x07\xe0\x1c\x38\x1c\x38\x38\x3c\x38\x3c\x38\x7c\x38\x7c\x38\xdc\x38\xdc\x39\x9c\x39\x9c\x3b\x1c\x3b\x1c\x3e\x1c\x3e\x1c\x3c\x1c\x3c\x1c\x1c\x38\x1c\x38\x07\xe0\x07\xe0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x01\xc0\x01\xc0\x03\xc0\x03\xc0\x0f\xc0\x0f\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x0f\xf0\x0f\xf0\x38\x1c\x38\x1c\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x1c\x00\x1c\x00\x70\x00\x70\x01\xc0\x01\xc0\x07\x00\x07\x00\x1c\x00\x1c\x00\x38\x00\x38\x00\x3f\xfe\x3f\xfe\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x0f\xf0\x0f\xf0\x38\x1c\x38\x1c\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x1c\x00\x1c\x01\xf0\x01\xf0\x00\x1c\x00\x1c\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x38\x1c\x38\x1c\x0f\xf0\x0f\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x01\xf0\x01\xf0\x03\xf0\x03\xf0\x07\x70\x07\x70\x0e\x70\x0e\x70\x1c\x70\x1c\x70\x38\x70\x38\x70\x3f\xfc\x3f\xfc\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xfe\x3f\xfe\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x3f\xf0\x3f\xf0\x00\x1c\x00\x1c\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x38\x1c\x38\x1c\x0f\xf0\x0f\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\xf0\x07\xf0\x1c\x00\x1c\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x3f\xf0\x3f\xf0\x38\x1c\x38\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x1c\x1c\x1c\x1c\x07\xf0\x07\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xf8\x3f\xf8\x00\x38\x00\x38\x00\x38\x00\x38\x00\x70\x00\x70\x00\xe0\x00\xe0\x01\xc0\x01\xc0\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\xf0\x07\xf0\x1c\x1c\x1c\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x1c\x1c\x1c\x1c\x07\xf0\x07\xf0\x1c\x1c\x1c\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x1c\x1c\x1c\x1c\x07\xf0\x07\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\xf0\x07\xf0\x1c\x1c\x1c\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x1c\x0e\x1c\x0e\x07\xfe\x07\xfe\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x1c\x00\x1c\x0f\xf0\x0f\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x03\x80\x03\x80\x03\x80\x03\x80\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x03\x80\x03\x80\x03\x80\x03\x80\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x03\x80\x03\x80\x03\x80\x03\x80\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x03\x80\x03\x80\x03\x80\x03\x80\x07\x00\x07\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\xe0\x00\xe0\x01\xc0\x01\xc0\x03\x80\x03\x80\x07\x00\x07\x00\x0e\x00\x0e\x00\x1c\x00\x1c\x00\x0e\x00\x0e\x00\x07\x00\x07\x00\x03\x80\x03\x80\x01\xc0\x01\xc0\x00\xe0\x00\xe0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xfc\x3f\xfc\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xfc\x3f\xfc\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\x00\x07\x00\x03\x80\x03\x80\x01\xc0\x01\xc0\x00\xe0\x00\xe0\x00\x70\x00\x70\x00\x38\x00\x38\x00\x70\x00\x70\x00\xe0\x00\xe0\x01\xc0\x01\xc0\x03\x80\x03\x80\x07\x00\x07\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\xe0\x07\xe0\x1c\x38\x1c\x38\x38\x1c\x38\x1c\x00\x38\x00\x38\x00\x70\x00\x70\x00\xe0\x00\xe0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x0f\xf0\x0f\xf0\x38\x1c\x38\x1c\x70\x0e\x70\x0e\x71\xfe\x71\xfe\x73\x8e\x73\x8e\x73\x8e\x73\x8e\x73\x8e\x73\x8e\x71\xfc\x71\xfc\x70\x00\x70\x00\x38\x00\x38\x00\x0f\xfc\x0f\xfc\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x03\xc0\x03\xc0\x07\xe0\x07\xe0\x0e\x70\x0e\x70\x1c\x38\x1c\x38\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x3f\xfc\x3f\xfc\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xf0\x3f\xf0\x38\x1c\x38\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x1c\x38\x1c\x3f\xf0\x3f\xf0\x38\x1c\x38\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x1c\x38\x1c\x3f\xf0\x3f\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\xf0\x07\xf0\x1c\x1c\x1c\x1c\x38\x0e\x38\x0e\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x0e\x38\x0e\x1c\x1c\x1c\x1c\x07\xf0\x07\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xf0\x3f\xf0\x38\x1c\x38\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x1c\x38\x1c\x3f\xf0\x3f\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xfc\x3f\xfc\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x3f\xe0\x3f\xe0\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x3f\xfc\x3f\xfc\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xfc\x3f\xfc\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x3f\xe0\x3f\xe0\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\xf0\x07\xf0\x1c\x1c\x1c\x1c\x38\x0e\x38\x0e\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x3e\x38\x3e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x1c\x1c\x1c\x1c\x07\xf0\x07\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x3f\xfe\x3f\xfe\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x00\x1c\x1c\x1c\x1c\x1c\x0e\x38\x0e\x38\x03\xe0\x03\xe0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x1c\x38\x1c\x38\x1c\x70\x1c\x70\x1c\xe0\x1c\xe0\x1d\xc0\x1d\xc0\x1f\x80\x1f\x80\x1f\x80\x1f\x80\x1d\xc0\x1d\xc0\x1c\xe0\x1c\xe0\x1c\x70\x1c\x70\x1c\x38\x1c\x38\x1c\x1c\x1c\x1c\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x3f\xfc\x3f\xfc\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x78\x1e\x78\x1e\x7c\x3e\x7c\x3e\x7e\x7e\x7e\x7e\x77\xee\x77\xee\x73\xce\x73\xce\x71\x8e\x71\x8e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x38\x0e\x38\x0e\x3c\x0e\x3c\x0e\x3e\x0e\x3e\x0e\x3f\x0e\x3f\x0e\x3b\x8e\x3b\x8e\x39\xce\x39\xce\x38\xee\x38\xee\x38\x7e\x38\x7e\x38\x3e\x38\x3e\x38\x1e\x38\x1e\x38\x0e\x38\x0e\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\xf0\x07\xf0\x1c\x1c\x1c\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x1c\x1c\x1c\x1c\x07\xf0\x07\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xf0\x3f\xf0\x38\x1c\x38\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x1c\x38\x1c\x3f\xf0\x3f\xf0\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\xf0\x07\xf0\x1c\x1c\x1c\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\xee\x38\xee\x1c\x7c\x1c\x7c\x07\xf8\x07\xf8\x00\x1c\x00\x1c\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xf0\x3f\xf0\x38\x1c\x38\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x1c\x38\x1c\x3f\xf0\x3f\xf0\x38\xe0\x38\xe0\x38\x70\x38\x70\x38\x38\x38\x38\x38\x1c\x38\x1c\x38\x0e\x38\x0e\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x0f\xf0\x0f\xf0\x38\x1c\x38\x1c\x70\x0e\x70\x0e\x70\x00\x70\x00\x38\x00\x38\x00\x0f\xf0\x0f\xf0\x00\x1c\x00\x1c\x00\x0e\x00\x0e\x70\x0e\x70\x0e\x38\x1c\x38\x1c\x0f\xf0\x0f\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xfe\x3f\xfe\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x1c\x1c\x1c\x1c\x07\xf0\x07\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x1c\x38\x1c\x38\x0e\x70\x0e\x70\x07\xe0\x07\xe0\x03\xc0\x03\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x70\x0e\x71\x8e\x71\x8e\x73\xce\x73\xce\x77\xee\x77\xee\x3e\x7c\x3e\x7c\x1c\x38\x1c\x38\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x1c\x38\x1c\x38\x0e\x70\x0e\x70\x07\xe0\x07\xe0\x03\xc0\x03\xc0\x07\xe0\x07\xe0\x0e\x70\x0e\x70\x1c\x38\x1c\x38\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x38\x0e\x38\x0e\x1c\x1c\x1c\x1c\x0e\x38\x0e\x38\x07\x70\x07\x70\x03\xe0\x03\xe0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xfe\x3f\xfe\x00\x1c\x00\x1c\x00\x38\x00\x38\x00\x70\x00\x70\x00\xe0\x00\xe0\x01\xc0\x01\xc0\x03\x80\x03\x80\x07\x00\x07\x00\x0e\x00\x0e\x00\x1c\x00\x1c\x00\x3f\xfe\x3f\xfe\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\xf0\x07\xf0\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\x00\x07\xf0\x07\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x1c\x00\x1c\x00\x0e\x00\x0e\x00\x07\x00\x07\x00\x03\x80\x03\x80\x01\xc0\x01\xc0\x00\xe0\x00\xe0\x00\x70\x00\x70\x00\x38\x00\x38\x00\x1c\x00\x1c\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\xf0\x07\xf0\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x07\xf0\x07\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x03\xc0\x03\xc0\x07\xe0\x07\xe0\x0e\x70\x0e\x70\x1c\x38\x1c\x38\x38\x1c\x38\x1c\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\xff\xff\xff\xff\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x07\x00\x07\x00\x03\x80\x03\x80\x01\xc0\x01\xc0\x00\xe0\x00\xe0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x0f\xf8\x0f\xf8\x00\x0e\x00\x0e\x0f\xfe\x0f\xfe\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x0f\xfe\x0f\xfe\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x3f\xf0\x3f\xf0\x38\x1c\x38\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x3f\xf8\x3f\xf8\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x0f\xf8\x0f\xf8\x38\x0e\x38\x0e\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x0e\x38\x0e\x0f\xf8\x0f\xf8\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x07\xfe\x07\xfe\x1c\x0e\x1c\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x0f\xfe\x0f\xfe\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x0f\xf8\x0f\xf8\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x3f\xfe\x3f\xfe\x38\x00\x38\x00\x38\x00\x38\x00\x0f\xfc\x0f\xfc\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\xf8\x00\xf8\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x0f\xf0\x0f\xf0\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x0f\xf8\x0f\xf8\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x0f\xfe\x0f\xfe\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x1f\xf8\x1f\xf8\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x3b\xf8\x3b\xf8\x3c\x0e\x3c\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x70\x00\x70\x00\x70\x00\x70\x00\x00\x00\x00\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\x70\x00\xe0\x00\xe0\x0f\x80\x0f\x80\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x38\x0e\x38\x0e\x70\x0e\x70\x0e\xe0\x0e\xe0\x0f\xc0\x0f\xc0\x0e\xe0\x0e\xe0\x0e\x70\x0e\x70\x0e\x38\x0e\x38\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x3e\x78\x3e\x78\x39\xce\x39\xce\x39\xce\x39\xce\x39\xce\x39\xce\x39\xce\x39\xce\x39\xce\x39\xce\x39\xce\x39\xce\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xe0\x3f\xe0\x38\x38\x38\x38\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x07\xf0\x07\xf0\x1c\x1c\x1c\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x1c\x1c\x1c\x1c\x07\xf0\x07\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xf0\x3f\xf0\x38\x1c\x38\x1c\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x1c\x38\x1c\x3f\xf0\x3f\xf0\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x07\xfe\x07\xfe\x1c\x0e\x1c\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x1c\x0e\x1c\x0e\x07\xfe\x07\xfe\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xf0\x3f\xf0\x38\x1c\x38\x1c\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x38\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x0f\xfc\x0f\xfc\x38\x00\x38\x00\x38\x00\x38\x00\x0f\xf8\x0f\xf8\x00\x0e\x00\x0e\x00\x0e\x00\x0e\x1f\xf8\x1f\xf8\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x1f\xfc\x1f\xfc\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x38\x1c\x0f\xfc\x0f\xfc\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x70\x0e\x70\x0e\x38\x1c\x38\x1c\x1c\x38\x1c\x38\x0e\x70\x0e\x70\x07\xe0\x07\xe0\x03\xc0\x03\xc0\x01\x80\x01\x80\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x38\x0e\x39\xce\x39\xce\x3b\xee\x3b\xee\x1f\x7c\x1f\x7c\x0e\x38\x0e\x38\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x1c\x38\x1c\x38\x0e\x70\x0e\x70\x07\xe0\x07\xe0\x03\xc0\x03\xc0\x07\xe0\x07\xe0\x0e\x70\x0e\x70\x1c\x38\x1c\x38\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x38\x0e\x38\x0e\x1c\x1c\x1c\x1c\x0e\x38\x0e\x38\x07\x70\x07\x70\x03\xe0\x03\xe0\x01\xc0\x01\xc0\x03\x80\x03\x80\x07\x00\x07\x00\x0e\x00\x0e\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x3f\xfe\x3f\xfe\x00\x1c\x00\x1c\x00\x70\x00\x70\x01\xc0\x01\xc0\x07\x00\x07\x00\x1c\x00\x1c\x00\x3f\xfe\x3f\xfe\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\xf8\x00\xf8\x01\xc0\x01\xc0\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x1e\x00\x1e\x00\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x03\x80\x01\xc0\x01\xc0\x00\xf8\x00\xf8\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x1f\x00\x1f\x00\x03\x80\x03\x80\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x00\x78\x00\x78\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x01\xc0\x03\x80\x03\x80\x1f\x00\x1f\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x07\x9e\x07\x9e\x3c\xf0\x3c\xf0\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x01\xc0\x01\xc0\x07\x70\x07\x70\x1c\x1c\x1c\x1c\x70\x07\x70\x07\x70\x07\x70\x07\x7f\xff\x7f\xff\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00\x00'\ FONT = memoryview(_FONT)

vga2_8x8.py

vga2_8x8.py

"""converted from vga_8x8.bin """ WIDTH = 8 HEIGHT = 8 FIRST = 0x00 LAST = 0xff _FONT =\ b'\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x7e\x81\xa5\x81\xbd\x99\x81\x7e'\ b'\x7e\xff\xdb\xff\xc3\xe7\xff\x7e'\ b'\x6c\xfe\xfe\xfe\x7c\x38\x10\x00'\ b'\x10\x38\x7c\xfe\x7c\x38\x10\x00'\ b'\x38\x7c\x38\xfe\xfe\xd6\x10\x38'\ b'\x10\x38\x7c\xfe\xfe\x7c\x10\x38'\ b'\x00\x00\x18\x3c\x3c\x18\x00\x00'\ b'\xff\xff\xe7\xc3\xc3\xe7\xff\xff'\ b'\x00\x3c\x66\x42\x42\x66\x3c\x00'\ b'\xff\xc3\x99\xbd\xbd\x99\xc3\xff'\ b'\x0f\x07\x0f\x7d\xcc\xcc\xcc\x78'\ b'\x3c\x66\x66\x66\x3c\x18\x7e\x18'\ b'\x3f\x33\x3f\x30\x30\x70\xf0\xe0'\ b'\x7f\x63\x7f\x63\x63\x67\xe6\xc0'\ b'\x18\xdb\x3c\xe7\xe7\x3c\xdb\x18'\ b'\x80\xe0\xf8\xfe\xf8\xe0\x80\x00'\ b'\x02\x0e\x3e\xfe\x3e\x0e\x02\x00'\ b'\x18\x3c\x7e\x18\x18\x7e\x3c\x18'\ b'\x66\x66\x66\x66\x66\x00\x66\x00'\ b'\x7f\xdb\xdb\x7b\x1b\x1b\x1b\x00'\ b'\x3e\x61\x3c\x66\x66\x3c\x86\x7c'\ b'\x00\x00\x00\x00\x7e\x7e\x7e\x00'\ b'\x18\x3c\x7e\x18\x7e\x3c\x18\xff'\ b'\x18\x3c\x7e\x18\x18\x18\x18\x00'\ b'\x18\x18\x18\x18\x7e\x3c\x18\x00'\ b'\x00\x18\x0c\xfe\x0c\x18\x00\x00'\ b'\x00\x30\x60\xfe\x60\x30\x00\x00'\ b'\x00\x00\xc0\xc0\xc0\xfe\x00\x00'\ b'\x00\x24\x66\xff\x66\x24\x00\x00'\ b'\x00\x18\x3c\x7e\xff\xff\x00\x00'\ b'\x00\xff\xff\x7e\x3c\x18\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00'\ b'\x18\x3c\x3c\x18\x18\x00\x18\x00'\ b'\x66\x66\x24\x00\x00\x00\x00\x00'\ b'\x6c\x6c\xfe\x6c\xfe\x6c\x6c\x00'\ b'\x18\x3e\x60\x3c\x06\x7c\x18\x00'\ b'\x00\xc6\xcc\x18\x30\x66\xc6\x00'\ b'\x38\x6c\x38\x76\xdc\xcc\x76\x00'\ b'\x18\x18\x30\x00\x00\x00\x00\x00'\ b'\x0c\x18\x30\x30\x30\x18\x0c\x00'\ b'\x30\x18\x0c\x0c\x0c\x18\x30\x00'\ b'\x00\x66\x3c\xff\x3c\x66\x00\x00'\ b'\x00\x18\x18\x7e\x18\x18\x00\x00'\ b'\x00\x00\x00\x00\x00\x18\x18\x30'\ b'\x00\x00\x00\x7e\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x18\x18\x00'\ b'\x06\x0c\x18\x30\x60\xc0\x80\x00'\ b'\x38\x6c\xc6\xd6\xc6\x6c\x38\x00'\ b'\x18\x38\x18\x18\x18\x18\x7e\x00'\ b'\x7c\xc6\x06\x1c\x30\x66\xfe\x00'\ b'\x7c\xc6\x06\x3c\x06\xc6\x7c\x00'\ b'\x1c\x3c\x6c\xcc\xfe\x0c\x1e\x00'\ b'\xfe\xc0\xc0\xfc\x06\xc6\x7c\x00'\ b'\x38\x60\xc0\xfc\xc6\xc6\x7c\x00'\ b'\xfe\xc6\x0c\x18\x30\x30\x30\x00'\ b'\x7c\xc6\xc6\x7c\xc6\xc6\x7c\x00'\ b'\x7c\xc6\xc6\x7e\x06\x0c\x78\x00'\ b'\x00\x18\x18\x00\x00\x18\x18\x00'\ b'\x00\x18\x18\x00\x00\x18\x18\x30'\ b'\x06\x0c\x18\x30\x18\x0c\x06\x00'\ b'\x00\x00\x7e\x00\x00\x7e\x00\x00'\ b'\x60\x30\x18\x0c\x18\x30\x60\x00'\ b'\x7c\xc6\x0c\x18\x18\x00\x18\x00'\ b'\x7c\xc6\xde\xde\xde\xc0\x78\x00'\ b'\x38\x6c\xc6\xfe\xc6\xc6\xc6\x00'\ b'\xfc\x66\x66\x7c\x66\x66\xfc\x00'\ b'\x3c\x66\xc0\xc0\xc0\x66\x3c\x00'\ b'\xf8\x6c\x66\x66\x66\x6c\xf8\x00'\ b'\xfe\x62\x68\x78\x68\x62\xfe\x00'\ b'\xfe\x62\x68\x78\x68\x60\xf0\x00'\ b'\x3c\x66\xc0\xc0\xce\x66\x3a\x00'\ b'\xc6\xc6\xc6\xfe\xc6\xc6\xc6\x00'\ b'\x3c\x18\x18\x18\x18\x18\x3c\x00'\ b'\x1e\x0c\x0c\x0c\xcc\xcc\x78\x00'\ b'\xe6\x66\x6c\x78\x6c\x66\xe6\x00'\ b'\xf0\x60\x60\x60\x62\x66\xfe\x00'\ b'\xc6\xee\xfe\xfe\xd6\xc6\xc6\x00'\ b'\xc6\xe6\xf6\xde\xce\xc6\xc6\x00'\ b'\x7c\xc6\xc6\xc6\xc6\xc6\x7c\x00'\ b'\xfc\x66\x66\x7c\x60\x60\xf0\x00'\ b'\x7c\xc6\xc6\xc6\xc6\xce\x7c\x0e'\ b'\xfc\x66\x66\x7c\x6c\x66\xe6\x00'\ b'\x3c\x66\x30\x18\x0c\x66\x3c\x00'\ b'\x7e\x7e\x5a\x18\x18\x18\x3c\x00'\ b'\xc6\xc6\xc6\xc6\xc6\xc6\x7c\x00'\ b'\xc6\xc6\xc6\xc6\xc6\x6c\x38\x00'\ b'\xc6\xc6\xc6\xd6\xd6\xfe\x6c\x00'\ b'\xc6\xc6\x6c\x38\x6c\xc6\xc6\x00'\ b'\x66\x66\x66\x3c\x18\x18\x3c\x00'\ b'\xfe\xc6\x8c\x18\x32\x66\xfe\x00'\ b'\x3c\x30\x30\x30\x30\x30\x3c\x00'\ b'\xc0\x60\x30\x18\x0c\x06\x02\x00'\ b'\x3c\x0c\x0c\x0c\x0c\x0c\x3c\x00'\ b'\x10\x38\x6c\xc6\x00\x00\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\xff'\ b'\x30\x18\x0c\x00\x00\x00\x00\x00'\ b'\x00\x00\x78\x0c\x7c\xcc\x76\x00'\ b'\xe0\x60\x7c\x66\x66\x66\xdc\x00'\ b'\x00\x00\x7c\xc6\xc0\xc6\x7c\x00'\ b'\x1c\x0c\x7c\xcc\xcc\xcc\x76\x00'\ b'\x00\x00\x7c\xc6\xfe\xc0\x7c\x00'\ b'\x3c\x66\x60\xf8\x60\x60\xf0\x00'\ b'\x00\x00\x76\xcc\xcc\x7c\x0c\xf8'\ b'\xe0\x60\x6c\x76\x66\x66\xe6\x00'\ b'\x18\x00\x38\x18\x18\x18\x3c\x00'\ b'\x06\x00\x06\x06\x06\x66\x66\x3c'\ b'\xe0\x60\x66\x6c\x78\x6c\xe6\x00'\ b'\x38\x18\x18\x18\x18\x18\x3c\x00'\ b'\x00\x00\xec\xfe\xd6\xd6\xd6\x00'\ b'\x00\x00\xdc\x66\x66\x66\x66\x00'\ b'\x00\x00\x7c\xc6\xc6\xc6\x7c\x00'\ b'\x00\x00\xdc\x66\x66\x7c\x60\xf0'\ b'\x00\x00\x76\xcc\xcc\x7c\x0c\x1e'\ b'\x00\x00\xdc\x76\x60\x60\xf0\x00'\ b'\x00\x00\x7e\xc0\x7c\x06\xfc\x00'\ b'\x30\x30\xfc\x30\x30\x36\x1c\x00'\ b'\x00\x00\xcc\xcc\xcc\xcc\x76\x00'\ b'\x00\x00\xc6\xc6\xc6\x6c\x38\x00'\ b'\x00\x00\xc6\xd6\xd6\xfe\x6c\x00'\ b'\x00\x00\xc6\x6c\x38\x6c\xc6\x00'\ b'\x00\x00\xc6\xc6\xc6\x7e\x06\xfc'\ b'\x00\x00\x7e\x4c\x18\x32\x7e\x00'\ b'\x0e\x18\x18\x70\x18\x18\x0e\x00'\ b'\x18\x18\x18\x18\x18\x18\x18\x00'\ b'\x70\x18\x18\x0e\x18\x18\x70\x00'\ b'\x76\xdc\x00\x00\x00\x00\x00\x00'\ b'\x00\x10\x38\x6c\xc6\xc6\xfe\x00'\ b'\x7c\xc6\xc0\xc0\xc6\x7c\x0c\x78'\ b'\xcc\x00\xcc\xcc\xcc\xcc\x76\x00'\ b'\x0c\x18\x7c\xc6\xfe\xc0\x7c\x00'\ b'\x7c\x82\x78\x0c\x7c\xcc\x76\x00'\ b'\xc6\x00\x78\x0c\x7c\xcc\x76\x00'\ b'\x30\x18\x78\x0c\x7c\xcc\x76\x00'\ b'\x30\x30\x78\x0c\x7c\xcc\x76\x00'\ b'\x00\x00\x7e\xc0\xc0\x7e\x0c\x38'\ b'\x7c\x82\x7c\xc6\xfe\xc0\x7c\x00'\ b'\xc6\x00\x7c\xc6\xfe\xc0\x7c\x00'\ b'\x30\x18\x7c\xc6\xfe\xc0\x7c\x00'\ b'\x66\x00\x38\x18\x18\x18\x3c\x00'\ b'\x7c\x82\x38\x18\x18\x18\x3c\x00'\ b'\x30\x18\x00\x38\x18\x18\x3c\x00'\ b'\xc6\x38\x6c\xc6\xfe\xc6\xc6\x00'\ b'\x38\x6c\x7c\xc6\xfe\xc6\xc6\x00'\ b'\x18\x30\xfe\xc0\xf8\xc0\xfe\x00'\ b'\x00\x00\x7e\x18\x7e\xd8\x7e\x00'\ b'\x3e\x6c\xcc\xfe\xcc\xcc\xce\x00'\ b'\x7c\x82\x7c\xc6\xc6\xc6\x7c\x00'\ b'\xc6\x00\x7c\xc6\xc6\xc6\x7c\x00'\ b'\x30\x18\x7c\xc6\xc6\xc6\x7c\x00'\ b'\x78\x84\x00\xcc\xcc\xcc\x76\x00'\ b'\x60\x30\xcc\xcc\xcc\xcc\x76\x00'\ b'\xc6\x00\xc6\xc6\xc6\x7e\x06\xfc'\ b'\xc6\x38\x6c\xc6\xc6\x6c\x38\x00'\ b'\xc6\x00\xc6\xc6\xc6\xc6\x7c\x00'\ b'\x18\x18\x7e\xc0\xc0\x7e\x18\x18'\ b'\x38\x6c\x64\xf0\x60\x66\xfc\x00'\ b'\x66\x66\x3c\x7e\x18\x7e\x18\x18'\ b'\xf8\xcc\xcc\xfa\xc6\xcf\xc6\xc7'\ b'\x0e\x1b\x18\x3c\x18\xd8\x70\x00'\ b'\x18\x30\x78\x0c\x7c\xcc\x76\x00'\ b'\x0c\x18\x00\x38\x18\x18\x3c\x00'\ b'\x0c\x18\x7c\xc6\xc6\xc6\x7c\x00'\ b'\x18\x30\xcc\xcc\xcc\xcc\x76\x00'\ b'\x76\xdc\x00\xdc\x66\x66\x66\x00'\ b'\x76\xdc\x00\xe6\xf6\xde\xce\x00'\ b'\x3c\x6c\x6c\x3e\x00\x7e\x00\x00'\ b'\x38\x6c\x6c\x38\x00\x7c\x00\x00'\ b'\x18\x00\x18\x18\x30\x63\x3e\x00'\ b'\x00\x00\x00\xfe\xc0\xc0\x00\x00'\ b'\x00\x00\x00\xfe\x06\x06\x00\x00'\ b'\x63\xe6\x6c\x7e\x33\x66\xcc\x0f'\ b'\x63\xe6\x6c\x7a\x36\x6a\xdf\x06'\ b'\x18\x00\x18\x18\x3c\x3c\x18\x00'\ b'\x00\x33\x66\xcc\x66\x33\x00\x00'\ b'\x00\xcc\x66\x33\x66\xcc\x00\x00'\ b'\x22\x88\x22\x88\x22\x88\x22\x88'\ b'\x55\xaa\x55\xaa\x55\xaa\x55\xaa'\ b'\x77\xdd\x77\xdd\x77\xdd\x77\xdd'\ b'\x18\x18\x18\x18\x18\x18\x18\x18'\ b'\x18\x18\x18\x18\xf8\x18\x18\x18'\ b'\x18\x18\xf8\x18\xf8\x18\x18\x18'\ b'\x36\x36\x36\x36\xf6\x36\x36\x36'\ b'\x00\x00\x00\x00\xfe\x36\x36\x36'\ b'\x00\x00\xf8\x18\xf8\x18\x18\x18'\ b'\x36\x36\xf6\x06\xf6\x36\x36\x36'\ b'\x36\x36\x36\x36\x36\x36\x36\x36'\ b'\x00\x00\xfe\x06\xf6\x36\x36\x36'\ b'\x36\x36\xf6\x06\xfe\x00\x00\x00'\ b'\x36\x36\x36\x36\xfe\x00\x00\x00'\ b'\x18\x18\xf8\x18\xf8\x00\x00\x00'\ b'\x00\x00\x00\x00\xf8\x18\x18\x18'\ b'\x18\x18\x18\x18\x1f\x00\x00\x00'\ b'\x18\x18\x18\x18\xff\x00\x00\x00'\ b'\x00\x00\x00\x00\xff\x18\x18\x18'\ b'\x18\x18\x18\x18\x1f\x18\x18\x18'\ b'\x00\x00\x00\x00\xff\x00\x00\x00'\ b'\x18\x18\x18\x18\xff\x18\x18\x18'\ b'\x18\x18\x1f\x18\x1f\x18\x18\x18'\ b'\x36\x36\x36\x36\x37\x36\x36\x36'\ b'\x36\x36\x37\x30\x3f\x00\x00\x00'\ b'\x00\x00\x3f\x30\x37\x36\x36\x36'\ b'\x36\x36\xf7\x00\xff\x00\x00\x00'\ b'\x00\x00\xff\x00\xf7\x36\x36\x36'\ b'\x36\x36\x37\x30\x37\x36\x36\x36'\ b'\x00\x00\xff\x00\xff\x00\x00\x00'\ b'\x36\x36\xf7\x00\xf7\x36\x36\x36'\ b'\x18\x18\xff\x00\xff\x00\x00\x00'\ b'\x36\x36\x36\x36\xff\x00\x00\x00'\ b'\x00\x00\xff\x00\xff\x18\x18\x18'\ b'\x00\x00\x00\x00\xff\x36\x36\x36'\ b'\x36\x36\x36\x36\x3f\x00\x00\x00'\ b'\x18\x18\x1f\x18\x1f\x00\x00\x00'\ b'\x00\x00\x1f\x18\x1f\x18\x18\x18'\ b'\x00\x00\x00\x00\x3f\x36\x36\x36'\ b'\x36\x36\x36\x36\xff\x36\x36\x36'\ b'\x18\x18\xff\x18\xff\x18\x18\x18'\ b'\x18\x18\x18\x18\xf8\x00\x00\x00'\ b'\x00\x00\x00\x00\x1f\x18\x18\x18'\ b'\xff\xff\xff\xff\xff\xff\xff\xff'\ b'\x00\x00\x00\x00\xff\xff\xff\xff'\ b'\xf0\xf0\xf0\xf0\xf0\xf0\xf0\xf0'\ b'\x0f\x0f\x0f\x0f\x0f\x0f\x0f\x0f'\ b'\xff\xff\xff\xff\x00\x00\x00\x00'\ b'\x00\x00\x76\xdc\xc8\xdc\x76\x00'\ b'\x78\xcc\xcc\xd8\xcc\xc6\xcc\x00'\ b'\xfe\xc6\xc0\xc0\xc0\xc0\xc0\x00'\ b'\x00\x00\xfe\x6c\x6c\x6c\x6c\x00'\ b'\xfe\xc6\x60\x30\x60\xc6\xfe\x00'\ b'\x00\x00\x7e\xd8\xd8\xd8\x70\x00'\ b'\x00\x00\x66\x66\x66\x66\x7c\xc0'\ b'\x00\x76\xdc\x18\x18\x18\x18\x00'\ b'\x7e\x18\x3c\x66\x66\x3c\x18\x7e'\ b'\x38\x6c\xc6\xfe\xc6\x6c\x38\x00'\ b'\x38\x6c\xc6\xc6\x6c\x6c\xee\x00'\ b'\x0e\x18\x0c\x3e\x66\x66\x3c\x00'\ b'\x00\x00\x7e\xdb\xdb\x7e\x00\x00'\ b'\x06\x0c\x7e\xdb\xdb\x7e\x60\xc0'\ b'\x1e\x30\x60\x7e\x60\x30\x1e\x00'\ b'\x00\x7c\xc6\xc6\xc6\xc6\xc6\x00'\ b'\x00\xfe\x00\xfe\x00\xfe\x00\x00'\ b'\x18\x18\x7e\x18\x18\x00\x7e\x00'\ b'\x30\x18\x0c\x18\x30\x00\x7e\x00'\ b'\x0c\x18\x30\x18\x0c\x00\x7e\x00'\ b'\x0e\x1b\x1b\x18\x18\x18\x18\x18'\ b'\x18\x18\x18\x18\x18\xd8\xd8\x70'\ b'\x00\x18\x00\x7e\x00\x18\x00\x00'\ b'\x00\x76\xdc\x00\x76\xdc\x00\x00'\ b'\x38\x6c\x6c\x38\x00\x00\x00\x00'\ b'\x00\x00\x00\x18\x18\x00\x00\x00'\ b'\x00\x00\x00\x18\x00\x00\x00\x00'\ b'\x0f\x0c\x0c\x0c\xec\x6c\x3c\x1c'\ b'\x6c\x36\x36\x36\x36\x00\x00\x00'\ b'\x78\x0c\x18\x30\x7c\x00\x00\x00'\ b'\x00\x00\x3c\x3c\x3c\x3c\x00\x00'\ b'\x00\x00\x00\x00\x00\x00\x00\x00'\ FONT = memoryview(_FONT)

PicoS_Lib

(以下のプログラムは予めPicoSubに書き込んでおいてください)
プログラムは、PicoM_libにあるものと同じです。
mojipy.py
ymf825py.py

Hyodotのアイコン画像
真空管アンプ、トランジスタ回路、IC(SN74xx)、Z80。 その後、24時間戦えますかモード?に突入、電子工作から途絶。 定年で戦いを終え、電子工作を再開、浦島太郎。 PIC、IchigoJam、Arduino、RaspberryPi、micro:bit、 今は、Pico Pythonで遊んでいます。 アセンブラーもやるアラ・セブン。 ・・・アラセブン 脳トレ成果を 置き土産・・・
  • Hyodot さんが 2024/10/27 に 編集 をしました。 (メッセージ: 初版)
ログインしてコメントを投稿する