Python代码实现Flippy Bird-程序员宅基地

技术标签: python  java  游戏开发  

140行代码实现Flippy Bird
话说这游戏中文名叫什么来着,死活想不起来了,算了话不多说,140行实现小游戏系列第二章,依然是简单小游戏,与数独游戏相比,在游戏界面显示上更难一些,但是在逻辑方面更简单一些,需要处理的无非是速度、加速度、时间、位置、碰撞检测,界面方面则要实现整个动态的显示;

依旧在最后会给出全部代码,不过依然可以从我的 Github仓库 Fork下来直接运行,图片资源也在那里,have fun.

运行以及玩法:

python main.py运行游戏;
鼠标点击是暂停,再点击则是继续;
空格键进行跳跃;
后续扩展:

管道的出现可以更加随机,包括位置和长度等,目前是很简单的方式出现;
游戏速度可以越来越快,目前是固定的;
小鸟的自由落体速度、跳跃速度等需要优化,目前操作感觉没有那么流畅;
增加计分系统、开始、重来等按钮;
小鸟图,需要的自取
游戏截图
进行中
在这里插入图片描述
暂停时
在这里插入图片描述
死亡时
在这里插入图片描述
关键代码分析
随时间移动的管道
可以看到对于这个游戏,实际上移动的是管道而不是小鸟,因此这里主要是处理管道绘制的位置变化,以及整个一个循环的过程,如果屏幕上显示的管道是N个,那么可以想象是N+2个管道在不停地转圈圈出现在我们的界面上就行了;

tunnel_list = [x-speed if x-speed>-200 else 2100 for x in tunnel_list ]

def draw_tunnel():
for x in tunnel_list:
    pygame.draw.rect(screen,COLORS['darkgreen'],(x,0,100,350),0)
    pygame.draw.rect(screen,COLORS['darkgreen'],(x+100,550,100,350),0)

自由落体的小鸟和点击空格后跳起
不操作的情况下,小鸟的上下移动是做自由落体,也就是越来越快的下降的过程,而当我们点击 空格 进行跳跃后,实际上改变的就是小鸟的当前速度,因此小鸟会向上越来越慢的跳跃,直到速度为0后,继续下降,符合基本的物理规则;

G = 9.8*30 # g
JUMP_V = -300
bird_x,bird_y = 700,450
bird_v = 0

if not jump:
    bird_v += G*frame
else:
bird_v = JUMP_V
jump = False
bird_y += frame*bird_v

def draw_bird():
screen.blit(birdImg,[bird_x,bird_y])

碰撞检测
检测小鸟是否碰到管道或者是掉到地上,这么说是只无脚鸟咯,实际上就是检测两个矩形是否有重叠的部分;

def rect_cover(rect1,rect2,up=True):
# bird
left_up1 = (rect1[0],rect1[1])
left_down1 = (rect1[0],left_up1[1]+rect1[3])
right_up1 = (left_up1[0]+rect1[2],rect1[1])
right_down1 = (left_up1[0]+rect1[2],left_up1[1]+rect1[3])
# tunnel
left_up2 = (rect2[0],rect2[1])
left_down2 = (rect2[0],left_up2[1]+rect2[3])
right_up2 = (left_up2[0]+rect2[2],rect2[1])
right_down2 = (left_up2[0]+rect2[2],left_up2[1]+rect2[3])
# check
if (left_up2[0]<=right_up1[0]<=right_up2[0]): # x,肯定是右侧线接触,因此判断bird的right即可
    if up and (left_up2[1]<=right_up1[1]<=left_down2[1]):
        return True
    elif (not up) and (left_up2[1]<=right_down1[1]<=left_down2[1]):
        return True
return False

pygame绘制rect不支持透明度下实现的透明图层
看到,实际上是借助了Surface,将其设置为想要的透明度后blit到我们的screen上即可,直接draw rect是不支持RGBA的A设置alpha的,不知道为啥这么坑爹的设计;

