pygame实现俄罗斯方块游戏(基础篇1)
作者:冰风漫天 发布时间:2022-03-04 07:51:47
标签:pygame,俄罗斯方块
本文实例为大家分享了pygame实现俄罗斯方块游戏的具体代码,基础的第一篇,供大家参考,具体内容如下
一、初始界面
之前的游戏都比较简单,所以代码都是面向过程的写法,这次游戏后面可能会写比较复杂(比如人机对战、联机对战、使用道具对战等),这次面向对象一点来写这个项目。
游戏的窗口设计一个专门的Panel类便于负责单个游戏窗口的管理控制。
游戏主窗口按每个方块30像素,那么宽3010=300,高是3020=600
# -*- coding=utf-8 -*-
import random
import pygame
class Panel(object): # 用于绘制整个游戏窗口的版面
def __init__(self,bg, position):
self._bg=bg;
self._x,self._y,self._width,self._height=position
self._bgcolor=[0,0,0]
def paint(self):
mid_x=self._x+self._width/2
pygame.draw.line(self._bg,self._bgcolor,[mid_x,self._y],[mid_x,self._y+self._height],self._width)
def run():
pygame.init()
space=40
main_panel_width=300
main_panel_height=main_panel_width*2
screencaption = pygame.display.set_caption('Tetris')
screen = pygame.display.set_mode((main_panel_width+160+space*3,main_panel_height+space*2)) #设置窗口长宽
main_panel=Panel(screen,[space,space,main_panel_width,main_panel_height])
while True:
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit()
exit()
screen.fill((100,100,100)) # 将界面设置为灰色
main_panel.paint() # 主面盘绘制
pygame.display.update() # 必须调用update才能看到绘图显示
run()
效果图
二、方块管理
这里首先想到方块不同种类的可以使用工厂模式,所以先定义一个基类的Block,然后不同种类的方块分别继承自这个Block类,分别有这样七种方块
class Block(object):
def __init__(self):
self.rect_arr=[]
def get_rect_arr(self): # 用于获取方块种的四个矩形列表
return self.rect_arr
def move(self,xdiff,ydiff): # 用于移动方块的方法
self.new_rect_arr=[]
for x,y in self.rect_arr:
self.new_rect_arr.append((x+xdiff,y+ydiff))
self.rect_arr=self.new_rect_arr
class LongBlock(Block):
def __init__(self, n=None): # 两种形态
super(LongBlock, self).__init__()
if n is None: n=random.randint(0,1)
self.rect_arr=[(1,0),(1,1),(1,2),(1,3)] if n==0 else [(0,2),(1,2),(2,2),(3,2)]
class SquareBlock(Block): # 一种形态
def __init__(self, n=None):
super(SquareBlock, self).__init__()
self.rect_arr=[(1,1),(1,2),(2,1),(2,2)]
class ZBlock(Block): # 两种形态
def __init__(self, n=None):
super(ZBlock, self).__init__()
if n is None: n=random.randint(0,1)
self.rect_arr=[(2,0),(2,1),(1,1),(1,2)] if n==0 else [(0,1),(1,1),(1,2),(2,2)]
class SBlock(Block): # 两种形态
def __init__(self, n=None):
super(SBlock, self).__init__()
if n is None: n=random.randint(0,1)
self.rect_arr=[(1,0),(1,1),(2,1),(2,2)] if n==0 else [(0,2),(1,2),(1,1),(2,1)]
class LBlock(Block): # 四种形态
def __init__(self, n=None):
super(LBlock, self).__init__()
if n is None: n=random.randint(0,3)
if n==0: self.rect_arr=[(1,0),(1,1),(1,2),(2,2)]
elif n==1: self.rect_arr=[(0,1),(1,1),(2,1),(0,2)]
elif n==2: self.rect_arr=[(0,0),(1,0),(1,1),(1,2)]
else: self.rect_arr=[(0,1),(1,1),(2,1),(2,0)]
class JBlock(Block): # 四种形态
def __init__(self, n=None):
super(JBlock, self).__init__()
if n is None: n=random.randint(0,3)
if n==0: self.rect_arr=[(1,0),(1,1),(1,2),(0,2)]
elif n==1: self.rect_arr=[(0,1),(1,1),(2,1),(0,0)]
elif n==2: self.rect_arr=[(2,0),(1,0),(1,1),(1,2)]
else: self.rect_arr=[(0,1),(1,1),(2,1),(2,2)]
class TBlock(Block): # 四种形态
def __init__(self, n=None):
super(TBlock, self).__init__()
if n is None: n=random.randint(0,3)
if n==0: self.rect_arr=[(0,1),(1,1),(2,1),(1,2)]
elif n==1: self.rect_arr=[(1,0),(1,1),(1,2),(0,1)]
elif n==2: self.rect_arr=[(0,1),(1,1),(2,1),(1,0)]
else: self.rect_arr=[(1,0),(1,1),(1,2),(2,1)]
三、创建方块和方块落下
定义一个创建方块的函数
def create_block():
n = random.randint(0,19)
if n==0: return SquareBlock(n=0)
elif n==1 or n==2: return LongBlock(n=n-1)
elif n==3 or n==4: return ZBlock(n=n-3)
elif n==5 or n==6: return SBlock(n=n-5)
elif n>=7 and n<=10: return LBlock(n=n-7)
elif n>=11 and n<=14: return JBlock(n=n-11)
else: return TBlock(n=n-15)
给Panel类加一下当前移动方块的属性,并且修改它的paint方法,将移动方块绘制
class Panel(object): # 用于绘制整个游戏窗口的版面
moving_block=None # 正在落下的方块
def __init__(self,bg, block_size, position):
self._bg=bg;
self._x,self._y,self._width,self._height=position
self._block_size=block_size
self._bgcolor=[0,0,0]
def create_move_block(self):
block = create_block()
block.move(5-2,-2) # 方块挪到中间
self.moving_block=block
def move_block(self):
self.moving_block.move(0,1)
def paint(self):
mid_x=self._x+self._width/2
pygame.draw.line(self._bg,self._bgcolor,[mid_x,self._y],[mid_x,self._y+self._height],self._width) # 用一个粗线段来填充背景
# 绘制正在落下的方块
if self.move_block:
for rect in self.moving_block.get_rect_arr():
x,y=rect
pygame.draw.line(self._bg,[0,0,255],[self._x+x*bz+bz/2,self._y+y*bz],[self._x+x*bz+bz/2,self._y+(y+1)*bz],bz)
pygame.draw.rect(self._bg,[255,255,255],[self._x+x*bz,self._y+y*bz,bz,bz],1)
主循环中创建方块并将方块调整到下落的起始位置
main_panel.create_move_block()
设定位置刷新时间
diff_ticks = 300 # 移动一次蛇头的事件,单位毫秒
ticks = pygame.time.get_ticks() + diff_ticks
在主循环中刷新当前移动方块的位置
if pygame.time.get_ticks() >= ticks:
ticks+=diff_ticks
main_panel.move_block()
当前可以看到方块下落的效果了
四、方块落地的判断
在Block类里增加一个移动判断函数,下面这个这个can_move函数可以判断方块是不是落到底部了
def can_move(self,xdiff,ydiff):
for x,y in self.rect_arr:
if y+ydiff>=20: return False
return True
修改Panel的move函数,改为
def move_block(self):
if self.moving_block is None: create_move_block()
if self.moving_block.can_move(0,1):
self.moving_block.move(0,1)
else:
self.add_block(self.moving_block)
self.create_move_block()
这里增加了一个add_block函数,用于将已经落地的方块存起来,所以Panel另外做了三处改动
1.增加一个存已落下方块的数组变量
rect_arr=[] # 已经落底下的方块
2.定义add_block函数
def add_block(self,block):
for rect in block.get_rect_arr():
self.rect_arr.append(rect)
3.在paint里进行self.rect_arr的绘制
# 绘制已经落底下的方块
bz=self._block_size
for rect in self.rect_arr:
x,y=rect
pygame.draw.line(self._bg,[0,0,255],[self._x+x*bz+bz/2,self._y+y*bz],[self._x+x*bz+bz/2,self._y+(y+1)*bz],bz)
pygame.draw.rect(self._bg,[255,255,255],[self._x+x*bz,self._y+y*bz,bz,bz],1)
现在可以看到方块会落到底部,然后新的方块落下了
贴下目前的完整程序
# -*- coding=utf-8 -*-
import random
import pygame
class Panel(object): # 用于绘制整个游戏窗口的版面
rect_arr=[] # 已经落底下的方块
moving_block=None # 正在落下的方块
def __init__(self,bg, block_size, position):
self._bg=bg;
self._x,self._y,self._width,self._height=position
self._block_size=block_size
self._bgcolor=[0,0,0]
def add_block(self,block):
for rect in block.get_rect_arr():
self.rect_arr.append(rect)
def create_move_block(self):
block = create_block()
block.move(5-2,-2) # 方块挪到中间
self.moving_block=block
def move_block(self):
if self.moving_block is None: create_move_block()
if self.moving_block.can_move(0,1):
self.moving_block.move(0,1)
else:
self.add_block(self.moving_block)
self.create_move_block()
def paint(self):
mid_x=self._x+self._width/2
pygame.draw.line(self._bg,self._bgcolor,[mid_x,self._y],[mid_x,self._y+self._height],self._width) # 用一个粗线段来填充背景
# 绘制已经落底下的方块
bz=self._block_size
for rect in self.rect_arr:
x,y=rect
pygame.draw.line(self._bg,[0,0,255],[self._x+x*bz+bz/2,self._y+y*bz],[self._x+x*bz+bz/2,self._y+(y+1)*bz],bz)
pygame.draw.rect(self._bg,[255,255,255],[self._x+x*bz,self._y+y*bz,bz,bz],1)
# 绘制正在落下的方块
if self.move_block:
for rect in self.moving_block.get_rect_arr():
x,y=rect
pygame.draw.line(self._bg,[0,0,255],[self._x+x*bz+bz/2,self._y+y*bz],[self._x+x*bz+bz/2,self._y+(y+1)*bz],bz)
pygame.draw.rect(self._bg,[255,255,255],[self._x+x*bz,self._y+y*bz,bz,bz],1)
class Block(object):
def __init__(self):
self.rect_arr=[]
def get_rect_arr(self): # 用于获取方块种的四个矩形列表
return self.rect_arr
def move(self,xdiff,ydiff): # 用于移动方块的方法
self.new_rect_arr=[]
for x,y in self.rect_arr:
self.new_rect_arr.append((x+xdiff,y+ydiff))
self.rect_arr=self.new_rect_arr
def can_move(self,xdiff,ydiff):
for x,y in self.rect_arr:
if y+ydiff>=20: return False
return True
class LongBlock(Block):
def __init__(self, n=None): # 两种形态
super(LongBlock, self).__init__()
if n is None: n=random.randint(0,1)
self.rect_arr=[(1,0),(1,1),(1,2),(1,3)] if n==0 else [(0,2),(1,2),(2,2),(3,2)]
class SquareBlock(Block): # 一种形态
def __init__(self, n=None):
super(SquareBlock, self).__init__()
self.rect_arr=[(1,1),(1,2),(2,1),(2,2)]
class ZBlock(Block): # 两种形态
def __init__(self, n=None):
super(ZBlock, self).__init__()
if n is None: n=random.randint(0,1)
self.rect_arr=[(2,0),(2,1),(1,1),(1,2)] if n==0 else [(0,1),(1,1),(1,2),(2,2)]
class SBlock(Block): # 两种形态
def __init__(self, n=None):
super(SBlock, self).__init__()
if n is None: n=random.randint(0,1)
self.rect_arr=[(1,0),(1,1),(2,1),(2,2)] if n==0 else [(0,2),(1,2),(1,1),(2,1)]
class LBlock(Block): # 四种形态
def __init__(self, n=None):
super(LBlock, self).__init__()
if n is None: n=random.randint(0,3)
if n==0: self.rect_arr=[(1,0),(1,1),(1,2),(2,2)]
elif n==1: self.rect_arr=[(0,1),(1,1),(2,1),(0,2)]
elif n==2: self.rect_arr=[(0,0),(1,0),(1,1),(1,2)]
else: self.rect_arr=[(0,1),(1,1),(2,1),(2,0)]
class JBlock(Block): # 四种形态
def __init__(self, n=None):
super(JBlock, self).__init__()
if n is None: n=random.randint(0,3)
if n==0: self.rect_arr=[(1,0),(1,1),(1,2),(0,2)]
elif n==1: self.rect_arr=[(0,1),(1,1),(2,1),(0,0)]
elif n==2: self.rect_arr=[(2,0),(1,0),(1,1),(1,2)]
else: self.rect_arr=[(0,1),(1,1),(2,1),(2,2)]
class TBlock(Block): # 四种形态
def __init__(self, n=None):
super(TBlock, self).__init__()
if n is None: n=random.randint(0,3)
if n==0: self.rect_arr=[(0,1),(1,1),(2,1),(1,2)]
elif n==1: self.rect_arr=[(1,0),(1,1),(1,2),(0,1)]
elif n==2: self.rect_arr=[(0,1),(1,1),(2,1),(1,0)]
else: self.rect_arr=[(1,0),(1,1),(1,2),(2,1)]
def create_block():
n = random.randint(0,19)
if n==0: return SquareBlock(n=0)
elif n==1 or n==2: return LongBlock(n=n-1)
elif n==3 or n==4: return ZBlock(n=n-3)
elif n==5 or n==6: return SBlock(n=n-5)
elif n>=7 and n<=10: return LBlock(n=n-7)
elif n>=11 and n<=14: return JBlock(n=n-11)
else: return TBlock(n=n-15)
def run():
pygame.init()
space=30
main_block_size=30
main_panel_width=main_block_size*10
main_panel_height=main_block_size*20
screencaption = pygame.display.set_caption('Tetris')
screen = pygame.display.set_mode((main_panel_width+160+space*3,main_panel_height+space*2)) #设置窗口长宽
main_panel=Panel(screen,main_block_size,[space,space,main_panel_width,main_panel_height])
main_panel.create_move_block()
diff_ticks = 300 # 移动一次蛇头的事件,单位毫秒
ticks = pygame.time.get_ticks() + diff_ticks
while True:
for event in pygame.event.get():
if event.type == pygame.QUIT:
pygame.quit()
exit()
screen.fill((100,100,100)) # 将界面设置为灰色
main_panel.paint() # 主面盘绘制
pygame.display.update() # 必须调用update才能看到绘图显示
if pygame.time.get_ticks() >= ticks:
ticks+=diff_ticks
main_panel.move_block()
run()
这章先写到这,下章继续
来源:https://blog.csdn.net/zhangenter/article/details/89299077


