[Python] Spaceship游戏初步实现 Mini-project #7

2023-11-07 14:10

本文主要是介绍[Python] Spaceship游戏初步实现 Mini-project #7,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

Spaceship顾名思义是一款太空游戏,受经典的街机游戏Asteroid(1979)启发编写而成。在Mini-project #7中,我们先实现飞船的控制、导弹的发射、星球的随机出现和运动,简单设置UI,从而展示Lives和Score。在Mini-project #8中会增加更多功能。

具体思路如下(注:这次描述不是按照代码显示顺序,而是按照完成的思路写的):

游戏细节描述在Coursera的interactive-python教程中,见此:https://www.coursera.org/learn/interactive-python-2/supplement/2ZVsF/mini-project-description

首先导入所需库,设置全局变量

import simplegui
import math
import random# globals for user interface
WIDTH = 800
HEIGHT = 600
score = 0
lives = 3
time = 0

1. 创建飞船类

# Ship class
class Ship:def __init__(self, pos, vel, angle, image, info):self.pos = [pos[0],pos[1]]self.vel = [vel[0],vel[1]]self.thrust = Falseself.angle = angleself.angle_vel = 0self.image = imageself.image_center = info.get_center()self.image_size = info.get_size()self.radius = info.get_radius()def draw(self,canvas):#canvas.draw_circle(self.pos, self.radius, 1, "White", "White")if self.thrust: #change image statuscanvas.draw_image(self.image, [130, 45], self.image_size, self.pos, self.image_size, self.angle)else:canvas.draw_image(self.image, self.image_center, self.image_size, self.pos, self.image_size, self.angle)def update(self): #这里是重头戏,飞船的加速度改变影响速度,速度也受阻力影响#update angleself.angle += self.angle_velc = 0.2#update positionself.pos[0] = (self.pos[0] + self.vel[0]) % WIDTHself.pos[1] = (self.pos[1] + self.vel[1]) % HEIGHT#update velocityself.vel[0] *= (1-c) #摩擦力self.vel[1] *= (1-c)if self.thrust:forward = angle_to_vector(self.angle)self.vel[0] += forward[0]self.vel[1] += forward[1]#change angle
def set_thrust(self, on):
self.thrust = on
if on: #add sound to thrust
ship_thrust_sound.rewind()
ship_thrust_sound.play()
else:
ship_thrust_sound.pause()

    def increase_angvel(self):self.angle_vel += .05def decrease_angvel(self):self.angle_vel -= .05#add missile and shootingdef shoot(self):global a_missileforward = angle_to_vector(self.angle)pos = [self.pos[0], self.pos[1]]pos[0] += vel[0] + self.radiuspos[1] += vel[1] + self.radiusvel[0] = self.vel[0] + (vel[0] * 6)vel[1] = self.vel[1] + (vel[1] * 6)a_missile = Sprite(pos, vel, self.angle, 0, missile_image, missile_info, missile_sound)


def increase_angvel(self): self.angle_vel += .05 #试合适的值 def decrease_angvel(self): self.angle_vel -= .05 #add missile and shooting 发射导弹 def shoot(self): global a_missile forward = angle_to_vector(self.angle) missile_pos = [self.pos[0]+self.radius*forward[0],self.pos[1]+self.radius*forward[1]] missile_vel = [self.vel[0]+5*forward[0], self.vel[1]+5*forward[1]] a_missile = Sprite(missile_pos, missile_vel, self.angle, 0, missile_image, missile_info, missile_sound) #使用Sprite类,见下面说明

 键盘控制: 

def keydown(key):if key == simplegui.KEY_MAP['left']: #逆时针旋转 所以decrease angle_velocitymy_ship.decrease_angvel()elif key == simplegui.KEY_MAP['right']:my_ship.increase_angvel()elif key == simplegui.KEY_MAP['up']:my_ship.set_thrust(True)elif key == simplegui.KEY_MAP['space']:my_ship.shoot()def keyup(key):if key == simplegui.KEY_MAP["left"]:my_ship.increase_angvel()elif key == simplegui.KEY_MAP['right']:my_ship.decrease_angvel()elif key == simplegui.KEY_MAP['up']:my_ship.set_thrust(False)
要想键盘能控制,在注册处理器(register handlers)时要添加set_keydown_handler和set_keyup_handler(最后代码中有写)

frame.set_keydown_handler(keydown)#添加键盘控制
frame.set_keyup_handler(keyup)
2. 创建Sprite类。星石(rock)、导弹(missile)都会使用该类

