python实现浪漫的烟花秀


Posted in Python onJanuary 30, 2019

无意中看到一段用Tkinter库写的放烟花的程序,就跟着跑了一遍。

设计理念:通过让画面上一个粒子分裂为X数量的粒子来模拟爆炸效果。粒子会发生“膨胀”,意思是它们会以恒速移动且相互之间的角度相等。这样就能让我们以一个向外膨胀的圆圈形式模拟出烟花绽放的画面。经过一定时间后,粒子会进入“自由落体”阶段,也就是由于重力因素它们开始坠落到地面,仿若绽放后熄灭的烟花。

python实现浪漫的烟花秀

首先我们写一个粒子类,表示烟花事件中的每个粒子,包含大小,颜色,位置,速度等属性以及粒子经历的三个阶段的函数,即:膨胀、坠落、消失。

'''
particles 类
粒子在空中随机生成随机,变成一个圈、下坠、消失
属性:
 - id: 粒子的id
 - x, y: 粒子的坐标
 - vx, vy: 在坐标的变化速度
 - total: 总数
 - age: 粒子存在的时长
 - color: 颜色
 - cv: 画布
 - lifespan: 最高存在时长
'''
 
 
class Particle:
 
 def __init__(self, cv, idx, total, explosion_speed, x=0., y=0., vx=0., vy=0., size=2., color='red', lifespan=2,
     **kwargs):
  self.id = idx
  self.x = x
  self.y = y
  self.initial_speed = explosion_speed
  self.vx = vx
  self.vy = vy
  self.total = total
  self.age = 0
  self.color = color
  self.cv = cv
  self.cid = self.cv.create_oval(
   x - size, y - size, x + size,
   y + size, fill=self.color)
  self.lifespan = lifespan
 
 def update(self, dt):
  self.age += dt
 
  # 粒子范围扩大
  if self.alive() and self.expand():
   move_x = cos(radians(self.id * 360 / self.total)) * self.initial_speed
   move_y = sin(radians(self.id * 360 / self.total)) * self.initial_speed
   self.cv.move(self.cid, move_x, move_y)
   self.vx = move_x / (float(dt) * 1000)
 
  # 以自由落体坠落
  elif self.alive():
   move_x = cos(radians(self.id * 360 / self.total))
   # we technically don't need to update x, y because move will do the job
   self.cv.move(self.cid, self.vx + move_x, self.vy + GRAVITY * dt)
   self.vy += GRAVITY * dt
 
  # 移除超过最高时长的粒子
  elif self.cid is not None:
   cv.delete(self.cid)
   self.cid = None
 
 # 扩大的时间
 def expand (self):
  return self.age <= 1.2
 
 # 粒子是否在最高存在时长内
 def alive(self):
  return self.age <= self.lifespan

接下来我们需要创建一列列表,每个子列表是一个烟花,其包含一列粒子,每个列表中的粒子有相同的x,y坐标、大小、颜色、初始速度。

源码如下:

import tkinter as tk
from PIL import Image, ImageTk
from time import time, sleep
from random import choice, uniform, randint
from math import sin, cos, radians
 
# 模拟重力
GRAVITY = 0.05
# 颜色选项(随机或者按顺序)
colors = ['red', 'blue', 'yellow', 'white', 'green', 'orange', 'purple', 'seagreen', 'indigo', 'cornflowerblue']
 
'''
particles 类
粒子在空中随机生成随机,变成一个圈、下坠、消失
属性:
 - id: 粒子的id
 - x, y: 粒子的坐标
 - vx, vy: 在坐标的变化速度
 - total: 总数
 - age: 粒子存在的时长
 - color: 颜色
 - cv: 画布
 - lifespan: 最高存在时长
'''
 
 
class Particle:
 
 def __init__(self, cv, idx, total, explosion_speed, x=0., y=0., vx=0., vy=0., size=2., color='red', lifespan=2,
     **kwargs):
  self.id = idx
  self.x = x
  self.y = y
  self.initial_speed = explosion_speed
  self.vx = vx
  self.vy = vy
  self.total = total
  self.age = 0
  self.color = color
  self.cv = cv
  self.cid = self.cv.create_oval(
   x - size, y - size, x + size,
   y + size, fill=self.color)
  self.lifespan = lifespan
 
 def update(self, dt):
  self.age += dt
 
  # 粒子范围扩大
  if self.alive() and self.expand():
   move_x = cos(radians(self.id * 360 / self.total)) * self.initial_speed
   move_y = sin(radians(self.id * 360 / self.total)) * self.initial_speed
   self.cv.move(self.cid, move_x, move_y)
   self.vx = move_x / (float(dt) * 1000)
 
  # 以自由落体坠落
  elif self.alive():
   move_x = cos(radians(self.id * 360 / self.total))
   # we technically don't need to update x, y because move will do the job
   self.cv.move(self.cid, self.vx + move_x, self.vy + GRAVITY * dt)
   self.vy += GRAVITY * dt
 
  # 移除超过最高时长的粒子
  elif self.cid is not None:
   cv.delete(self.cid)
   self.cid = None
 
 # 扩大的时间
 def expand (self):
  return self.age <= 1.2
 
 # 粒子是否在最高存在时长内
 def alive(self):
  return self.age <= self.lifespan
 
