LoginSignup
7
2

More than 3 years have passed since last update.

電光掲示板シェル芸 その3

Last updated at Posted at 2019-12-04

多色電光掲示板シェル芸

前回までの電光掲示板シェル芸は無圧縮PBMという白黒2色のみの画像フォーマットを介しているため、矩形領域内に2色の図形を描くものでした。
しかし、💩.png💃.pngなど、フルカラーの絵文字を描くためには2色では足りません。
今回は矩形領域内に3色以上の多色で図形を描く電光掲示板シェル芸をやっていきます。

PGM電光掲示板シェル芸

多色の電光掲示板シェル芸をやるには、画像の各ピクセルをどの絵文字で置き換えるかを決定する必要があります。
この置換絵文字決定を、無圧縮PGMフォーマットを介して、グレースケールの明るさを用いて行うのが、PGM電光掲示板シェル芸です。
まず無圧縮PGMってなんなの?ということを💩.pngを無圧縮PGMにして見てみましょう。

$ echo -e "\n💩 "|textimg|convert - -trim -compress none pgm:-
P2
20 20
255
0 0 0 0 0 0 0 49 71 0 0 0 0 0 0 0 0 0 0 0 
0 0 0 0 0 0 0 0 108 110 71 0 0 0 0 0 0 0 0 0 
0 0 0 0 0 0 0 70 155 160 160 71 2 0 0 0 0 0 0 0 
0 0 0 0 0 0 71 155 156 156 156 156 71 0 0 0 0 0 0 0 
0 0 0 0 0 0 71 151 151 151 151 151 71 0 0 0 0 0 0 0 
0 0 0 0 0 7 71 145 145 145 145 145 107 71 1 0 0 0 0 0 
0 0 0 0 23 90 140 140 140 140 140 140 140 140 79 10 0 0 0 0 
0 0 0 0 71 135 135 135 135 135 135 135 135 135 135 71 0 0 0 0 
0 0 0 0 71 154 255 255 255 130 130 255 255 255 154 71 0 0 0 0 
0 0 0 36 71 255 36 36 255 133 133 255 36 36 255 71 13 0 0 0 
0 0 68 119 122 255 44 36 255 123 123 255 36 46 255 122 97 53 0 0 
0 0 71 116 116 119 255 255 253 116 116 253 255 255 119 116 116 71 0 0 
0 0 71 111 111 111 111 111 111 111 111 111 111 111 111 111 111 71 0 0 
0 30 71 106 106 255 255 245 232 220 220 232 245 255 255 106 106 71 22 0 
71 98 101 101 101 0 5 214 255 255 255 255 214 5 0 101 101 101 96 71 
78 97 97 97 97 97 0 0 0 0 0 0 0 0 97 97 97 97 97 78 
89 93 93 93 93 93 93 0 0 0 0 0 0 93 93 93 93 93 93 89 
71 93 93 93 93 93 93 93 93 90 90 93 93 93 93 93 93 93 93 71 
0 64 71 71 75 84 89 93 93 93 93 93 93 85 80 74 71 71 71 1 
0 0 0 0 0 0 0 7 14 14 14 7 7 0 0 0 0 0 0 0 

1行目の P2 は、無圧縮PGMであることを示しています。
2行目 20 20 は画像の横縦のサイズ。
3行目 255 は、画素値が 0-255 の256段階であることを示しています。
4行目以降は画素値ですね。値が大きいほど明るく、0が黒で255が白になります。
白目のあたりに255、黒目の当たりに36という値がありますね。
グレースケール画像の見た目も見ておきましょう。

$ echo -e "\n💩 "|textimg|convert - -trim -compress none pgm:-|convert - -scale 200x200 /images/x.png

x.png

色分けを考えます。
背景・黒目・口の中は黒、白目と歯は白、それ以外のグレーの部分を茶色にすればよさそうです。
3色ですね。

現時点で0-255の256色ある状態です。すこし色数を減らしてみます。

$ echo -e "\n💩 "|textimg|convert - -trim -compress none pgm:-|tail -n +4|opy '[x//26 for x in F[1:]]'|tr -d ' '
00000001200000000000
00000000442000000000
00000002566200000000
00000025666620000000
00000025555520000000
00000025555542000000
00000355555555300000
00002555555555520000
00002599955999520000
00012911955911920000
00244911944911943200
00244499944999444200
00244444444444444200
01244999888899944200
23333008999980033332
33333300000000333333
33333330000003333333
23333333333333333332
02222333333333322220
00000000000000000000

tailでPGMのヘッダを削って、データ部分をopyで26で割り、trでスペースを削除しました。
あとはsedで0-9の10文字を3色の絵文字に置き換えれば電光掲示板になります。

$ echo -e "\n💩 "|textimg|convert - -trim -compress none pgm:-|tail -n +4|opy '[x//26 for x in F[1:]]'|tr -d ' '|sed 'y/0123456789/⚫⚫💩💩💩💩⚪⚪⚪⚪/'|textimg -s

01を⚫、2345を💩.png、6789を⚪で置き換えてみました。
t.png
頭のてっぺんが白くなっちゃってますね。
6も💩.pngにして調整しましょう。

$ echo -e "\n💩 "|textimg|convert - -trim -compress none pgm:-|tail -n +4|opy '[x//26 for x in F[1:]]'|tr -d ' '|sed 'y/0123456789/⚫⚫💩💩💩💩💩⚪⚪⚪/'|textimg -s