def draw_dead():
s = pygame.Surface(SIZE, pygame.SRCALPHA)
s.fill((255,255,255,240))
screen.blit(s, (0,0))
txt = font120.render('YOU DEAD',True,COLORS['black'])
x,y = 450,400
screen.blit(txt,(x,y))

全部代码

import sys

import pygame
from pygame.color import THECOLORS as COLORS

def draw_background():
# white background
screen.fill(COLORS['lightblue'])
pygame.draw.rect(screen,COLORS['black'],(-100,902,3000,200),5)

def draw_tunnel():
for x in tunnel_list:
    pygame.draw.rect(screen,COLORS['darkgreen'],(x,0,100,350),0)
    pygame.draw.rect(screen,COLORS['darkgreen'],(x+100,550,100,350),0)

def draw_bird():
screen.blit(birdImg,[bird_x,bird_y])

def draw_context():
txt = font50.render('Count time: '+str(int(count_time))+' S',True,COLORS['black'])
x,y = 10,920
screen.blit(txt,(x,y))

def draw_pause():
s = pygame.Surface(SIZE, pygame.SRCALPHA)
s.fill((255,255,255,220))
screen.blit(s, (0,0))
txt = font120.render('PAUSE',True,COLORS['darkgray'])
x,y = 550,400
screen.blit(txt,(x,y))

def draw_dead():
s = pygame.Surface(SIZE, pygame.SRCALPHA)
s.fill((255,255,255,240))
screen.blit(s, (0,0))
txt = font120.render('YOU DEAD',True,COLORS['black'])
x,y = 450,400
screen.blit(txt,(x,y))

def rect_cover(rect1,rect2,up=True):
# bird
left_up1 = (rect1[0],rect1[1])
left_down1 = (rect1[0],left_up1[1]+rect1[3])
right_up1 = (left_up1[0]+rect1[2],rect1[1])
right_down1 = (left_up1[0]+rect1[2],left_up1[1]+rect1[3])
# tunnel
left_up2 = (rect2[0],rect2[1])
left_down2 = (rect2[0],left_up2[1]+rect2[3])
right_up2 = (left_up2[0]+rect2[2],rect2[1])
right_down2 = (left_up2[0]+rect2[2],left_up2[1]+rect2[3])
# check
if (left_up2[0]<=right_up1[0]<=right_up2[0]): # x,肯定是右侧线接触,因此判断bird的right即可
    if up and (left_up2[1]<=right_up1[1]<=left_down2[1]):
        return True
    elif (not up) and (left_up2[1]<=right_down1[1]<=left_down2[1]):
        return True
return False


def check_dead():
bird_rect = (bird_x,bird_y,70,70)
if bird_rect[1]+bird_rect[3]>900:
    return True
for x in tunnel_list:
    up_rect = (x,0,100,350)
    down_rect = (x+100,550,100,350)
    if rect_cover(bird_rect,up_rect) or rect_cover(bird_rect,down_rect,up=False):
        return True
return False

if __name__ == "__main__":
# init pygame
pygame.init()

# contant
SIZE = [1500,1000]
font50 = pygame.font.SysFont('Times', 50)
font120 = pygame.font.SysFont('Times', 120)
G = 9.8*30 # g
JUMP_V = -300

# brid
birdPath = 'bird.png'
birdImg = pygame.image.load(birdPath)

# tunnel
tunnel_list = [100,600,1100,1600,2100]

# create screen 500*500
screen = pygame.display.set_mode(SIZE)

# variable parameter
bird_x,bird_y = 700,450
bird_v = 0
count_time = 0

# level
speed = 5
frame = 0.02

