一、简单说明

80、90后的小伙伴都玩过“俄罗斯方块”,那种“叱咤风云”场景 偶尔闪现在脑海 真的是太爽了;如果没有来得及玩过的同学,这次可以真正的自己做一个了

本实例用的是Python3(当然了Python3.5 3.6 3.7....都行 )+ pygame实现的

运行之前需要安装pygame模块,安装命令如下

pip install pygame -i https://mirrors.aliyun.com/pypi/simple/

二、运行效果

三、完整代码

下面的代码,用到了字体文件yh.ttf,下载地址 https://www.itprojects.cn/detail.html?example_id=8bf48a0ac4cee29043d4122fe6e563cb

文件main.py代码如下:

"""
作者:it项目实例网
更多项目实例,请访问:www.itprojects.cn
""" import random
import sys
import time import pygame from blocks import block_s, block_i, block_j, block_l, block_o, block_t, block_z SCREEN_WIDTH, SCREEN_HEIGHT = 450, 750
BG_COLOR = (40, 40, 60) # 背景色
BLOCK_COL_NUM = 10 # 每行的方格数
SIZE = 30 # 每个小方格大小
BLOCK_ROW_NUM = 25 # 每列的方个数
BORDER_WIDTH = 4 # 游戏区边框宽度
RED = (200, 30, 30) # 红色,GAME OVER 的字体颜色 def judge_game_over(stop_all_block_list):
"""
判断游戏是否结束
"""
if "O" in stop_all_block_list[0]:
return True def change_speed(score):
speed_level = [("1", 0.5, 0, 20), ("2", 0.4, 21, 50), ("3", 0.3, 51, 100), ("4", 0.2, 101, 200), ("5", 0.1, 201, None)]
for speed_info, speed, score_start, score_stop in speed_level:
if score_stop and score_start <= score <= score_stop:
return speed_info, speed
elif score_stop is None and score >= score_start:
return speed_info, speed def judge_lines(stop_all_block_list):
"""
判断是否有同一行的方格,如果有则消除
"""
# 记录刚刚消除的行数
move_row_list = list()
# 消除满格的行
for row, line in enumerate(stop_all_block_list):
if "." not in line:
# 如果这一行没有. 那么就意味着全部是O,则消除这一行
stop_all_block_list[row] = ['.' for _ in range(len(line))]
move_row_list.append(row) # 如果没有满格的行,则结束此函数
if not move_row_list:
return 0 # 移动剩余的行到下一行
for row in move_row_list:
stop_all_block_list.pop(row)
stop_all_block_list.insert(0, ['.' for _ in range(len(line))]) return len(move_row_list) * 10 def add_to_stop_all_block_list(stop_all_block_list, current_block, current_block_start_row, current_block_start_col):
"""
将当前已经停止移动的block添加到列表中
"""
for row, line in enumerate(current_block):
for col, block in enumerate(line):
if block != '.':
stop_all_block_list[current_block_start_row + row][current_block_start_col + col] = "O" def change_current_block_style(current_block):
"""
改变图形的样式
"""
# 计算出,当前图形样式属于哪个图形
current_block_style_list = None
for block_style_list in [block_s, block_i, block_j, block_l, block_o, block_t, block_z]:
if current_block in block_style_list:
current_block_style_list = block_style_list # 得到当前正在用的图形的索引(下标)
index = current_block_style_list.index(current_block)
# 它的下一个图形的索引
index += 1
# 防止越界
index = index % len(current_block_style_list)
# 返回下一个图形
return current_block_style_list[index] def judge_move_right(current_block, current_block_start_col):
"""
判断是否可以向右移动
"""
# 先判断列的方式是从右到左
for col in range(len(current_block[0]) - 1, -1, -1):
# 得到1列的所有元素
col_list = [line[col] for line in current_block]
# 判断是否碰到右边界
if 'O' in col_list and current_block_start_col + col >= BLOCK_COL_NUM:
return False
return True def judge_move_left(current_block, current_block_start_col):
"""
判断是否可以向左移动
"""
# 先判断列的方式是从左到右
for col in range(len(current_block[0])):
# 得到1列的所有元素
col_list = [line[col] for line in current_block]
# 判断是否碰到右边界
if 'O' in col_list and current_block_start_col + col < 0:
return False
return True def judge_move_down(current_block, current_block_start_row, current_block_start_col, stop_all_block_list):
"""
判断是否碰撞到其它图形或者底边界
"""
# 得到其它图形所有的坐标
stop_all_block_position = list()
for row, line in enumerate(stop_all_block_list):
for col, block in enumerate(line):
if block != ".":
stop_all_block_position.append((row, col))
# print(stop_all_block_position) # 判断碰撞
for row, line in enumerate(current_block):
if 'O' in line and current_block_start_row + row >= BLOCK_ROW_NUM:
# 如果当前行有0,且从起始行开始算+当前显示的行,超过了总行数,那么就认为碰到了底部
return False
for col, block in enumerate(line):
if block != "." and (current_block_start_row + row, current_block_start_col + col) in stop_all_block_position:
return False return True def get_block():
"""
创建一个图形
"""
block_style_list = random.choice([block_s, block_i, block_j, block_l, block_o, block_t, block_z])
return random.choice(block_style_list) def main():
pygame.init()
screen = pygame.display.set_mode((SCREEN_WIDTH, SCREEN_HEIGHT))
pygame.display.set_caption('俄罗斯方块') current_block = get_block() # 当前图形
current_block_start_row = -2 # 当前图片从哪一行开始显示图形
current_block_start_col = 4 # 当前图形从哪一列开始显示
next_block = get_block() # 下一个图形
last_time = time.time()
speed = 0.5 # 降落的速度
speed_info = '1' # 显示的速度等级 # 定义一个列表,用来存储所有的已经停止移动的形状
stop_all_block_list = [['.' for i in range(BLOCK_COL_NUM)] for j in range(BLOCK_ROW_NUM)] # 字体
font = pygame.font.Font('yh.ttf', 24) # 黑体24
game_over_font = pygame.font.Font("yh.ttf", 72)
game_over_font_width, game_over_font_height = game_over_font.size('GAME OVER')
game_again_font_width, game_again_font_height = font.size('鼠标点击任意位置,再来一局') # 得分
score = 0 # 标记游戏是否结束
game_over = False # 创建计时器(防止while循环过快,占用太多CPU的问题)
clock = pygame.time.Clock()
while True:
for event in pygame.event.get():
if event.type == pygame.QUIT:
sys.exit()
elif event.type == pygame.KEYDOWN:
if event.key == pygame.K_LEFT:
if judge_move_left(current_block, current_block_start_col - 1):
current_block_start_col -= 1
elif event.key == pygame.K_RIGHT:
if judge_move_right(current_block, current_block_start_col + 1):
current_block_start_col += 1
elif event.key == pygame.K_UP:
current_block_next_style = change_current_block_style(current_block)
if judge_move_left(current_block_next_style, current_block_start_col) and \
judge_move_right(current_block_next_style, current_block_start_col) and \
judge_move_down(current_block, current_block_start_row, current_block_start_col, stop_all_block_list):
# 判断新的样式没有越界
current_block = current_block_next_style
elif event.key == pygame.K_DOWN:
# 判断是否可以向下移动,如果碰到底部或者其它的图形就不能移动了
if judge_move_down(current_block, current_block_start_row + 1, current_block_start_col, stop_all_block_list):
current_block_start_row += 1
elif event.type == pygame.MOUSEBUTTONDOWN and event.button:
if game_over:
# 重置游戏用到的变量
current_block = get_block() # 当前图形
current_block_start_row = -2 # 当前图片从哪一行开始显示图形
current_block_start_col = 4 # 当前图形从哪一列开始显示
next_block = get_block() # 下一个图形
stop_all_block_list = [['.' for i in range(BLOCK_COL_NUM)] for j in range(BLOCK_ROW_NUM)]
score = 0
game_over = False # 判断是否修改当前图形显示的起始行
if not game_over and time.time() - last_time > speed:
last_time = time.time()
# 判断是否可以向下移动,如果碰到底部或者其它的图形就不能移动了
if judge_move_down(current_block, current_block_start_row + 1, current_block_start_col, stop_all_block_list):
current_block_start_row += 1
else:
# 将这个图形存储到统一的列表中,这样便于判断是否成为一行
add_to_stop_all_block_list(stop_all_block_list, current_block, current_block_start_row, current_block_start_col)
# 判断是否有同一行的,如果有就消除,且加上分数
score += judge_lines(stop_all_block_list)
# 判断游戏是否结束(如果第一行中间有O那么就表示游戏结束)
game_over = judge_game_over(stop_all_block_list)
# 调整速度
speed_info, speed = change_speed(score)
# 创建新的图形
current_block = next_block
next_block = get_block()
# 重置数据
current_block_start_col = 4
current_block_start_row = -2 # 画背景(填充背景色)
screen.fill(BG_COLOR) # 画游戏区域分隔线
pygame.draw.line(screen, (100, 40, 200), (SIZE * BLOCK_COL_NUM, 0), (SIZE * BLOCK_COL_NUM, SCREEN_HEIGHT), BORDER_WIDTH) # 显示当前图形
for row, line in enumerate(current_block):
for col, block in enumerate(line):
if block != '.':
pygame.draw.rect(screen, (20, 128, 200), ((current_block_start_col + col) * SIZE, (current_block_start_row + row) * SIZE, SIZE, SIZE), 0) # 显示所有停止移动的图形
for row, line in enumerate(stop_all_block_list):
for col, block in enumerate(line):
if block != '.':
pygame.draw.rect(screen, (20, 128, 200), (col * SIZE, row * SIZE, SIZE, SIZE), 0) # 画网格线 竖线
for x in range(BLOCK_COL_NUM):
pygame.draw.line(screen, (0, 0, 0), (x * SIZE, 0), (x * SIZE, SCREEN_HEIGHT), 1)
# 画网格线 横线
for y in range(BLOCK_ROW_NUM):
pygame.draw.line(screen, (0, 0, 0), (0, y * SIZE), (BLOCK_COL_NUM * SIZE, y * SIZE), 1) # 显示右侧(得分、速度、下一行图形)
# 得分
score_show_msg = font.render('得分: ', True, (255, 255, 255))
screen.blit(score_show_msg, (BLOCK_COL_NUM * SIZE + 10, 10))
score_show_msg = font.render(str(score), True, (255, 255, 255))
screen.blit(score_show_msg, (BLOCK_COL_NUM * SIZE + 10, 50))
# 速度
speed_show_msg = font.render('速度: ', True, (255, 255, 255))
screen.blit(speed_show_msg, (BLOCK_COL_NUM * SIZE + 10, 100))
speed_show_msg = font.render(speed_info, True, (255, 255, 255))
screen.blit(speed_show_msg, (BLOCK_COL_NUM * SIZE + 10, 150))
# 下一个图形(文字提示)
next_style_msg = font.render('下一个: ', True, (255, 255, 255))
screen.blit(next_style_msg, (BLOCK_COL_NUM * SIZE + 10, 200))
# 下一个图形(图形)
for row, line in enumerate(next_block):
for col, block in enumerate(line):
if block != '.':
pygame.draw.rect(screen, (20, 128, 200), (320 + SIZE * col, (BLOCK_COL_NUM + row) * SIZE, SIZE, SIZE), 0)
# 显示这个方格的4个边的颜色
# 左
pygame.draw.line(screen, (0, 0, 0), (320 + SIZE * col, (BLOCK_COL_NUM + row) * SIZE), (320 + SIZE * col, (BLOCK_COL_NUM + row + 1) * SIZE), 1)
# 上
pygame.draw.line(screen, (0, 0, 0), (320 + SIZE * col, (BLOCK_COL_NUM + row) * SIZE), (320 + SIZE * (col + 1), (BLOCK_COL_NUM + row) * SIZE), 1)
# 下
pygame.draw.line(screen, (0, 0, 0), (320 + SIZE * col, (BLOCK_COL_NUM + row + 1) * SIZE), (320 + SIZE * (col + 1), (BLOCK_COL_NUM + row + 1) * SIZE), 1)
# 右
pygame.draw.line(screen, (0, 0, 0), (320 + SIZE * (col + 1), (BLOCK_COL_NUM + row) * SIZE), (320 + SIZE * (col + 1), (BLOCK_COL_NUM + row + 1) * SIZE), 1) # 显示游戏结束画面
if game_over:
game_over_tips = game_over_font.render('GAME OVER', True, RED)
screen.blit(game_over_tips, ((SCREEN_WIDTH - game_over_font_width) // 2, (SCREEN_HEIGHT - game_over_font_height) // 2))
# 显示"鼠标点击任意位置,再来一局"
game_again = font.render('鼠标点击任意位置,再来一局', True, RED)
screen.blit(game_again, ((SCREEN_WIDTH - game_again_font_width) // 2, (SCREEN_HEIGHT - game_again_font_height) // 2 + 80)) # 刷新显示(此时窗口才会真正的显示)
pygame.display.update()
# FPS(每秒钟显示画面的次数)
clock.tick(60) # 通过一定的延时,实现1秒钟能够循环60次 if __name__ == '__main__':
main()

文件blocks.py代码如下:

# S形方块
block_s = [['.OO',
'OO.',
'...'],
['O..',
'OO.',
'.O.']]
# Z形方块
block_z = [['OO.',
'.OO',
'...'],
['.O.',
'OO.',
'O..']]
# I型方块
block_i = [['.O..',
'.O..',
'.O..',
'.O..'],
['....',
'....',
'OOOO',
'....']]
# O型方块
block_o = [['OO',
'OO']]
# J型方块
block_j = [['O..',
'OOO',
'...'],
['.OO',
'.O.',
'.O.'],
['...',
'OOO',
'..O'],
['.O.',
'.O.',
'OO.']]
# L型方块
block_l = [['..O',
'OOO',
'...'],
['.O.',
'.O.',
'.OO'],
['...',
'OOO',
'O..'],
['OO.',
'.O.',
'.O.']]
# T型方块
block_t = [['.O.',
'OOO',
'...'],
['.O.',
'.OO',
'.O.'],
['...',
'OOO',
'.O.'],
['.O.',
'OO.',
'.O.']]

Python3+pygame实现的俄罗斯方块 代码完整 有演示效果的更多相关文章

  1. Python3+pygame实现的90坦克大战 代码完整 有演示效果

    我是一个典型的80后,年轻时玩过了特别多的游戏,所以这几天用Python3+pygame实现了一个另外小游戏"坦克大战"(其他的游戏,请翻阅我的博客) 本实例代码量有些多,完整的版 ...

  2. Python3+pygame实现的flappy bird游戏,代码完整,还有音乐

    之前一直在手机上玩flappy bird游戏,闲暇时间就编写了一个 是采用python3+pygame模块制作而成的,运行效果非常流畅,会让你大吃一惊哦哈哈 一.运行效果展示 下载游戏之后,注意在自己 ...

  3. python3+tkinter实现的黑白棋,代码完整 100%能运行

    今天分享给大家的是采用Python3+tkinter制作而成的小项目--黑白棋 tkinter是Python内置的图形化模块,简单易用,一般的小型UI程序可以快速用它实现,具体的tkinter相关知识 ...

  4. 吴裕雄--天生自然python学习笔记:python 用pygame模块开发俄罗斯方块游戏

    俄罗斯方块游戏 多年前,游戏机中最流行的游戏就是“俄罗斯方块”了.时至今日,虽然网络 游戏日新月异 ,但“俄罗斯方块”这款小游戏仍在许多人心中 占有一席之地.本例中, 我们将亲手设计一个简单的俄罗斯方 ...

  5. 在python3.x下使用如下代码: import cPickle as pk 报错

    在python3.x下使用如下代码: import cPickle as pk会报如下错误:ImportError: No module named 'cPickle' 原因:python2有cPic ...

  6. python3.6从含有html代码的json的中取出某个值

    python3.6从含有html代码的json的中取出某个值 之前在做接口测试的时候,网站的后端是用java写的,然后接口的response返回的都是json格式,json很简单,就是字典,类似这样子 ...

  7. CountDownTimer完整具体演示样例

    MainActivity例如以下: package cc.cv; import android.os.Bundle; import android.os.CountDownTimer; import ...

  8. [刘阳Java]_纯CSS代码实现内容过滤效果

    继续我们技术专题课,我们今天给大家带来的是一个比较酷炫的"纯CSS代码实现内容过滤效果",没有加入任何JS的效果.全部都是应用CSS3的新增选择器来实现的.先看效果截图 实现思路 ...

  9. 关于html5与jsp页面同样的html代码展示的页面效果不一样的问题

    原文:关于html5与jsp页面同样的html代码展示的页面效果不一样的问题 html5默认的声明为 <!DOCTYPE html> jsp默认的声明头部为 <%@ page con ...

随机推荐

  1. HDU-6704 K-th occurrence(后缀数组+主席树)

    题意 给一个长度为n的字符串,Q次询问,每次询问\((l,r,k)\) , 回答子串\(s_ls_{l+1}\cdots s_r\) 第\(k\) 次出现的位置,若不存在输出-1.\(n\le 1e5 ...

  2. 博弈论入门——Nim游戏引入

    说实话,我真的对这个游戏看得是一脸懵逼,因为(我太弱了)我没有明白一些变量的意思,所以一直很懵,现在才明白,这让我明白博弈论(还可以骗钱)博大精深; 以下是我自己思考的过程,也许不严谨,但是最终明白了 ...

  3. Codeforces Round #171 (Div. 2) B. Books (模拟队列)

    题意:有一组数,问子数组和最大不超过\(t\)的最多元素个数. 题解:用数组模拟队列,不断的往里面放,队列中的元素之和大于\(t\),就不断地从队头弹出直到满足条件,维护一个最大值即可. 代码: in ...

  4. Educational Codeforces Round 95 (Rated for Div. 2) B. Negative Prefixes (贪心,构造)

    题意:给你一串长度为\(n\)的序列,有的位置被锁上了,你可以对没锁的位置上的元素任意排序,使得最后一个\(\le0\)的前缀和的位置最小,求重新排序后的序列. 题解:贪心,将所有能动的位置从大到小排 ...

  5. 2019-2020 ACM-ICPC Brazil Subregional Programming Contest Problem A Artwork (并查集)

    题意:有一个矩形,有\(k\)个警报器,警报器所在半径\(r\)内不能走,问是否能从左上角走到右下角. 题解:用并查集将所有相交的圆合并,那么不能走的情况如下图所示 所以最后查询判断一下即可. 代码: ...

  6. PowerShell随笔10---Module

    我们总不能把所有脚本文件零散的放着,就像创建一个项目一样,我们创建一个Module. 创建好之后如何让系统认识我们创建的Module呢,我们需要放在指定的位置. 那到底应该放在什么位置呢? 我们可以通 ...

  7. Dubbo从入门到实践

    1 Dubbo出现的背景 随着互联网的发展,网站应用的规模不断扩大,常规的垂直应用架构已无法应对,分布式服务架构以及流动计算架构势在必行,亟需一个治理系统确保架构有条不紊的演进. 我们传统的网站结构为 ...

  8. hautoj 1268 小天使改名

    1268: 小天使改名 时间限制: 2 秒  内存限制: 128 MB提交: 437  解决: 123提交 状态 题目描述 小天使的b站帐号被大家发现啦.于是小天使决定改名,将他原有ID中的两个不同位 ...

  9. PicGo:搭建图床

    PicGo:搭建图床 PicGo 免费搭建个人图床工具PicGo: 支持Windows.MacOS 和 Linux 软件目前覆盖的图床有8个平台: SM.MS图床.腾讯云COS.GitHub图床.七牛 ...

  10. 论文 公式 排版 编辑 工具 NoteExpress & MathType

    1 1 论文 公式 排版 编辑 工具: NoteExpress http://www.noteexpress.net/index2.html 1 MathType http://www.dessci. ...