お知らせ
GIF
キャラクターについて書いていこうと思いますが、aidiaryさんのコードを参考にした方が早いかもしれませんね。
人工知能に関する断創録
ぼくのコードはaidiaryさんのコードをちょっと短くしただけです。
キャラクターの描画
ここをクリックしてください
まずはキャラクターの画像をダウンロードしましょう。
キャラチップ.zipを選びましょう。
ぴぽや倉庫 ここからダウンロード
トリミングします。と書きましたが、実はしなくてもいいです。
screen.blit(player,(64,64))のところを
screen.blit(player,(64,64),(0,0,32,32))と書くとそこだけ表示されます。
(0,0,32,32)
(キャラ画像の横座標、キャラ画像の縦座標、横の大きさ、縦の大きさ)
失礼しました。
このコードを使用する際は、rを"パス"の前につけましょう。
つけないと、
FileNotFoundError
というエラーが出て、ファイルが正しく読み込まれません。
import pygame
import sys
from pygame.locals import *
pygame.init()
screen = pygame.display.set_mode((160, 160))
pygame.display.set_caption('chara only')
# ここにキャラクター画像ののパスを貼り付ける
player = pygame.image.load(パス)
while (1):
screen.fill((0, 0, 0))
screen.blit(player,(64,64),(0,0,32,32))
pygame.display.update()
for event in pygame.event.get():
if event.type == QUIT:
pygame.quit()
sys.exit()
トリミング
いらなくなってしまいましたが、何か用途があったら使ってください。
import cv2
import numpy as np
# 背景色を指定(透明)
background_color = (0, 0, 0, 0)
# 元画像のパスを指定
input_img = cv2.imread(ここにパスを貼り付ける,cv2.IMREAD_UNCHANGED)
# トリミングする範囲を指定
output_img = input_img[0:32,0:32]
# 背景色を設定
bg_img = np.full((32, 32, 4), background_color, dtype=np.uint8)
bg_img[0:output_img.shape[0], 0:output_img.shape[1], :3] = output_img[:, :, :3]
bg_img[0:output_img.shape[0], 0:output_img.shape[1], 3] = output_img[:, :, 3]
# ファイル名
filename = 'image.png'
# トリミングした画像を新しいファイルに保存
cv2.imwrite(filename, output_img)
print("完了しました")
TypeError: 'NoneType' object is not subscriptable
あるいは
FileNotFoundError
というエラーが出たら、ファイルが正しく読み込まれてません。
フォルダー名が日本語なら英語に変えましょう。”新しいフォルダー”という名前もダメです。
さらに、rを引用符の前につけましょう。
キャラクターの移動
ここをクリックしてください
import pygame
from pygame.locals import *
import sys
pygame.init()
screen = pygame.display.set_mode((160,160))
pygame.display.set_caption('chara_move')
x,y = 2,2
GS = 32
#ここにキャラのパスを貼り付ける
player = pygame.image.load(パス)
while (1):
screen.fill((0,0,0))
screen.blit(player, (x*GS, y*GS), (0, 0, GS, GS))
pygame.display.update()
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit()
sys.exit()
elif event.type==KEYDOWN:
if event.key==K_DOWN:
y += 1
elif event.key==K_UP:
y -= 1
elif event.key==K_RIGHT:
x += 1
elif event.key==K_LEFT:
x -= 1
x,y = 0,0
マス単位の座標です。
ここを変えることでキャラクターの初期位置を変えることができます。
GS = 32
マスの大きさです。
screen.blit(player, (x*GS, y*GS))
マス単位の座標にマスの大きさをかけています。
x,y = 0, 0
(0*32, 0*32) つまり (0, 0)
キャラクターの描画では(64, 64)でしたね。これをここと同じように書くと
x,y = 2, 2
(2*32, 2*32) つまり (64, 64)
図で位置を表してみました。赤が(0, 0) 青が(64, 64)
if event.key==K_DOWN:
何かキーが押された時。
elif event.type==KEYDOWN:
ダウンキーが押されたとき。
y += 1
yを変更しています。
x,y = 0, 0 だと x,y = 0, 1 に変更。
screen.blit(player, (x*GS, y*GS))だから
(0*GS, 0*GS) だと (0*GS, 1*GS) に変更。
ピクセル単位の座標にすると (0, 0) が (0, 32) になります。
キャラアニメ
ここをクリックしてください
ここからキャラクターチップをダウンロードできます。
でもちょっと足りないので編集しましょう。
以下のコードで足りない画像を追加できます。
上の画像が下の画像のようになります。
エラーが出たら、フォルダ名が日本語になってないか、ファイル名が数字で始まってないか確認しましょう。r文字も忘れずつけましょう。r"ファイルのパス"
import cv2
import numpy as np
# 画像を読み込む(アルファチャンネルを含む)
image = cv2.imread(ここにキャラクター画像のパスを貼り付ける, cv2.IMREAD_UNCHANGED)
# 画像が正しく読み込まれたか確認
if image is None:
print("画像が読み込まれませんでした。ファイルパスを確認してください。")
else:
# コピーする部分の座標とサイズ
x, y, w, h = 32, 0, 32, 128
part = image[y:y+h, x:x+w]
# 元の画像の右端に追加するために、新しい画像を作成
new_width = image.shape[1] + w
new_image = np.zeros((image.shape[0], new_width, 4), dtype=np.uint8)
# 元の画像を新しい画像にコピー
if image.shape[2] == 4:
new_image[:, :image.shape[1], :] = image
else:
new_image[:, :image.shape[1], :3] = image
new_image[:, :image.shape[1], 3] = 255 # アルファチャンネルを追加
# コピーした部分を新しい画像の右端に追加
if part.shape[2] == 4:
new_image[:, image.shape[1]:, :] = part
else:
new_image[:, image.shape[1]:, :3] = part
new_image[:, image.shape[1]:, 3] = 255 # アルファチャンネルを追加
# 結果を保存
cv2.imwrite('edited_image.png', new_image)
print("完了しました")
フォルダ内のすべてのファイルに適用するコードもあります。
すべてのファイルに適用 クリックして移動
@@@@@@@@@@@@@@@@@@@@@@@@@@
キャラが、その場足踏みしているように見えるコードです。
aidiaryさんのコードを短くしたコードです。
import pygame
from pygame.locals import *
import sys
def split_image(image):
"""32x128のキャラクターイメージを32x32の4枚のイメージに分割
分割したイメージを格納したリストを返す"""
imageList = []
for i in range(0, 128, GS):
surface = pygame.Surface((GS,GS))
surface.blit(image, (0,0), (i,0,GS,GS))
surface.set_colorkey(surface.get_at((0,0)), RLEACCEL)
surface.convert()
imageList.append(surface)
return imageList
pygame.init()
screen = pygame.display.set_mode((160,160))
pygame.display.set_caption('chara anime')
x,y = 2,2
GS = 32
#ここにキャラのパスを貼り付ける
playerImgList = split_image(pygame.image.load(パス))
# アニメーション速度
animcycle = 24
frame = 0
clock = pygame.time.Clock()
while True:
screen.fill((255,255,255))
clock.tick(60)
# 経過フレーム数に応じて表示する画像を変える
frame += 1
player = playerImgList[int(frame/animcycle%4)]
screen.blit(player, (x*GS, y*GS),(0,0,GS,GS))
pygame.display.update()
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit()
sys.exit()
elif event.type==KEYDOWN:
if event.key==K_DOWN:
y += 1
elif event.key==K_UP:
y -= 1
elif event.key==K_RIGHT:
x += 1
elif event.key==K_LEFT:
x -= 1
animcycle = 24
ずっと24のままです。
frame = 0 と frame += 1
数字が一つずつ増えていきます。frameと書いてあると、なじみのない人だとわかりにくく思えますが、frame が a でも同じです。数字が一つずつ増えていくことには変わりありません。
player = playerImgList[frame//animcycle%4)]
キャラクター画像はリストに入っている。
計算式でインデックスを決めている。
playerImgList[0]
playerImgList[1]
playerImgList[2]
playerImgList[3]
%の計算
a = 0//3%3
b = 1//3%3
c = 2//3%3
d = 3//3%3
e = 4//3%3
f = 5//3%3
g = 6//3%3
h = 7//3%3
i = 8//3%3
j = 9//3%3
k = 10//3%3
l = 11//3%3
print(a,b,c,d,e,f,g,h,i,j,k,l)
このコードをちょっと動かしてみてください。
結果は思った通りでしたか?思った通りならこのまま進んで問題ないです。
思ったのとは違ったのなら、立ち止まって考えてみましょう。
%の計算
a = 0//3%3 0割る3で0、0%3余り0
b = 1//3%3
c = 2//3%3
d = 3//3%3 3割る3で1、1%3余り1
e = 4//3%3 4割る3で1.33……整数除算で小数点以下が切り捨てられ1、1%3余り1
f = 5//3%3
g = 6//3%3
h = 7//3%3
i = 8//3%3
j = 9//3%3 9割る3で3、3%3割り切れるので余り0
k = 10//3%3
l = 11//3%3
向きを変えながら歩く
ここをクリックしてください
import pygame
from pygame.locals import *
import sys
def split_image(image):
"""32x128のキャラクターイメージを32x32の4枚のイメージに分割
分割したイメージを格納したリストを返す"""
imageList = []
for i in range(0, 128, GS):
for j in range(0, 128, GS):
surface = pygame.Surface((GS,GS))
surface.blit(image, (0,0), (j, i,GS,GS))
surface.set_colorkey(surface.get_at((0,0)), RLEACCEL)
surface.convert()
imageList.append(surface)
return imageList
pygame.init()
screen = pygame.display.set_mode((160,160))
pygame.display.set_caption('walking')
x,y = 2,2
GS = 32
DOWN,LEFT,RIGHT,UP = 0,1,2,3
#ここキャラのパスを貼り付ける
playerImgList = split_image(pygame.image.load(パス)) # プレイヤーx,y = 1,1
direction = DOWN
animcycle = 24 # アニメーション速度
frame = 0
clock = pygame.time.Clock()
while True:
clock.tick(60)
screen.fill((255,255,255))
# 経過フレーム数に応じて表示する画像を変える
frame += 1
player = playerImgList[int(direction*4+frame/animcycle%4)]
screen.blit(player, (x*GS, y*GS))
pygame.display.update()
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit()
sys.exit()
elif event.type==KEYDOWN:
if event.key==K_DOWN:
direction = DOWN
y += 1
elif event.key==K_UP:
direction = UP
y -= 1
elif event.key==K_RIGHT:
direction = RIGHT
x += 1
elif event.key==K_LEFT:
direction = LEFT
x -= 1
def split_image(image):の説明
概要
キャラクター画像を16個に分割してリストに入れる。
補足
キャラクター画像の大きさは128x128
GSは32
def split_image(image):
imageはキャラクター画像。
imageList = []
空のリストを作る。
for i in range(0, 128, GS):
0から128の範囲でGS(32)間隔の数字を作る。
0 32 64 96
for j in range(0, 128, GS):
0から128の範囲でGS(32)間隔の数字を作る。
0 32 64 96
surface = pygame.Surface((GS,GS))
サーフェイスを作る。
たとえるなら、ペイントのキャンパスのようなもの。
色はデフォルトで、黒。
((GS,GS))は大きさ。
surface.blit(image, (0,0), (j, i,GS,GS))
作ったサーフェイスに画像の一部を貼り付ける。
imageはキャラクター画像。
(0,0)はサーフェイスの座標。
j,iはキャラクター画像の座標。
GS,GSは切り取る大きさ。
surface.set_colorkey(surface.get_at((0,0)), RLEACCEL)
.set_colorkeyは指定した色を透明にする。
.get_atは指定した座標の色を取得する。
ここでは(0,0)の色は黒なので黒がすべて透明になる。
ぼくは最初背景が白の画像を使っていたので、マントが透明になってました。
RLEACCELは描画を高速化する。
surface.convert()
サーフェイスを最適化する。
imageList.append(surface)
リストに追加する。
return imageList
結果を返します。
おまけ
set_alpha: サーフェス全体の透明度を設定します。透明度は0から255までの範囲で指定し、サーフェス全体が均一に透明になります。
set_colorkey: 特定の色を透明に設定します。サーフェス上の指定した色が透明になります。
スクロール
ここをクリックしてください
キャラクターの移動処理にはスクロールというものがあるのですが、ぼくは理解してないので紹介だけしておきます。
(aidiaryさんの記事です)
タイルスクロール
変更箇所(すべて変更済みのコードです)
pygame.quit()をつけてます。
for event in pygame.event.get():
if event.type == QUIT:
pygame.quit()#ここ
sys.exit()
()をつけたり、isを==に変えたりしてます。
def load_image(filename, colorkey=None):
filename = os.path.join("data", filename)
try:
image = pygame.image.load(filename)
except (pygame.error, message):#ここ
print ("Cannot load image:"), filename#ここ
raise (SystemExit, message)#ここ
image = image.convert()
if colorkey is not None:
if colorkey == -1:#ここ
colorkey = image.get_at((0,0))
image.set_colorkey(colorkey, RLEACCEL)
return image
int()をつけてます。
def draw(self, screen, offset):
"""マップを描画する"""
offsetx, offsety = offset
# マップの描画範囲を計算
startx = int(offsetx / GS)#ここ
endx = int(startx + SCR_RECT.width/GS + 1)#ここ
starty = int(offsety / GS)#ここ
endy = int(starty + SCR_RECT.height/GS + 1)#ここ
/ を // にします。
def update(self):
# キャラクターアニメーション(frameに応じて描画イメージを切り替える)
self.frame += 1
self.image = self.images[self.direction*4+self.frame//self.animcycle%4]#ここ
(aidiaryさんの記事です)
ピクセルスクロール
変更箇所。(変更済みのコードです)
/ を // にしてます。
# プレイヤーの移動処理
if self.moving == True:
# ピクセル移動中ならマスにきっちり収まるまで移動を続ける
self.rect.move_ip(self.vx, self.vy)
if self.rect.left % GS == 0 and self.rect.top % GS == 0: # マスにおさまったら移動完了
self.moving = False
self.x = self.rect.left // GS#ここ
self.y = self.rect.top // GS#ここ
これ以外はタイルベーススクロールと同じところを変更してください。
マウス移動(白背景)
左クリックでキャラクターが移動します。
ここをクリックしてください
import pygame
import sys
import struct
import os
from pygame.locals import *
pygame.init()
screen=pygame.display.set_mode((480,480))
pygame.display.set_caption("マウス移動")
GS = 32
x,y = 7,7
vx,vy = 0,0
mx,my = 7,7
DOWN,LEFT,RIGHT,UP = 0,1,2,3
direction = DOWN
def main():
global game_state,chara,x,y
# キャラクターチップをロード
Player.images["player"] = split_image(pygame.image.load(ここにキャラのパスを貼り付ける))
player = Player("player")
clock = pygame.time.Clock()
running = True
while running:
clock.tick(60)
screen.fill((255,255,255))
player.draw()
player.update()
pygame.display.update()
for event in pygame.event.get():
if event.type==QUIT:
pygame.quit()
sys.exit()
pygame.display.update()
def split_image(image):
"""128x128のキャラクターイメージを32x32の16枚のイメージに分割
分割したイメージを格納したリストを返す"""
imageList = []
for i in range(0, 128, GS):
for j in range(0, 128, GS):
surface = pygame.Surface((GS,GS))
surface.blit(image, (0,0), (j,i,GS,GS))
surface.set_colorkey(surface.get_at((0,0)), RLEACCEL)
surface.convert()
imageList.append(surface)
return imageList
class Player:
speed = 4 # 1フレームの移動ピクセル数
animcycle = 24 # アニメーション速度
frame = 0
images = {}
def __init__(self, name):
self.name = name # プレイヤー名(ファイル名と同じ)
self.image = self.images[name][0] # 描画中のイメージ
self.rect = self.image.get_rect(topleft=(x*GS, y*GS))
self.moving = False # 移動中か?
def update(self):
global x,y,vx,vy,direction,mx,my
"""プレイヤー状態を更新する。"""
# プレイヤーの移動処理
if self.moving == True:
# ピクセル移動中ならマスにきっちり収まるまで移動を続ける
self.rect.move_ip(vx, vy)
if self.rect.left % GS == 0 and self.rect.top % GS == 0: # マスにおさまったら移動完了
self.moving = False
x = int(self.rect.left / GS)
y = int(self.rect.top / GS)
else:
# 入力があったら移動を開始する(速度をセットする)
if y < my:
direction = DOWN
vx, vy = 0, self.speed
self.moving = True
elif y > my:
direction = UP
vx, vy = 0, -self.speed
self.moving = True
elif x > mx:
direction = LEFT
vx, vy = -self.speed, 0
self.moving = True
elif x < mx:
direction = RIGHT
vx, vy = self.speed, 0
self.moving = True
mouse_pressed = pygame.mouse.get_pressed()
if mouse_pressed[0]:
mouse_pos = pygame.mouse.get_pos()
mx = mouse_pos[0]
my = mouse_pos[1]
mx = mx // GS
my = my // GS
# キャラクターアニメーション(frameに応じて描画イメージを切り替える)
self.frame += 1
self.image = self.images[self.name][int(direction*4+self.frame/self.animcycle%4)]
def draw(self):
px = self.rect.topleft[0]
py = self.rect.topleft[1]
screen.blit(self.image, (px,py))
if __name__ == "__main__":
main()
Player.images["player"] = split_image(pygame.image.load(ここにパスを貼り付け))
[""]にはファイル名を書きます。
player = Player("player")
("")にもファイル名を書きます。
マウス移動(マップ)
ここをクリックしてください
aidiaryさんのフォルダをお借りすると速やかに実行できるかと思います。いつもaidiaryさん頼りで申し訳ないです。(汗)
import pygame
from pygame import *
import sys
import struct
import os
pygame.init()
screen=pygame.display.set_mode((640,480))
x,y = 1,1
vx,vy = 0,0
mx,my = 1,1
nx,ny = 0,0
DOWN,LEFT,RIGHT,UP = 0,1,2,3
direction = DOWN
TRANS_COLOR = (190,179,145) # マップチップの透明色
while(True):
screen.fill((0,0,0))
pygame.display.update()
SCR_RECT = Rect(0, 0, 640, 480)
GS = 32
DOWN,LEFT,RIGHT,UP = 0,1,2,3
def main():
global game_state,chara,x,y
pygame.init()
screen = pygame.display.set_mode(SCR_RECT.size)
pygame.display.set_caption("マウス移動")
font = pygame.font.Font(フォントのパス, 30)
# キャラクターチップをロード
load_charachips("data", "charachip.dat")
# マップチップをロード
load_mapchips("data", "mapchip.dat")
# マップとプレイヤーを作成
map = Map("field")
player = Player("swordman_female")
num1 = str(x)
num2 = str(y)
clock = pygame.time.Clock()
running = True
while running:
clock.tick(60)
offset = calc_offset(player)
map.draw(screen, offset)
player.draw(screen, offset)
player.update(map,offset)
text1 = font.render(num1+" "+num2, True, (255,255,255))
screen.blit(text1, (16, 16))
num1 = str(x)
num2 = str(y)
pygame.display.update()
for event in pygame.event.get():
if event.type == QUIT:
pygame.quit()
sys.exit()
def load_charachips(dir, file):
"""キャラクターチップをロードしてCharacter.imagesに格納"""
file = os.path.join(dir, file)
fp = open(file, "r")
for line in fp:
line = line.rstrip()
data = line.split(",")
chara_id = int(data[0])
chara_name = data[1]
Player.images[chara_name] = split_image(load_image("charachip", "%s.png" % chara_name))
fp.close()
def load_mapchips(dir, file):
"""マップチップをロードしてMap.imagesに格納"""
file = os.path.join(dir, file)
fp = open(file, "r")
for line in fp:
line = line.rstrip()
data = line.split(",")
mapchip_id = int(data[0])
mapchip_name = data[1]
movable = int(data[2]) # 移動可能か?
transparent = int(data[3]) # 背景を透明にするか?
if transparent == 0:
Map.images.append(load_image("mapchip", "%s.png" % mapchip_name))
else:
Map.images.append(load_image("mapchip", "%s.png" % mapchip_name, TRANS_COLOR))
Map.movable_type.append(movable)
fp.close()
def calc_offset(player):
"""オフセットを計算する"""
offsetx = player.rect.topleft[0] - SCR_RECT.width/2
offsety = player.rect.topleft[1] - SCR_RECT.height/2
return offsetx, offsety
def load_image(dir, file, colorkey=None):
file = os.path.join(dir, file)
try:
image = pygame.image.load(file)
except (pygame.error, message):
print ("Cannot load image:"), file
raise (SystemExit, message)
image = image.convert()
if colorkey is not None:
if colorkey == -1:
colorkey = image.get_at((0,0))
image.set_colorkey(colorkey, RLEACCEL)
return image
def split_image(image):
"""128x128のキャラクターイメージを32x32の16枚のイメージに分割
分割したイメージを格納したリストを返す"""
imageList = []
for i in range(0, 128, GS):
for j in range(0, 128, GS):
surface = pygame.Surface((GS,GS))
surface.blit(image, (0,0), (j,i,GS,GS))
surface.set_colorkey(surface.get_at((0,0)), RLEACCEL)
surface.convert()
imageList.append(surface)
return imageList
class Map:
# main()のload_mapchips()でセットされる
images = [] # マップチップ(ID->イメージ)
movable_type = [] # マップチップが移動可能か?(0:移動不可, 1:移動可)
def __init__(self, name):
self.name = name
self.row = -1 # 行数
self.col = -1 # 列数
self.map = [] # マップデータ(2次元リスト)
self.charas = [] # マップにいるキャラクターリスト#追加した
self.load()
def draw(self, screen, offset):
"""マップを描画する"""
offsetx, offsety = offset
# マップの描画範囲を計算
startx = int(offsetx / GS)
endx = int(startx + SCR_RECT.width/GS + 1)
starty = int(offsety / GS)
endy = int(starty + SCR_RECT.height/GS + 1)
# マップの描画
for y in range(starty, endy):
for x in range(startx, endx):
# マップの範囲外はデフォルトイメージで描画
# この条件がないとマップの端に行くとエラー発生
if x < 0 or y < 0 or x > self.col-1 or y > self.row-1:
screen.blit(self.images[self.default], (x*GS-offsetx,y*GS-offsety))
else:
screen.blit(self.images[self.map[y][x]], (x*GS-offsetx,y*GS-offsety))
def is_movable(self, x, y):
"""(x,y)は移動可能か?"""
# マップ範囲内か?
if x < 0 or x > self.col-1 or y < 0 or y > self.row-1:
return False
# マップチップは移動可能か?
if self.movable_type[self.map[y][x]] == 0:
return False
# キャラクターと衝突しないか?#追加した
for chara in self.charas:
if chara.x == x and chara.y == y:
return False
return True
def load(self):
"""バイナリファイルからマップをロード"""
file = os.path.join("data", self.name + ".map")
fp = open(file, "rb")
# unpack()はタプルが返されるので[0]だけ抽出
self.row = struct.unpack("i", fp.read(struct.calcsize("i")))[0] # 行数
self.col = struct.unpack("i", fp.read(struct.calcsize("i")))[0] # 列数
self.default = struct.unpack("B", fp.read(struct.calcsize("B")))[0] # デフォルトマップチップ
# マップ
self.map = [[4 for c in range(self.col)] for r in range(self.row)]
for r in range(self.row):
for c in range(self.col):
self.map[r][c] = struct.unpack("B", fp.read(struct.calcsize("B")))[0]
fp.close()
class Player:
speed = 4 # 1フレームの移動ピクセル数
animcycle = 24 # アニメーション速度
frame = 0
images = {}
def __init__(self, name):
self.name = name # プレイヤー名(ファイル名と同じ)
self.image = self.images[name][0] # 描画中のイメージ
self.rect = self.image.get_rect(topleft=(x*GS, y*GS))
self.moving = False # 移動中か?
def update(self, map,offset):
global x,y,vx,vy,direction,mx,my
offsetx, offsety = offset
px = self.rect.topleft[0]
py = self.rect.topleft[1]
"""プレイヤー状態を更新する。
mapは移動可能かの判定に必要。"""
# プレイヤーの移動処理
if self.moving == True:
# ピクセル移動中ならマスにきっちり収まるまで移動を続ける
self.rect.move_ip(vx, vy)
if self.rect.left % GS == 0 and self.rect.top % GS == 0: # マスにおさまったら移動完了
self.moving = False
x = int(self.rect.left / GS)
y = int(self.rect.top / GS)
else:
# キー入力があったら移動を開始する(速度をセットする)
if y < my:
direction = DOWN # 移動できるかに関係なく向きは変える
if map.is_movable(x, y+1):
vx, vy = 0, self.speed
self.moving = True
elif x > mx:
direction = LEFT
if map.is_movable(x-1, y):
vx, vy = -self.speed, 0
self.moving = True
elif x < mx:
direction = RIGHT
if map.is_movable(x+1, y):
vx, vy = self.speed, 0
self.moving = True
elif y > my:
direction = UP
if map.is_movable(x, y-1):
vx, vy = 0, -self.speed
self.moving = True
mouse_pressed = pygame.mouse.get_pressed()
if mouse_pressed[0]:
mouse_pos = pygame.mouse.get_pos()
mx = mouse_pos[0] +offsetx
my = mouse_pos[1] +offsety
mx = mx // GS
my = my // GS
# キャラクターアニメーション(frameに応じて描画イメージを切り替える)
self.frame += 1
self.image = self.images[self.name][int(direction*4+self.frame/self.animcycle%4)]
def draw(self, screen, offset):
"""オフセットを考慮してプレイヤーを描画"""
offsetx, offsety = offset
px = self.rect.topleft[0]
py = self.rect.topleft[1]
screen.blit(self.image, (px-offsetx, py-offsety))#ないと透明になる
if __name__ == "__main__":
main()
for event in pygame.event.get():
if event.type==QUIT:
pygame.quit()
sys.exit()
セーブとロード
ここをクリックしてください
Sキーを押すと、座標を保存します。別の場所に移動して、Lキーを押してみてください。座標を保存した場所に戻ります。
import pygame
from pygame.locals import *
import sys
import pickle
pygame.init()
screen = pygame.display.set_mode((160,160))
pygame.display.set_caption('save load')
x,y = 2,2
GS = 32
#ここにキャラクター画像のパスを貼り付ける
player = pygame.image.load(パス)
while (1):
screen.fill((0,0,0))
screen.blit(player, (x*GS, y*GS))
pygame.display.update()
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit()
sys.exit()
elif event.type==KEYDOWN:
if event.key==K_DOWN:
y += 1
elif event.key==K_UP:
y -= 1
elif event.key==K_RIGHT:
x += 1
elif event.key==K_LEFT:
x -= 1
elif event.key==K_s:
# ゲームの状態を辞書として定義
save_data = {
"x": x,"y": y
}
# ゲームの状態をファイルに保存
with open('save_data.pickle', 'wb') as file:
pickle.dump(save_data, file)
print("セーブしました")
elif event.key==K_l:
# セーブデータを読み込む
with open('save_data.pickle', 'rb') as file:
save_data = pickle.load(file)
x = save_data["x"]
y = save_data["y"]
print("ロードしました")
loaded_save_data = pickle.load(file)
ここをクリックしてください