# main loop
running = True
pause = False
jump = False
dead = False
while running:
    for event in pygame.event.get():
        if event.type == pygame.QUIT:
            running = False
            break
        elif event.type == pygame.MOUSEBUTTONDOWN:
            pause = not pause
        elif event.type == pygame.KEYUP:
            if chr(event.key) == ' ':
                jump = True

    # update data
    if not pause and not dead:
        count_time += frame
        tunnel_list = [x-speed if x-speed>-200 else 2100 for x in tunnel_list ]

        if not jump:
            bird_v += G*frame
        else:
            bird_v = JUMP_V
            jump = False
        bird_y += frame*bird_v

    # background
    draw_background()
    # tunnel
    draw_tunnel()
    # choose item
    draw_bird()
    # point
    draw_context()
    # pause
    if not dead and pause:
        draw_pause()
    # dead
    if dead:
        draw_dead()
    # flip
    pygame.display.flip()

    # pause 20ms
    pygame.time.delay(int(frame*1000))

    # check win or not
    if check_dead():
        #print('You dead, dumb ass!!!')
        #break
        dead = True

pygame.quit()

最后
数独和FlippyBird都在 这里 ,欢迎大家Fork下来直接运行,后续会不定期更新其他小游戏,目前以简单的动作小游戏、棋牌类为主,想到啥做啥,或者大家有什么想做想玩的可以评论区告诉我哈,搞得定的我会尽快完成;

最后的最后
大家可以到我的Github上看看有没有其他需要的东西,目前主要是自己做的机器学习项目、Python各种脚本工具、有意思的小项目以及Follow的大佬、Fork的项目等:

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_46577306/article/details/105497516

智能推荐

ActivityMQ介绍与使用-程序员宅基地

文章浏览阅读2.5k次。这里写自定义目录标题1.什么是消息中间件(1)ActiveMQ(2)RabbitMQ(3)ZeroMQ(4)Kafka2.JMS简介2.1什么是JMS2.2 JMS消息传递类型3.ActiveMQ下载与安装3.1下载3.2安装(Linux)3.3启动欢迎使用Markdown编辑器新的改变功能快捷键合理的创建标题,有助于目录的生成如何改变文本的样式插入链接与图片如何插入一段漂亮的代码片生成一个适合你..._activitymq

安卓4.2,4.4 安装失败,安卓5.0以上运行没问题_tivimate安卓4.4.2不能安装-程序员宅基地

文章浏览阅读565次。原先跑得好好的项目,在studio3里面build出来后居然在好几台Android 4.2的机器上运行不了。(Android 5以上的的机器就没发现问题。)都是已启动就爆错误:java.lang.RuntimeException: Unable to get provider com.alibaba.sdk.android.feedback.util.FBFileProvider: java.lang.ClassNotFoundException: Didn't find class "com.a_tivimate安卓4.4.2不能安装

织梦(dedecms)仿站教程第十六讲—首页其它部分及站内调查-程序员宅基地

文章浏览阅读182次。接上一讲,继续我们的仿站之路。打开index.htm文档,接着看代码,接下来是论坛新帖,这里我们不讨论论坛的调用,我们改为调用一个特定的栏目,所以要参考织梦仿站系列教程第十三讲——特定栏目调用 。接下来是热门信息,我们细讲一下,找到如下代码:&lt;DIV class=right_a&gt; &lt;UL&gt; &lt;LI class=mall_hover&gt; ..._dedecms 在线调查

高斯函数及其各阶导数_高斯分布求导-程序员宅基地

文章浏览阅读2w次。高斯函数及其各阶导数在图像处理中经常要用到高斯函数,高斯滤波是典型的低通滤波,对图像有平滑作用。高斯函数的一阶、二阶导数也可以进行高通滤波,比如canny算子中用到的是高斯函数的一阶导数,LOG算子中用到的是高斯函数的二阶导数。高斯函数的相关公式如下所示:  一维和二维高斯函数表达式分别为:       二维高斯函数的一阶偏导数表达式为:    二维高斯函数的二阶偏导数表达_高斯分布求导

dbms_backup_restore恢复数据库-程序员宅基地

