Python Pygame实战之打砖块小游戏
作者:顾木子吖 发布时间:2021-04-10 01:41:11
标签:Python,Pygame,打砖块,游戏
导语
嘿!前不久刚刚给大家过一款反弹球的小游戏嘛!
不知道大家还记得不?不记得可以看下往期的内容呢,在上一期的基础上升级了这款打砖块的小游戏,界面的话也挺简单的,经典配色原汁原味哈哈哈。
大家好,我是木木子,一个上的编程下的厅堂的女码农!今天带大家编写一款经典的打砖块儿小游戏!
小科普:
打砖块最早是由雅达利公司开发的一款独立游戏,也是无数人的童年记忆。
在谷歌图片中搜索“atari breakout”(雅利达打砖块游戏),搜索结果就会变成这款游戏。把所有砖块都清除后,还能继续进入下一轮挑战。
大家也可以试试 哈哈哈~希望给大家介绍更多编程方各种知识!
一、准备中
1)游戏规则:
把所有砖块都清除后,还能继续进入下一轮挑战
初始化每个玩家2次机会,打完所有砖块儿即可胜利,否则失败游戏不过关!
(关卡素材、背景音乐等比较少也不展示了需要的主页源码基地见哈)
2)环境安装
本文用到的环境:Python3、Pycharm、Pygame模块以及部分自带。
环境安装:pip install -i https://pypi.douban.com/simple/ +模块名
二、开始敲代码
1)配置文件
import os
'''游戏界面一些数值'''
SCREENWIDTH = 640
SCREENHEIGHT = 480
BRICKWIDTH = 10
BRICKHEIGHT = 10
PADDLEWIDTH = 60
PADDLEHEIGHT = 12
BALLRADIUS = 8
'''游戏素材路径'''
FONTPATH = os.path.join(os.getcwd(), 'resources/font/font.TTF')
HITSOUNDPATH = os.path.join(os.getcwd(), 'resources/audios/hit.wav')
BGMPATH = os.path.join(os.getcwd(), 'resources/audios/bgm.mp3')
LEVELROOTPATH = os.path.join(os.getcwd(), 'resources/levels')
LEVELPATHS = [os.path.join(LEVELROOTPATH, '%s.level' % str(i+1)) for i in range(len(os.listdir(LEVELROOTPATH)))]
'''一些颜色'''
BLACK = (0, 0, 0)
WHITE = (255, 255, 255)
PINK = (212, 149, 174)
PURPLE = (168, 152, 191)
YELLOW = (245, 237, 162)
BLUE = (51, 170, 230)
AQUA = (182, 225, 225)
2)定义一些类
import random
import pygame
'''板子'''
class Paddle(pygame.sprite.Sprite):
def __init__(self, x, y, width, height, SCREENWIDTH, SCREENHEIGHT, **kwargs):
pygame.sprite.Sprite.__init__(self)
self.init_state = [x, y, width, height]
self.rect = pygame.Rect(x, y, width, height)
self.base_speed = 10
self.SCREENWIDTH = SCREENWIDTH
self.SCREENHEIGHT = SCREENHEIGHT
'''移动板子'''
def move(self, direction):
if direction == 'left':
self.rect.left = max(0, self.rect.left-self.base_speed)
elif direction == 'right':
self.rect.right = min(self.SCREENWIDTH, self.rect.right+self.base_speed)
else:
raise ValueError('Paddle.move.direction unsupport %s...' % direction)
return True
'''绑定到屏幕上'''
def draw(self, screen, color):
pygame.draw.rect(screen, color, self.rect)
return True
'''重置'''
def reset(self):
self.rect = pygame.Rect(self.init_state[0], self.init_state[1], self.init_state[2], self.init_state[3])
return True
'''球'''
class Ball(pygame.sprite.Sprite):
def __init__(self, x, y, radius, SCREENWIDTH, SCREENHEIGHT, **kwargs):
pygame.sprite.Sprite.__init__(self)
self.init_state = [x, y, radius*2, radius*2]
self.rect = pygame.Rect(x, y, radius*2, radius*2)
self.base_speed = [5, 5]
self.direction = [random.choice([1, -1]), -1]
self.radius = radius
self.SCREENWIDTH = SCREENWIDTH
self.SCREENHEIGHT = SCREENHEIGHT
'''移动球'''
def move(self):
self.rect.left += self.direction[0] * self.base_speed[0]
self.rect.top += self.direction[1] * self.base_speed[1]
if self.rect.left <= 0:
self.rect.left = 0
self.direction[0] = -self.direction[0]
elif self.rect.right >= self.SCREENWIDTH:
self.rect.right = self.SCREENWIDTH
self.direction[0] = -self.direction[0]
if self.rect.top <= 0:
self.rect.top = 0
self.direction[1] = -self.direction[1]
elif self.rect.bottom >= self.SCREENHEIGHT:
return False
return True
'''改变运动速度和方向(与拍相撞时)'''
def change(self):
self.base_speed = [random.choice([4, 5, 6]), random.choice([4, 5, 6])]
self.direction = [random.choice([1, -1]), -1]
return True
'''绑定到屏幕上'''
def draw(self, screen, color):
pygame.draw.circle(screen, color, (self.rect.left+self.radius, self.rect.top+self.radius), self.radius)
return True
'''重置'''
def reset(self):
self.rect = pygame.Rect(self.init_state[0], self.init_state[1], self.init_state[2], self.init_state[3])
return True
'''砖块'''
class Brick(pygame.sprite.Sprite):
def __init__(self, x, y, width, height, **kwargs):
pygame.sprite.Sprite.__init__(self)
self.init_state = [x, y, width, height]
self.rect = pygame.Rect(x, y, width, height)
'''绑定到屏幕上'''
def draw(self, screen, color):
pygame.draw.rect(screen, color, self.rect)
return True
'''重置'''
def reset(self):
self.rect = pygame.Rect(self.init_state[0], self.init_state[1], self.init_state[2], self.init_state[3])
return True
3)定义开始、结束界面
'''开始界面'''
def __startInterface(self):
clock = pygame.time.Clock()
while True:
for event in pygame.event.get():
if event.type == pygame.QUIT or (event.type == pygame.KEYDOWN and event.key == pygame.K_ESCAPE):
pygame.quit()
sys.exit(-1)
if event.type == pygame.KEYDOWN and event.key == pygame.K_RETURN:
return
self.screen.fill(self.cfg.AQUA)
text1 = 'Press <Enter> to start the game'
text2 = 'Press <Esc> to quit the game'
text_render1 = self.font_big.render(text1, False, self.cfg.BLUE)
text_render2 = self.font_big.render(text2, False, self.cfg.BLUE)
self.screen.blit(text_render1, ((self.cfg.SCREENWIDTH-text_render1.get_rect().width)//2, (self.cfg.SCREENHEIGHT-text_render1.get_rect().height)//4))
self.screen.blit(text_render2, ((self.cfg.SCREENWIDTH-text_render2.get_rect().width)//2, (self.cfg.SCREENHEIGHT-text_render2.get_rect().height)//2))
pygame.display.flip()
clock.tick(30)
'''结束界面'''
def __endInterface(self, is_win):
if is_win:
text1 = 'Congratulations! You win!'
else:
text1 = 'Game Over! You fail!'
text2 = 'Press <R> to restart the game'
text3 = 'Press <Esc> to quit the game.'
clock = pygame.time.Clock()
while True:
for event in pygame.event.get():
if event.type == pygame.QUIT or (event.type == pygame.KEYDOWN and event.key == pygame.K_ESCAPE):
pygame.quit()
sys.exit(-1)
if event.type == pygame.KEYDOWN and event.key == pygame.K_r:
return
self.screen.fill(self.cfg.AQUA)
text_render1 = self.font_big.render(text1, False, self.cfg.BLUE)
text_render2 = self.font_big.render(text2, False, self.cfg.BLUE)
text_render3 = self.font_big.render(text3, False, self.cfg.BLUE)
self.screen.blit(text_render1, ((self.cfg.SCREENWIDTH-text_render1.get_rect().width)//2, (self.cfg.SCREENHEIGHT-text_render1.get_rect().height)//4))
self.screen.blit(text_render2, ((self.cfg.SCREENWIDTH-text_render2.get_rect().width)//2, (self.cfg.SCREENHEIGHT-text_render2.get_rect().height)//2))
self.screen.blit(text_render3, ((self.cfg.SCREENWIDTH-text_render3.get_rect().width)//2, (self.cfg.SCREENHEIGHT-text_render2.get_rect().height)//1.5))
pygame.display.flip()
clock.tick(30)
4)定义游戏
'''打砖块游戏'''
class breakoutClone():
def __init__(self, cfg, **kwargs):
pygame.init()
pygame.display.set_caption('Breakout clone ')
pygame.mixer.init()
self.screen = pygame.display.set_mode((cfg.SCREENWIDTH, cfg.SCREENHEIGHT))
self.font_small = pygame.font.Font(cfg.FONTPATH, 20)
self.font_big = pygame.font.Font(cfg.FONTPATH, 30)
self.hit_sound = pygame.mixer.Sound(cfg.HITSOUNDPATH)
pygame.mixer.music.load(cfg.BGMPATH)
pygame.mixer.music.play(-1, 0.0)
self.cfg = cfg
'''运行游戏'''
def run(self):
while True:
self.__startInterface()
for idx, levelpath in enumerate(self.cfg.LEVELPATHS):
state = self.__runLevel(levelpath)
if idx == len(self.cfg.LEVELPATHS)-1:
break
if state == 'win':
self.__nextLevel()
else:
break
if state == 'fail':
self.__endInterface(False)
else:
self.__endInterface(True)
'''运行某关卡'''
def __runLevel(self, levelpath):
score = 0
num_lives = 2
# running: 游戏正在进行, fail: 游戏失败, win: 游戏成功.
state = 'running'
paddle = Paddle((self.cfg.SCREENWIDTH-self.cfg.PADDLEWIDTH)/2, self.cfg.SCREENHEIGHT-self.cfg.PADDLEHEIGHT-10, self.cfg.PADDLEWIDTH, self.cfg.PADDLEHEIGHT, self.cfg.SCREENWIDTH, self.cfg.SCREENHEIGHT)
ball = Ball(paddle.rect.centerx-self.cfg.BALLRADIUS, paddle.rect.top-self.cfg.BALLRADIUS*2, self.cfg.BALLRADIUS, self.cfg.SCREENWIDTH, self.cfg.SCREENHEIGHT)
brick_sprites = pygame.sprite.Group()
brick_positions = loadLevel(levelpath)
for bp in brick_positions:
brick_sprites.add(Brick(bp[0]*self.cfg.BRICKWIDTH, bp[1]*self.cfg.BRICKHEIGHT, self.cfg.BRICKWIDTH, self.cfg.BRICKHEIGHT))
clock = pygame.time.Clock()
while True:
if state != 'running':
return state
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit()
sys.exit(-1)
keys_pressed = pygame.key.get_pressed()
if keys_pressed[pygame.K_LEFT]:
paddle.move('left')
elif keys_pressed[pygame.K_RIGHT]:
paddle.move('right')
self.screen.fill(self.cfg.AQUA)
is_alive = ball.move()
# 判断有没有接住球
if not is_alive:
ball.reset()
paddle.reset()
num_lives -= 1
if num_lives == 0:
state = 'fail'
# 球和砖块碰撞检测
num_bricks = pygame.sprite.spritecollide(ball, brick_sprites, True)
score += len(num_bricks)
# 球和拍碰撞检测
if pygame.sprite.collide_rect(ball, paddle):
ball.change()
# 判断砖块是否已经打完
if len(brick_sprites) == 0:
state = 'win'
# 将游戏精灵绑定到屏幕
paddle.draw(self.screen, self.cfg.PURPLE)
ball.draw(self.screen, self.cfg.WHITE)
for brick in brick_sprites:
brick.draw(self.screen, self.cfg.YELLOW)
text_render = self.font_small.render('SCORE: %s, LIVES: %s' % (score, num_lives), False, self.cfg.BLUE)
self.screen.blit(text_render, (10, 10))
pygame.display.flip()
clock.tick(50)
'''关卡切换'''
def __nextLevel(self):
clock = pygame.time.Clock()
while True:
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit()
sys.exit(-1)
if event.type == pygame.KEYDOWN and event.key == pygame.K_RETURN:
return
self.screen.fill(self.cfg.AQUA)
text = 'Press <Enter> to enter the next level'
text_render = self.font_big.render(text, False, self.cfg.BLUE)
self.screen.blit(text_render, ((self.cfg.SCREENWIDTH-text_render.get_rect().width)//2, (self.cfg.SCREENHEIGHT-text_render.get_rect().height)//3))
pygame.display.flip()
clock.tick(30)
5)主函数与运行界面
import cfg
from modules import breakoutClone
'''主函数'''
def main():
game = breakoutClone(cfg)
game.run()
'''run'''
if __name__ == '__main__':
main()
三、效果展示
1)视频效果展示——
视频链接
带你回忆经典:原生Python开发一款打砖块儿小游戏~
2)截图效果展示——
游戏开始界面——
运行界面——
来源:https://blog.csdn.net/weixin_55822277/article/details/121944970


猜你喜欢
- IT行业,技术要比学历、年龄、从业经验更为重要,技术水平直接决定就业薪资,想要学好python,首先要先了解精通Python语言基础、Pyt
- 在默认情况下,Access 2000/2002数据库是以“共享”的方式打开的,这样可以保证多人能够同时使用同一个数据库。不过,在共享方式打开
- ORDER BY _column1, _column2; /* _column1升序,_column2升序 */ ORDER BY _col
- 一、闭包1.1 三要素 必须有一个内嵌函数内嵌函数必须引用外部函数中变量外部函数返回值必须是内嵌函数1.2 语法# 语法def 外部函数名(
- 最近在无忧脚本混了一阵子,回复了一些贴子,自己却没有做出什么东东让大家看看,心里有些不安,于是写了下边的一点东西,本来应该发在类封装区的,考
- 本文实例为大家分享了tensorflow神经网络实现mnist分类的具体代码,供大家参考,具体内容如下只有两层的神经网络,直接上代码#引入包
- 1.索引问题索引是数据库优化中最常用也是最重要的手段之一,通过索引通常可以帮助用户解决大多数 的SQL性能问题。本章节将对MySQL中的索引
- 1:为什么我得不到变量 我在一网页向另一网页POST数据name,为什么输出$name时却得不到任何值? 在PHP4.2以后的版本中regi
- python3.6.2环境安装配置图文教程,具体如下一、需要下载的软件》python3.6.2.exe (也可以选择更新的版本) ----
- 查询学生步骤1:设置导航 步骤2:添加路由步骤3:创建页面步骤:步骤1:准备2个变量(pageInfo、studentVo)步骤2
- 本文实例为大家分享了python树莓派红外反射传感器的程序,供大家参考,具体内容如下1、工具rpi3,微雪ARPI600,Infrared
- 本文实例讲述了JS实现json数组排序操作。分享给大家供大家参考,具体如下:有时需要根据json对象的某个属性排序json数组,javasc
- 本文实例讲述了JavaScript导出Excel的方法。分享给大家供大家参考。具体实现方法如下:<html xmlns="h
- 通常来说,Python的变量/数据类型非常多,但是它是不需要用户指定的,因为有些是根据部份系统函数生成,另外一些是自动根据变量的值识别的,这
- Python 中的 main 函数充当程序的执行点,在 Python 编程中定义 main 函数是启
- 我的坐标是深圳,2022年以来,大部分时候要求24小时,少部分时候要求48小时,更少的时候要求72小时,没有更长的情况。本文根据我的核酸检测
- 项目总览创建虚拟环境mkvirtualenv meiduo_malls创建项目django-admin startproject meidu
- 0. 前言深度学习已经成为机器学习中最受欢迎和发展最快的领域。自 2012 年深度学习性能超越机器学习等传统方法以来,深度学习架构开始快速应
- 以下来自John Hann的实现,这段代码引起了我的注意,它用巧妙的方法把方法调用的结果缓存起来了。代码解析:// memoize: 使用m
- 弹指间,2023已经到来,新的一年,祝大家新年快乐,阖家幸福呀~~~好吧,进入正题,2023的到来,肯定少不了烟花吧(外面不让放炮,那咱们就