概要
ArduinoようOLEDを買いました。
ちょっとしたものを作るのに便利です。
プログラムを書いていて疑問が増えてきたので検証していきます。
配線等の使い方は下記がおすすめです。
プログラムについてはこちら
文字関係
TextSize
display.clearDisplay();
display.setCursor(0, 0);
display.setTextSize(1);
display.setTextColor(WHITE);
display.println("hello");
display.setTextSize(2);
display.setTextColor(WHITE);
display.println("hello");
display.setTextSize(3);
display.setTextColor(WHITE);
display.println("hello");
display.setTextSize(4);
display.setTextColor(WHITE);
display.println("hello");
display.display();
ビットサイズも確認
textSize | 縦bit | 横bit |
---|---|---|
4 | 32bit | 24bit |
3 | 25bit | 18bit |
2 | 16bit | 12bit |
1 | 8bit | 6bit |
ぐらいの見積もり計算するのがおすすめです
textSize(1)で"2"を描画するとかなり見にくくなります。
textSize(1)はあまりおすすめしません。
textSize(4)
textSize(3)
textSize(2)
textSize(1)
setCursor:同じ場所に書く→かぶる
display.clearDisplay();
display.setCursor(0, 0);
display.setTextSize(2);
display.setTextColor(WHITE);
display.println("hello");
display.setCursor(0, 0);
display.setTextSize(2);
display.setTextColor(WHITE);
display.println("goodbye");
display.display();
setCursor:方向
setCursor(右方向、下方向)
setCursor:setCursorの場所
- 1文字分カーソル移動ではなく、ドット移動の可能性が高い
- 上下移動と左右移動で比率が異なる
display.clearDisplay();
display.setCursor(30, 30);
display.setTextSize(2);
display.setTextColor(WHITE);
display.println("hello");
setCursor:文字サイズの影響を受けるかどうか
画面最大サイズ:
今持っているものは
- 右方向110
- 下方向60
for(int i = 0;i<=200; i++){
display.clearDisplay();
display.setTextSize(4);
display.setCursor(i, 0);
display.setTextColor(WHITE);
display.println("hello");
display.display();
Serial.println(i);
delay(100);
}
数値表示
display.println()
の関係でintを直に入れられない
void num_to_string(int num, int digit) {
char buffer_num[10];
dtostrf(num, digit, 0, buffer_num);
String num_string = String(buffer_num);
return num_string;
}
int hoge = 10;
display.println(num_to_string(hoge,2));
こんな感じで関数を作ると楽かも
日本語表示
ライブラリが存在するがArduinoのメモリ容量的に厳しかった。導入は可能だった。
bitmapで表示するしかない?
文字を中央に配置したい
関数等は存在しない、文字数から先読みしてsetCursorするしかない。
点
横127,縦63のビットだと考えて数値を入力
display.drawPixel(50, 10, WHITE); //右.下.COLOR
線
点と同じ
for(int i = 0;i<=150; i++){
display.clearDisplay();
display.drawLine(0, 0, i, 63, WHITE);
display.display();
Serial.println(i);
delay(100);
}
bitmapイメージ
drawBitmapを使用する。
unsigned char logo_bmp[] =
{ 0b00000111, 0b11100000,
0b00000111, 0b11100000,
0b00000111, 0b11100000,
0b11100000, 0b00000111,
0b11100000, 0b00000111,
0b11100000, 0b00000111
};
display.clearDisplay();
//(右、下、bmp、横幅、縦幅、1
display.drawBitmap( 0, 0, logo_bmp, 16, 6, 1);
display.display();
上下に隙間があるように見える・・・
bitイメージを作成するプログラム欲しいな・・・
画像変換プログラム結果:python使用
若干わかりにくいし、元のかわいさが損なわれてしまった(目を閉じていると分かりにくくなるのかな)
むずいなぁ
デフォルメ絵なら行けるかな
細かい部分が混ざると潰れちゃう
表示されてない点があるなぁ
線の膨張を強めたら、表示される。
目の周りとか細かいディテールがだせてないなぁ
まぁ、細かい絵はこのディスプレイじゃ無理だね。
漢字とかそうゆう表示機として使うべきだ
画像から配列生成プログラム
def array_to_byte(array):
string = "0b"
for i in array:
if i == 255:
string = string + "1"
else:
string = string + "0"
return string
import cv2
import numpy as np
width = 128
height = 64
img = cv2.imread('ueno2.png')
img_gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
# 二値化(閾値100を超えた画素を255にする。)
ret, img_thresh = cv2.threshold(img_gray, 0, 255, cv2.THRESH_OTSU)
# 白黒反転
img_thresh = cv2.bitwise_not(img_thresh)
# 線を太くする
"""
bold = 1
#kernel = cv2.getStructuringElement(cv2.MORPH_CROSS, (bold, bold))
kernel = np.ones((bold,bold),np.uint8)
img_ker = cv2.dilate(img_thresh, kernel)
"""
img_ker = img_thresh
# 線を細くする 白黒反転しないときは使う
"""
thin = 0
kernel = cv2.getStructuringElement(cv2.MORPH_CROSS, (thin, thin))
img_ker = cv2.erode(img_thresh, kernel)
"""
# サイズを合わせる
k =0.4
img_size = cv2.resize(img_ker,None,fx = k, fy = k)
# 切り抜き
x_a = -10
y_a = 0
center_x = int(img_size.shape[0]/2)
center_y = int(img_size.shape[1]/2)
width_2 = int(width/2)
height_2 = int(height/2)
img_crop = img_size[center_x - height_2 + y_a : center_x + height_2 + y_a
,center_y - width_2 + x_a : center_y + width_2 + x_a]
# 表示用に拡大
img_show = cv2.resize(img_crop,None,fx = 5, fy = 5)
cv2.imshow("sample",img_show)
#bitimage 変数用に書き出し
x_num = int(width/8)
print("{",end="")
for y in range(height):
for x in range(x_num):
byte = img_crop[y][8*x:8*x+8]
print(array_to_byte(byte), end="")
if x == x_num-1 and y == height-1:
#最後にはカンマつけない
pass
else:
print(", ", end="")
print(" ")
print("};",end="")
cv2.waitKey(0)
画像サイズが大きいとエラーが出る
画像サイズ6464をbitmapdrawしようとしたところエラーが出た(3232だと行けるんだけど)
原因は
static const unsigned char PROGMEM logo_bmp[] =
のようにプログラムメモリ上に置くといけるけど、
unsigned char logo_bmp[] =
だとエラー吐く
(追記)部品形状について
0.96inch,OLED,SSD1306という商品の中でもディスプレイのサイズが若干異なる場合がある。
ネジ穴の位置・ディスプレイの横幅が異なる場合がある。
商品化する予定の方はお気をつけください。
(追記)メモリについて
128*64 = 8192bit = 1024byte = 1kB -> 50%以上のメモリ使用でAllocationエラーが起きる。
ガチでメモリがすくねぇ。
禄にプログラムが入らねぇ(笑)
float配列とか作るとすぐ超えるから注意ね
unsigned longとかでなんとかしな
もしくは
#define SCREEN_WIDTH 128 // OLED display width, in pixels
#define SCREEN_HEIGHT 32 // OLED display height, in pixels
ピクセル量をへらすしかねぇな・・・これで半分500Byteにできる。
128のほうは減らすと描画範囲が減るので注意ね