猜你喜欢
- 一份基于cnn的手写数字自识别的代码,供大家参考,具体内容如下# -*- coding: utf-8 -*-import tensorflo
- 测试sql: 代码如下:SET STATISTICS IO ON SET STATISTICS TIME ON SELECT COUNT(1
- 最近有一个需求要把dataframe转换为多维矩阵,然后可以使用values来实现,下面记录一下代码,方便以后使用。import panda
- 咱们Python 集中营有一个专题就是分享一些有意思的东西,今天大概看了一下pygame的这个非标准库就想着使用它来做个小游戏-拼图。通过加
- Python:2.7 IDE:Pycharm5.0.3 今天遇到一个问题,就是在使用json.load()时,中文字符被转化为Unicode
- 本文实例讲述了Python正则表达式分组概念与用法。分享给大家供大家参考,具体如下:正则表达式分组分组就是用一对圆括号“()”括起来的正则表
- 签名import base64import jsonimport timefrom datetime import datetimeimpo
- 关于窗口函数的基础,请看文章SQL窗口函数许多常见的聚合函数也可以作为窗口函数使用,包括AVG()、SUM()、COUNT()、MAX()以
- 用于操作数据库的SQL一般分为两种,一种是查询语句,也就是我们所说的 SELECT语句,另外一种就是更新语句,也叫做数据操作语句。言外之 意
- 先建立2个测试表,在id列上创建unique约束。 mysql> create table test1(id int,name var
- var yData = [];//Y轴数据 var xData = [];//X轴数据 $(data.rows).each(function
- 在为一个客户排除死锁问题时我遇到了一个有趣的包括InnoDB间隙锁的情形。对于一个WHERE子句不匹配任何行的非插入的写操作中,
- 本文介绍了vscode 远程调试python的方法,分享给大家,具有如下:实验环境远程服务器:京东云,1核2G,centos7.3 64bi
- 为了减少页面的加载速度,提高用户体验,对于一些图片决定使用图标代替,但是发现element-ui的图标少得可怜,完全满足不了我的要求,于是决
- 有时候,因为内容的更改或者隐私问题,我们往往不 希望别人通过“百度快照”的方法 查看 自己网站的某一些网页,对于网站管理员来说,百度快照也分
- 已经pip安装好第三方库了,但是在pycharm中import还是标红,像下图:我记得上次重装系统,出现这种情况的时候,我重启一下pycha
- Golang 开发者遇到的许多问题之一是尝试将一个函数的参数设置为可选. 这是一个非常常见的用例, 有些对象应该使用一些基本的默认设置来开箱
- 如下所示:# -*- coding: utf-8 -*-# 要求:输入一行字符,分别统计出其中英文字母、空格、数字和其它字符的个数。def
- 迭代首先理解下什么是迭代,python中所有从左往右扫面对象的方式都是可迭代的有哪些方式是可迭代的:1.文件操作 我
- 前言需求: 必须现在需要动态创建16个list,每个list的名字不一样,但是是有规律可循,比如第一个list的名字叫:arriage_li