python 实现简单的飞机大战
0 引言
1 环境
操作系统:windows10
Python版本:3.7.3
引用的模块:os,sys,random,pygame
因为我们实现代码使用到了一个pygame的第三方模块,没有的大家可以先pip install 一下,这里顺便提供一个比较好的pygame的教程.
https://eyehere.net/2011/python-pygame-novice-professional-index/
2 需求分析
3 代码实现
3.1 首先我们先指定素材文件的文件目录.方便我们后面的使用
import os
# 得到当前文件夹下面的material_images目录的路径
source_dir = os.path.join(os.path.dirname(os.path.abspath(__file__)), 'material_images')
3.2 实现一个Game类,用来完成这个游戏的主要逻辑。
import pygame
class Game():
def __init__(self, background_image_path, size=(480, 700), title='飞机大战', font_name='SIMYOU.TTF', font_size=30, speed=2000):
'''
:param background_image_path: 背景图片的路径地址
:param size: 游戏窗口的大小
:param title: 游戏窗口的标题
:param font_name: 指定字体
:param font_size: 指定字体大小
:param speed: 背景图滚动整个窗口一次所用时间,单位为ms
'''
self.size = size
self.screen = pygame.display.set_mode(size)
self.title = title
self.background_image_path = background_image_path
self.background = pygame.image.load(self.background_image_path).convert()
# 设置字体对象,得到系统中自带的字体
self.font = pygame.font.SysFont(font_name, font_size)
# 得到Clock对象,我们可以使用它来获取距离上次绘制图像的时间
self.clock = pygame.time.Clock()
# 背景图初始位置
self.height = 0
# 使用窗口的高度处于滚动的时间,就能得到每ms滚动的距离
self.every_ms_move_distance = self.size[1] / speed # 2秒
# 分数
self.score = 0
# 存放所有的敌机
self.enemies = []
def show_score(self):
'''
显示分数, 在窗口的的最上方距离上边距10px, 左右居中
'''
# 是否开启锯齿模式 字体颜色 背景颜色
score = self.font.render(f'得分 : {self.score} ', True, (0,0,0), (255,255,255))
score_position_x = (self.size[0]-score.get_size()[0]) / 2
score_position_y = 10
self.screen.blit(score, (score_position_x, score_position_y))
def set_time_passed(self):
# 控制画 的帧, 越大越快
# self.clock.tick(1000)
# 得到上一次绘制图像到到现在的时间, ms
self.time_passed = self.clock.tick()
def draw_background(self):
'''
绘制背景图片,一直向下滚动,营造飞机一直往上面飞的感觉
'''
# 每次移动的距离 = 每ms移动的距离 * 上次到现在的时间(ms)
move_distance = self.every_ms_move_distance * self.time_passed
self.height += move_distance
# 如果超出窗口的高度,将height重置为零
if self.height >= self.size[1]:
self.height = 0
# 两张背景图一起显示,营造背景图不间断的一直滚动的效果
self.screen.blit(self.background, (0, -self.size[1] + self.height))
self.screen.blit(self.background, (0, self.height))
def create_enemy(self, image_path=os.path.join(source_dir,'enemy1.png'), enemy_number=5):
'''
创建敌机
:param image_path: 敌机的图片地址
:param enemy_number: 最多有几个敌机在屏幕上
'''
if len(self.enemies) >= enemy_number:
return
enemy = Enemy(image_path=image_path)
self.enemies.append(enemy)
def draw_enemies(self, time_passed, screen):
'''
绘制敌机到屏幕上,清理跑出窗口的敌机,
:param time_passed: 上次绘制导向现在经过的时间
:param screen: 绘制的窗口对象
'''
# 清理跑出范围的敌机
for enemy in self.enemies:
# 当敌机跑出范围时,就将enemy.position[1]设置为-100了,所以我们这里判断enemy.position[1]==-100的,就是跑出范围的敌机
# enemy.destroyed 为True, 表示敌机被子弹击中,也需要清理这个enemy
if enemy.position[1] == -100 or enemy.destroyed:
self.enemies.remove(enemy)
# 更新敌机位置
for enemy in self.enemies:
# 调用每一个敌机的update方法,改变敌机的位置
enemy.update(time_passed)
# 绘制敌机
for enemy in self.enemies:
# 根据敌机的位置,绘制敌机
screen.blit(enemy.image, enemy.position)
def bullet_and_enemy_crash_detection(self, bullets):
'''
检测子弹是否击中敌机
:param bullets: 飞机的所有子弹
'''
for bullet in bullets:
# 遍历每一个子弹
for enemy in self.enemies:
# 遍历每一个敌机,判断是否被击中
if bullet.position[0] >= enemy.position[0] and bullet.position[0] <= enemy.position[0]+enemy.image.get_size()[0]:
if bullet.position[1] >= enemy.position[1] and bullet.position[1] <= enemy.position[1] + enemy.image.get_size()[1]:
# 如果被击中,敌机开始自毁
enemy.start_destroy = True
# 子弹自毁,消失
bullet.destroyed = True
# 分数加一
self.score += 1
def plan_and_enemy_crash_detection(self, plan, allow_crash_size=None):
'''
检测敌机与飞机是否相撞
:param plan: 飞机对象
:param allow_crash_size: 允许飞机碰撞的大小,只有左右有效
'''
# 如果没有传入这个参数,赋值为飞机宽度的 10%
if allow_crash_size is None:
allow_crash_size = 0.1 * plan.image_size[0]
for enemy in self.enemies:
# 遍历每一个敌机, 检测是否碰撞
if enemy.position[0]+enemy.image.get_size()[0] - allow_crash_size >= plan.position[0] and enemy.position[0] <= plan.position[0]+plan.image.get_size()[0] - allow_crash_size:
if enemy.position[1] + enemy.image.get_size()[1] >= plan.position[1] and enemy.position[1] <= plan.position[1] + plan.image.get_size()[1]:
# 检测到碰撞,飞机开始自毁
plan.start_destroy = True
def draw_plan(self, plan, time_passed):
'''
绘制飞机
:param plan: 飞机对象
:param time_passed: 距离上次绘制的时间
:return:
'''
# 如果飞机开始自毁,调用自毁函数,显示自毁动画
if plan.start_destroy:
plan.show_destroy_animation(time_passed, destroy_time=1000)
self.screen.blit(plan.image, plan.position)
def game_over(self):
'''
游戏结束
'''
while True:
# 绘制背景图
self.set_time_passed()
self.draw_background()
text = self.font.render(f'游戏结束,得分 : {self.score} ', True, (0, 0, 0), (255, 255, 255))
text_position_x = (self.size[0] - text.get_size()[0]) / 2
text_position_y = (self.size[1] - text.get_size()[1]) / 2
self.screen.blit(text, (text_position_x, text_position_y))
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit()
sys.exit()
elif event.type == pygame.KEYDOWN:
if event.key == pygame.K_ESCAPE:
pygame.quit()
sys.exit()
pygame.display.update()
def run(self):
'''
游戏入口函数,开始函数,主体函数
:return:
'''
# 设置游戏窗口的大小
pygame.display.set_caption(self.title)
# 初始化一个飞机对象
plan = Plan()
while True:
# 如果飞机自毁完成, 游戏结束, 调用game_over函数
if plan.destroyed:
self.game_over()
break
# 检测监听事件
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit()
sys.exit()
elif event.type == pygame.KEYDOWN:
if event.key == pygame.K_ESCAPE:
pygame.quit()
sys.exit()
# 如果用户的空格键弹起,说明用户按下过空格键,发射一颗子弹
elif event.type == pygame.KEYUP:
if event.key == pygame.K_SPACE:
# 调用plan.shut函数
plan.shut()
# 检测上下左右的移动案件.
# w,a,s,d 和 上,下,左,右键都可以
# 然后执行plan.update函数,改变飞机的位置
key_pressed = pygame.key.get_pressed()
if key_pressed[pygame.K_w] or key_pressed[pygame.K_UP]:
plan.update(direction=Direction.UP)
elif key_pressed[pygame.K_s] or key_pressed[pygame.K_DOWN]:
plan.update(direction=Direction.DOWN)
elif key_pressed[pygame.K_a] or key_pressed[pygame.K_LEFT]:
plan.update(direction=Direction.LEFT)
elif key_pressed[pygame.K_d] or key_pressed[pygame.K_RIGHT]:
plan.update(direction=Direction.RIGHT)
# 子弹和敌机的碰撞检测
self.bullet_and_enemy_crash_detection(plan.bullets)
# 飞机与敌机的碰撞检测
self.plan_and_enemy_crash_detection(plan)
# 设置属性time_passed的值, 距离上次的时间,方便后面使用
self.set_time_passed()
# 绘制背景图片
self.draw_background()
# 显示分数
self.show_score()
# 生成敌机
self.create_enemy()
# 绘制敌机
self.draw_enemies(time_passed=self.time_passed, screen=self.screen)
# 绘制飞机
self.draw_plan(plan=plan, time_passed=self.time_passed)
# 绘制子弹
plan.draw_bullets(time_passed=self.time_passed, screen=self.screen)
# 显示我们的图像
pygame.display.update()
这里说以下怎样查看自己的系统中有哪些自带的字体.
pygame.font.get_fonts(),这个函数就能够得到系统中所有的自带字体文件。不过,当我们游戏中有中文的时候,我们也得选择支持中文的字体,否则的话是显示不出中文的。
3.3 实现DestroyAnimationMixin类,这个类主要是用来显示飞机或敌机的自毁动画
# 显示飞机自毁动画的Mixin类, 可用于飞机和敌机的自毁动画显示
class DestroyAnimationMixin():
def show_destroy_animation(self, time_passed, destroy_time=200):
'''
显示自毁动画
动画其实就是几张图片切换的比较快,我们的眼睛识别不出来,所以认为他是动态的,也就是动画
:param time_passed: 距离上次绘制图像到现在的时间,单位ms
:param destroy_time: 自毁动画总共显示时间,单位ms
'''
# 因为我们的自毁图片有四张,需要依次显示,首先动画的效果
# self.destroy_image_position 表示第几章自毁图片,从零开始
# 如果大于等于4了,说明自毁动画显示完成,设置self.destroyed变量为True, 方便别处调用
if self.destroy_image_position >= 4:
self.destroyed = True
return
# 依次加载自毁图片
if self.time_passed >= destroy_time / 4:
self.image = pygame.image.load(os.path.join(source_dir, self.destroy_images[self.destroy_image_position])).convert_alpha()
self.destroy_image_position += 1
self.time_passed = 0
else:
self.time_passed += time_passed
3.4 实现飞机类,完成飞机的主要操作
# 飞机类,继承DestroyAnimationMixin, 方便使用显示自毁动画的函数
class Plan(DestroyAnimationMixin):
def __init__(self, image_path=os.path.join(source_dir,'plan.png'), background_size=(480, 700)):
'''
:param image_path: 飞机图片地址
:param background_size: 游戏窗口大小
'''
self.background_size = background_size
self.image = pygame.image.load(image_path).convert_alpha()
self.image_size = self.image.get_size()
self.position = [(background_size[0]-self.image_size[0]) / 2, 500]
# 飞机每次移动的距离
self.every_time_move_distance = 0.5
# 飞机的子弹
self.bullets = []
# destroy association attributes, 自毁相关属性
# 开始自毁
self.start_destroy = False
# 自毁结束
self.destroyed = False
# 自毁图片
self.destroy_images = ['me_destroy_1.png', 'me_destroy_2.png', 'me_destroy_3.png', 'me_destroy_4.png']
# 自毁图片位置
self.destroy_image_position = 0
# 距离上次绘制图像到现在的时间
self.time_passed = 0
def update(self, direction):
'''
更新飞机位置
:param direction: 飞机移动方向
'''
if direction == 1:
if self.position[1] <= 0:
return
self.position[1] -= self.every_time_move_distance
elif direction == 2:
if self.position[1] + self.image_size[1] >= self.background_size[1]:
return
self.position[1] += self.every_time_move_distance
elif direction == 3:
if self.position[0] <= 0:
return
self.position[0] -= self.every_time_move_distance
else:
if self.position[0] + self.image_size[0] >= self.background_size[0]:
return
self.position[0] += self.every_time_move_distance
def shut(self, image_path=os.path.join(source_dir,'bullet.png')):
'''
飞机发射子弹
:param image_path: 子弹图片
'''
bullet = Bullet(image_path, plan=self)
self.bullets.append(bullet)
def draw_bullets(self, time_passed, screen):
'''
绘制飞机的所有子弹
:param time_passed: 距离上次绘制图像到现在的时间
:param screen: 绘制到哪一个窗口中
'''
# 清理消失的子弹
for bullet in self.bullets:
if bullet.position[1] == -100:
self.bullets.remove(bullet)
# 更新每个子弹的位置
for bullet in self.bullets:
bullet.update(time_passed)
# 绘制每个子弹
for bullet in self.bullets:
screen.blit(bullet.image, bullet.position)
3.5 实现敌机类,完成敌机的主要操作
# 敌机类,继承DestroyAnimationMixin, 方便使用显示自毁动画的函数
class Enemy(DestroyAnimationMixin):
def __init__(self, image_path=os.path.join(source_dir, 'enemy1.png'), speed=2000, background_size=(480, 700)):
'''
:param image_path: 敌机图片地址
:param speed: 敌机移动整个窗口需要的时间,单位ms,也就是速度
:param background_size: 游戏窗口的尺寸
'''
self.image = pygame.image.load(image_path).convert_alpha()
self.speed = background_size[1] / speed
self.background_size = background_size
self.position = [random.randint(0, background_size[0]-self.image.get_size()[0]), -self.image.get_size()[1]]
# 开始自毁
self.start_destroy = False
# 自毁完成
self.destroyed = False
# 自毁图片路径
self.destroy_images = ['enemy1_down1.png', 'enemy1_down2.png', 'enemy1_down3.png', 'enemy1_down3.png']
# 距离上次绘制图像到现在的时间
self.time_passed = 0
# 自毁图片在self.destroy_images的位置
self.destroy_image_position = 0
def update(self, time_passed):
'''
更新敌机的位置
:param time_passed: 距离上次绘制图像到现在的时间
:return:
'''
# 敌机如果跑出游戏窗口,设置self.position[1] = -100, 方便其他位置使用
if self.position[1] >= self.background_size[1]:
self.position[1] = -100
return
# 如果开始自毁, 调用自毁函数,显示动画,如果没有,改变位置
if self.start_destroy:
self.show_destroy_animation(time_passed)
else:
self.position[1] += self.speed * time_passed
3.6 实现子弹类,完成子弹的主要操作
# 飞机子弹类
class Bullet():
def __init__(self, image_path=os.path.join(source_dir,'bullet.png'), background_size=(480, 700), plan=None, speed=1000):
'''
:param image_path: 子弹的图片地址
:param background_size: 游戏窗口大小
:param plan: 飞机对象
:param speed: 子弹飞行速度
'''
self.image = pygame.image.load(image_path).convert_alpha()
self.background_size = background_size
self.speed = background_size[1] / speed
# 子弹是否击中敌机
self.destroyed = False
self.position = self._get_position(plan)
def _get_position(self, plan):
'''
根据plan得到子弹发出位置
:param plan: 飞机对象
'''
bullet_size = self.image.get_size()
plan_width = plan.image_size[0]
x = (plan_width-bullet_size[0]) / 2
return [plan.position[0] + x, plan.position[1]]
def update(self, time_passed):
'''
改变子弹位置
:param time_passed: 距离上次绘制图像到现在的时间
'''
# 如果子弹超出屏幕或者击中敌机,就设置self.position[1]为-100,在plan.draw的时候就移除它
if self.position[1] + self.image.get_size()[1] <= 0 or self.destroyed:
self.position[1] = -100
return
# 改变的距离 = 时间 * 速率
self.position[1] -= time_passed * self.speed
4 后记
这样,我们就把所有的操作读实现完了,接下来我们只需要使用Game().run(),就可以运行我们的游戏了。
本文所涉及的完整代码详见HGzhao/PythonApps