t.png
完成です。
余談ですが、PGMフォーマットというのは、ロボットが移動経路を計画するための地図データに使われたりしますね。簡単に言うと「ロボットが現在位置から目的地まで移動する時に、なるべく白っぽいところを通って行く」みたいな感じです。なので、

$ convert map.pgm -compress none pgm:-

としてマップに入っている値をざっと見てみるみたいな、imagemagick を使ったPGMのハンドリングができると、ちょっと便利です。

denko-multi 関数

@egpl0_sh_2 さんがシェル芸botで使える denko-multi という関数を定義しています。これを利用してシェル芸botでPGM電光掲示板シェル芸ができます。
関数定義のツイートをコメント付きリツイートすることで、シェル芸botがツイートのチェーンを認識して、関数を呼ぶことができます。
関数定義ツイート


関数を呼ぶツイート

実行結果

PPM電光掲示板シェル芸

グレースケールの陰影だけでは色の決定がうまくいかないような場合、無圧縮PPMフォーマットを介して、RGB空間で置換絵文字を決定することができます。これがPPM電光掲示板シェル芸です。
ここでは💃.png(ダンサーの絵文字。コードポイントU+1F483)を電光掲示板にしてみましょう。
x.png
こんな感じの絵文字です。

$ echo -e "\n💃 "|textimg -F50|convert - -trim -compress none ppm:-|tail -n +4|awk '{for(i=0;i<NF/3;i++){r=3*i+1;g=3*i+2;b=3*i+3;printf int($r/128)*4+int($g/128)*2+int($b/28)}print ""}'|sed 'y/01234567/🌑🔵🔵🔵🔴🔵🌕🌕/'|textimg -s

RGBそれぞれの画素値が128以上か否かで 2^3 = 8象限に分割して、sedで置換しています。
t.png

髪色・肌色・服色が分別されているので、絵文字を変更するとちょっとした着せ替えができます。
ピンク髪・緑肌・青服にしてみましょう。

$ echo -e "\n💃 "|textimg -F50|convert - -trim -compress none ppm:-|tail -n +4|awk '{for(i=0;i<NF/3;i++){r=3*i+1;g=3*i+2;b=3*i+3;printf int($r/128)*4+int($g/128)*2+int($b/28)}print ""}'|sed 'y/01234567/🌑🍑🍑🍑🔵🍑🍈🍈/'|textimg -s

t.png
なかなかいいですね。
これをさらに高精度でやろうとすると、RGB空間を分割する数式を考えることになりますが大変そうですね。
それをやらずに済ませるためには、imagemagick でカラーパレットを指定して、カラーパレットに含まれる色だけを使って減色する方法があります。

$ convert <(echo -e "\n💃 "|textimg -F50) -trim -map <(echo P3 5 1 255 199 26 0 169 32 2 255 191 20 88 60 51 0 0 0) -compress none ppm:-|tail -n +4|sed 's/199 26 0/🍎/g;s/255 191 20/🌕/g;s/169 32 2/🍫/g;s/88 60 51/🍩/g;s/0 0 0/🌑/g'|tr -d ' '|textimg -s

準備段階として、カラーピッカー(gpick)で「服の明るい部分(199 26 0)」「肌(169 32 2)」「服の暗い部分(88 60 51)」「髪(88 60 51)」「背景(0 0 0)」の画素値を調べています。それをパレットにして減色して、無圧縮PPMで出力すると、パレットにある画素値の列になるので、それを絵文字に置換しています。
t.png
なかなかいいですね。
本当は1ピクセルのデータごとに区切って検索しないと、あるピクセルのGBと次のピクセルのRが、別の色のRGBとマッチしちゃったりしてよろしくないんですけど、今回はそうならないパレットになってるのでよしとしましょう。
この方式なら色数の多い画像でも強引に電光掲示板にできそうです。
試してみましょう。
入力画像: monalisa.jpg
monalisa.jpg

$ a="204 183 76";b="39 19 36";c="59 45 49";d="120 138 80";e="74 83 59";f="125 87 30";convert monalisa.jpg -scale 80x80 -map <(echo P3 6 1 255 $a $b $c $d $e $f) -compress none ppm:-|tail -n +4|sed "s/$a/🥚/g;s/$b/🌑/g;s/$c/🎱/g;s/$d/🍈/g;s/$e/🍀/g;s/$f/🍩/g"|tr -d ' '|textimg -s

t.png

なるほど。今回はとりあえずこんなところです。

所感

電光掲示板シェル芸の現状を3回に分けてやってみました。
もともとやっていたことをまとめるだけなので楽チンかなーと思っていたのですが、いくつか新手が見つかったりしてちょっと手間がかかりました。
僕の考える電光掲示板シェル芸の面白いところは、無圧縮PNMというフォーマットを介して「画像の空間」と「文字列の空間」を行ったり来たりできるところで、シェル芸で文字列操作・数値計算をして生成した文字列が画像になったり、imagemagickなどで生成した画像が文字列になったりする。そのあたりが醍醐味かと思います。
ところで🍩.pngみたいな暗い色って、実際の電光掲示板では出しにくそうな気がしますね。PWMで出せるのかな?環境の明るさの影響をすごく受けそう。

7
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
7
2