文章浏览阅读699次。一、查看RMAN备份参数[oracle@oraserver ~]$ rman target / Recovery Manager: Release 10.2.0.1.0 - Productionon Mon Aug 15 13:21:54 2011 Copyright (c) 1982, 2005, Oracle. All rights reserved. conne

Android基础之SQLite数据库及内容提供器-程序员宅基地

文章浏览阅读686次。哎,当失眠成为一种习惯,那就无所谓啦!!!今天主要学习了SQLite数据库以及内容提供器的相关知识一、SQLite数据库SQLite数据库是安卓自带的数据库,不仅支持标准的SQL数据,还支持ACID事务。为了让我们更加方便的管理数据库,专门提供了一个SQLiteOpenHelper帮助类,借助这个类我们可以很轻松的对数据库进行创建和升级。由于SQLiteOpenHelper

随便推点

oracle定时任务报“并非所有变量都已经绑定”以及创建,删除等_并非所有变量都已绑定怎么解决 定时任务-程序员宅基地

文章浏览阅读1.5k次,点赞2次,收藏5次。-- 查询该存储过程的定时任务select job, next_date, next_sec, failures,what, broken from dba_jobs where what like '%PRO_BD_MO_DEG_EVAL%'; var job number;begin sys.dbms_job.submit(job => :job, what =&..._并非所有变量都已绑定怎么解决 定时任务

zDialog 的使用-程序员宅基地

文章浏览阅读86次。内容来源:http://www.cool80.com/Demonstrate/465/提取自ZCMS的弹出框:代替window.open、window.alert、window.confirm;提供良好的用户体验;水晶质感,设计细腻,外观漂亮;兼容ie6/7/8、firefox2/3、Opera;弹出框在ie6下不会被select控件穿透;无外部css文件,引用Dialog...

儒猿秒杀季!互联网大厂的IM即时通信系统项目实战_石杉的架构笔记的博客-程序员宅基地

文章浏览阅读1k次。疯狂秒杀季:499元秒杀原价1899元的《互联网大厂的IM即时通信系统项目实战》今天上午10点,仅40套,先到先得!=== 课程背景 ===随着企业不断发展,业务扩大,人员增加,线下沟通变的越来越困难,为了提高企业内部办公效率、数据安全、信息安全,即时通讯系统由此诞生,自此IM系统成为了互联网大厂最重要的办公软件。=== 课程介绍 ===基于此儒猿推出了《互联...

CSS选择器,颜色表示方法,阴影,圆角,边框实现三角形,按钮_css填充button某个圆弧区域的颜色-程序员宅基地

文章浏览阅读213次。、CSS3属性选择器 [attribute] [target] 选择所有带有target属性元素 2 [attribute=value] [target=-blank] 选择所有使用target="-blank"的元素 2 [attribute~=value] _css填充button某个圆弧区域的颜色

(三)兴趣对于工作的重要性-程序员宅基地

文章浏览阅读6.8k次。兴趣是人生最大的导师,不用关心这句话是谁说的~对于学习、生活、工作,有了积极的兴趣,那会起到至关重要的促进作用。首先,对工作有兴趣,那将是非常幸福的一件事情。因为人一生中几十年都是在忙忙碌碌的工作中度过,对工作不喜欢,甚至厌恶现在的工作,那怎么会有主动积极开展工作的热情,只能是得过且过,浑浑噩噩地度日,更谈不上会有良好的职业发展了。工作的积极也会传导到生活中去,营造更好的生活状态,所以说对于工...

Django速成--创建一个简单的blog-程序员宅基地

文章浏览阅读2.5k次。Django速成--创建一个简单的blog 开启cmd…* 运用:>cd D:\projects 设定保存django项目目录为当前目录。* 运用:>python C:\python25\scripts\django-admin.py mysite 创建一个django项目mysite。* 运用:>python D:\projects\mysite\manage.py ru