# Sprite class
class Sprite:def __init__(self, pos, vel, ang, ang_vel, image, info, sound = None):self.pos = [pos[0],pos[1]]self.vel = [vel[0],vel[1]]self.angle = angself.angle_vel = ang_velself.image = imageself.image_center = info.get_center()self.image_size = info.get_size()self.radius = info.get_radius()self.lifespan = info.get_lifespan()self.animated = info.get_animated()self.age = 0if sound:sound.rewind()sound.play()def draw(self, canvas):# 第二个self.image_size是指按图像大小显示canvas.draw_image(self.image, self.image_center, self.image_size, self.pos, self.image_size, self.angle)def update(self):self.angle += self.angle_vel # make the sprite move and rotateself.pos[0] = (self.pos[0]+self.vel[0])%WIDTHself.pos[1] = (self.pos[1]+self.vel[1])%HEIGHT
创建函数使星石定时随机产生
# timer handler that spawns a rock    
def rock_spawner(): #set a_rock to be a new rock on every tickglobal a_rocka_rock = Sprite([random.randrange(WIDTH),random.randrange(HEIGHT)], [random.randrange(-1,1),random.randrange(-1,1)], 0.05, 0.05, asteroid_image, asteroid_info)
3. 在画面中显示背景、星云等

def draw(canvas):global time#计时# animiate backgroundtime += 1wtime = (time / 4) % WIDTHcenter = debris_info.get_center()size = debris_info.get_size()canvas.draw_image(nebula_image, nebula_info.get_center(), nebula_info.get_size(), [WIDTH / 2, HEIGHT / 2], [WIDTH, HEIGHT])canvas.draw_image(debris_image, center, size, (wtime - WIDTH / 2, HEIGHT / 2), (WIDTH, HEIGHT))canvas.draw_image(debris_image, center, size, (wtime + WIDTH / 2, HEIGHT / 2), (WIDTH, HEIGHT))# draw ship and spritesmy_ship.draw(canvas)a_rock.draw(canvas)a_missile.draw(canvas)# update ship and spritesmy_ship.update()a_rock.update()a_missile.update()
添加UI,显示所剩生命(Lives)和所得分数(Score):

    # draw score and livescanvas.draw_text('Lives:'+ str(lives), (50,50), 28, "Red")canvas.draw_text('Score:'+ str(score), (625,50), 28, "Red")
4. 导入飞船、导弹、星石、星云等图片(这个步骤中的代码写在“创建飞船类”之前,我是按照功能的实现描述的)

在一开始先创建好图片信息类:

class ImageInfo:def __init__(self, center, size, radius = 0, lifespan = None, animated = False):self.center = centerself.size = sizeself.radius = radiusif lifespan:self.lifespan = lifespanelse:self.lifespan = float('inf')self.animated = animateddef get_center(self):return self.centerdef get_size(self):return self.sizedef get_radius(self):return self.radiusdef get_lifespan(self):return self.lifespandef get_animated(self):return self.animated#导入图片    
# art assets created by Kim Lathrop, may be freely re-used in non-commercial projects, please credit Kim# debris images - debris1_brown.png, debris2_brown.png, debris3_brown.png, debris4_brown.png
#碎片               debris1_blue.png, debris2_blue.png, debris3_blue.png, debris4_blue.png, debris_blend.png
debris_info = ImageInfo([320, 240], [640, 480])
debris_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/debris2_blue.png")
#星云
# nebula images - nebula_brown.png, nebula_blue.png
nebula_info = ImageInfo([400, 300], [800, 600])
nebula_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/nebula_blue.f2014.png")
#飞溅
# splash image
splash_info = ImageInfo([200, 150], [400, 300])
splash_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/splash.png")
#飞船
# ship image
ship_info = ImageInfo([45, 45], [90, 90], 35)
ship_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/double_ship.png")
#导弹
# missile image - shot1.png, shot2.png, shot3.png
missile_info = ImageInfo([5,5], [10, 10], 3, 50)
missile_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/shot2.png")
#小行星
# asteroid images - asteroid_blue.png, asteroid_brown.png, asteroid_blend.png
asteroid_info = ImageInfo([45, 45], [90, 90], 40)
asteroid_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/asteroid_blue.png")
#爆炸
# animated explosion - explosion_orange.png, explosion_blue.png, explosion_blue2.png, explosion_alpha.png
explosion_info = ImageInfo([64, 64], [128, 128], 17, 24, True)
explosion_image = simplegui.load_image("http://commondatastorage.googleapis.com/codeskulptor-assets/lathrop/explosion_alpha.png")
#声音
# sound assets purchased from sounddogs.com, please do not redistribute
soundtrack = simplegui.load_sound("http://commondatastorage.googleapis.com/codeskulptor-assets/sounddogs/soundtrack.mp3")
missile_sound = simplegui.load_sound("http://commondatastorage.googleapis.com/codeskulptor-assets/sounddogs/missile.mp3")
missile_sound.set_volume(.5)
ship_thrust_sound = simplegui.load_sound("http://commondatastorage.googleapis.com/codeskulptor-assets/sounddogs/thrust.mp3")
explosion_sound = simplegui.load_sound("http://commondatastorage.googleapis.com/codeskulptor-assets/sounddogs/explosion.mp3")# alternative upbeat soundtrack by composer and former IIPP student Emiel Stopler
# please do not redistribute without permission from Emiel at http://www.filmcomposer.nl
#soundtrack = simplegui.load_sound("https://storage.googleapis.com/codeskulptor-assets/ricerocks_theme.mp3")
设置基本函数
#角度转为向量
# helper functions to handle transformations
def angle_to_vector(ang):return [math.cos(ang), math.sin(ang)]
#p,q距离,(a^2+b^2)^0.5
def dist(p,q):return math.sqrt((p[0] - q[0]) ** 2+(p[1] - q[1]) ** 2)
5. 初始化画面设置与开始