'''
循环调用保持不停
'''
def simulate(cv):
 t = time()
 explode_points = []
 wait_time = randint(10, 100)
 numb_explode = randint(6, 10)
 # 创建一个所有粒子同时扩大的二维列表
 for point in range(numb_explode):
  objects = []
  x_cordi = randint(50, 550)
  y_cordi = randint(50, 150)
  speed = uniform(0.5, 1.5)
  size = uniform(0.5, 3)
  color = choice(colors)
  explosion_speed = uniform(0.2, 1)
  total_particles = randint(10, 50)
  for i in range(1, total_particles):
   r = Particle(cv, idx=i, total=total_particles, explosion_speed=explosion_speed, x=x_cordi, y=y_cordi,
       vx=speed, vy=speed, color=color, size=size, lifespan=uniform(0.6, 1.75))
   objects.append(r)
  explode_points.append(objects)
 
 total_time = .0
 # 1.8s内一直扩大
 while total_time < 1.8:
  sleep(0.01)
  tnew = time()
  t, dt = tnew, tnew - t
  for point in explode_points:
   for item in point:
    item.update(dt)
  cv.update()
  total_time += dt
 # 循环调用
 root.after(wait_time, simulate, cv)
 
 
def close(*ignore):
 """退出程序、关闭窗口"""
 global root
 root.quit()
 
 
if __name__ == '__main__':
 root = tk.Tk()
 cv = tk.Canvas(root, height=360, width=480)
 # 选一个好看的背景会让效果更惊艳!
 image = Image.open("./image.jpg")
 photo = ImageTk.PhotoImage(image)
 
 cv.create_image(0, 0, image=photo, anchor='nw')
 cv.pack()
 
 root.protocol("WM_DELETE_WINDOW", close)
 root.after(100, simulate, cv)
 root.mainloop()

效果图(背景请忽略哈哈):

python实现浪漫的烟花秀

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
bpython 功能强大的Python shell
Feb 16 Python
使用Python写一个贪吃蛇游戏实例代码
Aug 21 Python
numpy中实现二维数组按照某列、某行排序的方法
Apr 04 Python
Python面向对象程序设计OOP入门教程【类,实例,继承,重载等】
Jan 05 Python
python实现Dijkstra静态寻路算法
Jan 17 Python
pycharm new project变成灰色的解决方法
Jun 27 Python
python读取word 中指定位置的表格及表格数据
Oct 23 Python
Python对wav文件的重采样实例
Feb 25 Python
解决django migrate报错ORA-02000: missing ALWAYS keyword
Jul 02 Python
Python 使用生成器代替线程的方法
Aug 04 Python
Python+Selenium随机生成手机验证码并检查页面上是否弹出重复手机号码提示框
Sep 21 Python
一文搞懂如何实现Go 超时控制
Mar 30 Python
新年快乐! python实现绚烂的烟花绽放效果
Jan 30 #Python
python+selenium 定位到元素,无法点击的解决方法
Jan 30 #Python
解决Python selenium get页面很慢时的问题
Jan 30 #Python
对python实现模板生成脚本的方法详解
Jan 30 #Python
ActiveMQ:使用Python访问ActiveMQ的方法
Jan 30 #Python
python 发送和接收ActiveMQ消息的实例
Jan 30 #Python
Python批量生成特定尺寸图片及图画任意文字的实例
Jan 30 #Python
You might like
PHP+Javascript实现在线拍照功能实例
2015/07/18 PHP
javascript oop开发滑动(slide)菜单控件
2010/08/25 Javascript
Javascript跨域请求的4种解决方式
2013/03/17 Javascript
在javascript中如何得到中英文混合字符串的长度
2014/01/17 Javascript
直接在JS里创建JSON数据然后遍历使用
2014/07/25 Javascript
jQuery实现的向下图文信息滚动效果
2015/05/03 Javascript
JavaScript中字符串(string)转json的2种方法
2015/06/25 Javascript
JavaScript学习笔记整理_关于表达式和语句
2016/09/19 Javascript
预防网页挂马的方法总结
2016/11/03 Javascript
基于javascript实现按圆形排列DIV元素(三)
2016/12/02 Javascript
JS前端笔试题分析
2016/12/19 Javascript
详解jquery插件jquery.viewport.js学习使用方法
2017/09/08 jQuery
基于Vuejs的搜索匹配功能实现方法
2018/03/03 Javascript
Vue插件打包与发布的方法示例
2018/08/20 Javascript
一文了解Vue中的nextTick
2019/05/06 Javascript
Python中字典和JSON互转操作实例
2015/01/19 Python
python中split方法用法分析
2015/04/17 Python
十个Python程序员易犯的错误
2015/12/15 Python
Python numpy生成矩阵、串联矩阵代码分享
2017/12/04 Python
Python实现一个服务器监听多个客户端请求
2018/04/12 Python
ActiveMQ:使用Python访问ActiveMQ的方法
2019/01/30 Python
Python后台开发Django的教程详解(启动)
2019/04/08 Python
python+OpenCV实现图像拼接
2020/03/05 Python
python 对象真假值的实例(哪些视为False)
2020/12/11 Python
canvas实现滑动验证的实现示例
2020/08/11 HTML / CSS
JAVA代码查错题
2014/10/10 面试题
2019年.net常见面试问题
2012/02/12 面试题
毕业求职自荐信格式是什么
2013/11/19 职场文书
《寓言两则》教学反思
2014/02/27 职场文书
幼儿园评语大全
2014/04/17 职场文书
厨师个人自我鉴定范文
2014/04/19 职场文书
新教师培训心得体会
2014/09/02 职场文书
2016春季田径运动会广播稿
2015/12/21 职场文书
2016年最美孝心少年事迹材料
2016/02/26 职场文书
Python办公自动化之教你用Python批量识别发票并录入到Excel表格中
2021/06/26 Python
Java实现经典游戏泡泡堂的示例代码
2022/04/04 Java/Android