Pyxel実験室「255色の画像を表示(その2)」
前回は力技で表示してましたが、data_ptrという技を習得したのでこちらに応用してみました。
ただし、255色のカラーデータをセットする方法がpyxpalファイルをロードするのみであることが少々ネックですが、そこさえ目を瞑れば可能になります。(前準備が必要です)
ひとまず手元にあったJPGファイルで試しています。
他の画像ファイルでも試す必要はありますが、ひとまず下記コードで実現できたので公開。
from PIL import Image
import numpy as np
import pyxel
#----------------------------------------------------
#前準備:
#あらかじめ「pyxel edit」を実行してエディタを起動&保存して
#空のファイル:my_resource.pyxresを作成しておく
#(255色カラーパレットファイル作成用)
#----------------------------------------------------
#画像ファイル読み込み
im = Image.open('nextfighter_06.JPG')
#画像サイズ変更
resize_im = im.resize(( 640, 480 ))
#減色
im_q = resize_im.quantize(colors=255, method=0, kmeans=100, dither=1)
#画像からパレットを取得してRGBの順になるように3つずつのlistに格納する
_palette = list(zip(*[iter(im_q.getpalette())]*3))
#画像データ取得
_pix = np.asarray(im_q)
#画像サイズ取得
SCREEN_WIDTH = int( im_q.size[0] )
SCREEN_HEIGHT = int( im_q.size[1] )
#画像サイズでPyxel初期化
pyxel.init( SCREEN_WIDTH, SCREEN_HEIGHT )
#255色減色のカラーデータファイル(my_resource.pyxpal)を作成
with open('./my_resource.pyxpal', mode='w') as f:
for _col in range(len(_palette)):
f.write( hex(_palette[_col][0]*0x10000 + _palette[_col][1]*0x100 + _palette[_col][2]).removeprefix('0x')+'\n' )
#255色カラーデータを読み込むためリソースロード
pyxel.load("my_resource.pyxres")
#画面クリア
pyxel.cls(0)
#描画するため画像データをセット
_scrptr = pyxel.screen.data_ptr()
for _yp in range(SCREEN_HEIGHT):
_s = _yp * SCREEN_WIDTH
_e = _s + SCREEN_WIDTH
_scrptr[ _s : _e ] = _pix[_yp]
#描画(Esc待ち)
pyxel.show()
追記0924:
上記応用してスライドショーを作成しました。
ソースのあるカレントフォルダにimgフォルダを作成し、その下にある画像ファイルを順次表示させます。詳細はソース内最初の方のコメントを参照してください。
from PIL import Image
import numpy as np
import pyxel
import os
#----------------------------------------------------
#★前準備:
#あらかじめ「pyxel edit」を実行してエディタを起動&保存して
#空のファイル:my_resource.pyxresを作成しておく
#(255色カラーパレットファイル作成用)
#★前準備2
#画像ファイルは、imgフォルダに格納してください
#★表示サイズや読み込みフォルダを変更したいときは下記のコードを変更してください
#※表示できないファイルはコンソールにエラー内容表示してとばします
#----------------------------------------------------
#Pillow対応画像形式(ただし必ず表示できるわけではない)
#参考:https://pillow.readthedocs.io/en/latest/handbook/image-file-formats.html
#[Full Support]
# BLP, BMP, DDS, DIB, EPS, GIF, ICNS, ICO, IM, JPEG, JPEG2000, MSP,
# PCX, PFM, PNG, APNG, PPM, SGI, SPIDER, TGA, TIFF, WebP, XBM,
#[Read-only]
# CUR, DCX, FITS, FLI, FLC, FPX, FTEX, GBR, GD, IMT, IPTC/NAA,
# MCIDAS, MIC, MPO, PCD, PIXAR, PSD, QOI, SUN, WAL, WMF, EMF, XPM
#[Write-only]
# PALM, PDF, XV Thumbnails
#[Identify-only]
# BUFR, GRIB, HDF5, MPEG
#----------------------------------------------------
#表示画像サイズ
SCREEN_WIDTH = 640
SCREEN_HEIGHT = 480
#画像フォルダ指定
_folder_path = './img'
#----------------------------------------------------
#画像表示 def disp_image(filename):
try:
im = Image.open(filename)
except ValueError:
disp_error(filename + ": Image.open ERROR[ValueError]")
return
except OSError:
disp_error(filename + ": Image.open ERROR[OSError]")
return
except PIL.UnidentifiedImageError:
disp_error(filename + ": Image.open ERROR[PIL.UnidentifiedImageError]")
return
except:
disp_error(filename + ": Image.open ERROR[Other Error]")
return
#----------------------------------------------------
#画像サイズ変更
try:
resize_im = im.resize(( 640, 480 ))
except ValueError:
disp_error(filename + ": resize ERROR[ValueError]")
return
except OSError:
disp_error(filename + ": resize ERROR[OSError]")
return
except:
disp_error(filename + ": resize ERROR[Other Error]")
return
#----------------------------------------------------
#減色
try:
im_q = resize_im.quantize(colors=255, method=0, kmeans=100, dither=1)
except ValueError:
disp_error(filename + ": resize_im.quantize ERROR[ValueError]")
return
except OSError:
disp_error(filename + ": resize_im.quantize ERROR[OSError]")
return
except:
disp_error(filename + ": resize_im.quantize ERROR[Other Error]")
return
#----------------------------------------------------
#画像からパレットを取得してRGBの順になるように3つずつのlistに格納する
_palette = list(zip(*[iter(im_q.getpalette())]*3))
#----------------------------------------------------
#画像データ抽出して表示
_pix = np.asarray(im_q)
with open('./my_resource.pyxpal', mode='w') as f:
for _col in range(len(_palette)):
f.write( hex(_palette[_col][0]*0x10000 + _palette[_col][1]*0x100 + _palette[_col][2]).removeprefix('0x')+'\n' )
#255色カラーデータを読み込むためリソース読み込み
pyxel.load("my_resource.pyxres")
#print( "disp_image: ",filename )
#画面クリア
pyxel.cls(0)
#描画
_scrptr = pyxel.screen.data_ptr()
for _yp in range(SCREEN_HEIGHT):
_s = _yp * SCREEN_WIDTH
_e = _s + SCREEN_WIDTH
_scrptr[ _s : _e ] = _pix[_yp]
#キー入力待ち
#----------------------------------
def disp_error(_str):
global _flag
#エラー画像はとばす
_flag = 2
print(_str)
#----------------------------------------------------
#左クリックで次の画像へ、右クリックで終了
_flag = 0
_count = 0
_piclist =[]
def update():
global _flag
global _count
global _piclist
if( _flag == 1 ):
if( pyxel.btnp(pyxel.MOUSE_BUTTON_LEFT) ):
_flag = 0
_count = _count + 1
if( _count >= len( _piclist ) ):
_count = 0
elif( pyxel.btnp(pyxel.MOUSE_BUTTON_RIGHT) ):
pyxel.quit()
#Error発生時はすぐに次へ
elif( _flag == 2 ):
_flag = 0
_count = _count + 1
if( _count >= len( _piclist ) ):
_count = 0
#----------------------------------
def draw():
global _flag
global _count
global _piclist
if( _flag == 0 ):
_flag = 1
disp_image( _folder_path + '/' + _piclist[_count].rstrip() )
#----------------------------------------------------
pyxel.init( SCREEN_WIDTH, SCREEN_HEIGHT )
try:
_piclist = os.listdir(_folder_path)
except FileNotFoundError:
print("イメージフォルダ",_folder_path,"が無いので終了")
pyxel.quit()
except:
pyxel.quit()
#初期化
_flag = 0
_count = 0
#実行
pyxel.run(update, draw)
なお、ここから有料エリアとしていますが、特に何もありません。
記事に関しまして、お気に召しましたら購入頂けると励みになります。
ここから先は
0字
¥ 100
この記事が気に入ったらサポートをしてみませんか?