# initialize frame
frame = simplegui.create_frame("Asteroids", WIDTH, HEIGHT)# initialize ship and two sprites
my_ship = Ship([WIDTH / 2, HEIGHT / 2], [0, 0], 0, ship_image, ship_info)
a_rock = Sprite([WIDTH / 3, HEIGHT / 3], [1, 1], 0, 0.05, asteroid_image, asteroid_info) #0.05秒旋转一下
a_missile = Sprite([2 * WIDTH / 3, 2 * HEIGHT / 3], [-1,1], 0, 0, missile_image, missile_info, missile_sound)# register handlers
frame.set_draw_handler(draw)
frame.set_keydown_handler(keydown)#添加键盘控制
frame.set_keyup_handler(keyup)
timer = simplegui.create_timer(1000.0, rock_spawner)# get things rolling
timer.start()
frame.start()
6. 游戏界面展示:

谢谢阅读!欢迎互相学习~

这篇关于[Python] Spaceship游戏初步实现 Mini-project #7的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/364150

相关文章

python: 多模块(.py)中全局变量的导入

文章目录 global关键字可变类型和不可变类型数据的内存地址单模块(单个py文件)的全局变量示例总结 多模块(多个py文件)的全局变量from x import x导入全局变量示例 import x导入全局变量示例 总结 global关键字 global 的作用范围是模块(.py)级别: 当你在一个模块(文件)中使用 global 声明变量时,这个变量只在该模块的全局命名空

hdu1043(八数码问题,广搜 + hash(实现状态压缩) )

利用康拓展开将一个排列映射成一个自然数,然后就变成了普通的广搜题。 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<queue>#include<map>#include<stdio.h>#include<stdlib.h>#include<ctype.h>#inclu

【C++】_list常用方法解析及模拟实现

相信自己的力量,只要对自己始终保持信心,尽自己最大努力去完成任何事,就算事情最终结果是失败了,努力了也不留遗憾。💓💓💓 目录   ✨说在前面 🍋知识点一:什么是list? •🌰1.list的定义 •🌰2.list的基本特性 •🌰3.常用接口介绍 🍋知识点二:list常用接口 •🌰1.默认成员函数 🔥构造函数(⭐) 🔥析构函数 •🌰2.list对象

【Prometheus】PromQL向量匹配实现不同标签的向量数据进行运算

✨✨ 欢迎大家来到景天科技苑✨✨ 🎈🎈 养成好习惯,先赞后看哦~🎈🎈 🏆 作者简介:景天科技苑 🏆《头衔》:大厂架构师,华为云开发者社区专家博主,阿里云开发者社区专家博主,CSDN全栈领域优质创作者,掘金优秀博主,51CTO博客专家等。 🏆《博客》:Python全栈,前后端开发,小程序开发,人工智能,js逆向,App逆向,网络系统安全,数据分析,Django,fastapi

让树莓派智能语音助手实现定时提醒功能

最初的时候是想直接在rasa 的chatbot上实现,因为rasa本身是带有remindschedule模块的。不过经过一番折腾后,忽然发现,chatbot上实现的定时,语音助手不一定会有响应。因为,我目前语音助手的代码设置了长时间无应答会结束对话,这样一来,chatbot定时提醒的触发就不会被语音助手获悉。那怎么让语音助手也具有定时提醒功能呢? 我最后选择的方法是用threading.Time

【Python编程】Linux创建虚拟环境并配置与notebook相连接

1.创建 使用 venv 创建虚拟环境。例如,在当前目录下创建一个名为 myenv 的虚拟环境: python3 -m venv myenv 2.激活 激活虚拟环境使其成为当前终端会话的活动环境。运行: source myenv/bin/activate 3.与notebook连接 在虚拟环境中,使用 pip 安装 Jupyter 和 ipykernel: pip instal

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

C#实战|大乐透选号器[6]:实现实时显示已选择的红蓝球数量

哈喽,你好啊,我是雷工。 关于大乐透选号器在前面已经记录了5篇笔记,这是第6篇; 接下来实现实时显示当前选中红球数量,蓝球数量; 以下为练习笔记。 01 效果演示 当选择和取消选择红球或蓝球时,在对应的位置显示实时已选择的红球、蓝球的数量; 02 标签名称 分别设置Label标签名称为:lblRedCount、lblBlueCount

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略

Kubernetes PodSecurityPolicy:PSP能实现的5种主要安全策略 1. 特权模式限制2. 宿主机资源隔离3. 用户和组管理4. 权限提升控制5. SELinux配置 💖The Begin💖点点关注,收藏不迷路💖 Kubernetes的PodSecurityPolicy(PSP)是一个关键的安全特性,它在Pod创建之前实施安全策略,确保P