新年快乐! python实现绚烂的烟花绽放效果


Posted in Python onJanuary 30, 2019

做了一个Python的小项目。利用了一点python的可视化技巧,做出烟花绽放的效果,文章的灵感来自网络上一位大神。

一.编译环境

Pycharm

二.模块

1.tkinter:这个小项目的主角,是一个python图形模块。且Python3已经自带了该模块,不用另外安装。它有点像java中的swing图形模块(由众多组件集成,组件通过创建实例添加,组件通过坐标定位在窗口上)。

2.PIL:Python Imaging Library,是Python平台的图像处理标准模块。在Python3也是自带的,在这个项目中用于背景图片的导入。

3.time:相信这个模块大家都不会陌生,导入它用来控制烟花的绽放,坠落及消失时间。

4.random:随机数模块,用于生成烟花随机坐标点,随机绽放速度,随机消失时间。

5.math:这个模块大家应该也很熟悉了,导入它的目的是使烟花绽放的粒子以一定角度散开。

三.效果

新年快乐! python实现绚烂的烟花绽放效果

项目最终实现的效果就跟上面一样了。

四.代码

以下是我学习(copy)了那位大神的代码,再添加了一些浅显的注解。

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.5
#listof colors,can choose randomly or use as queue(FIFO
colors=['red','blue','yellow','white','green','orange','purple','seagreen','indigo','cornflowerblue']
'''
create a class for particles粒子
particles are emitted almost randomly on the sky,
forming around(组成一个圈) of circle(a star)before falling and getting removed from canvas

Attributes(属性):
id:每个特定烟花的标识符
x,y:烟花的绽放坐标
vx,vy:烟花的绽放速度
total:一颗烟花里的星星总数
age:一颗星星会在画布上停留多久
color:自我移植
cv:画布
lifespan:星星在画布上停留的最后时间
'''
class part:#为每一个烟花绽放出来的粒子单独构建一个类的对象
 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#烟花的绽放x轴
  self.y=y#烟花的绽放x轴
  self.initial_speed=explosion_speed#初速度
  self.vx=vx#外放x轴速度
  self.vy=vy#外放y轴速度
  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)#create_oval()创建一个椭圆,参数为左上x,左上y,右下x,右下y,填满的颜色,该函数返回一个id
  self.lifespan=lifespan#应该停留时间
 def update(self,dt):#更新数据,已停留时间增加
  self.age+=dt
  #粒子膨胀
  if self.alive() and self.expand():#如果停留时间(2s)足够&&膨胀时间(1.2s)足够
   move_x=cos(radians(self.id*360/self.total))*self.initial_speed#粒子的x轴继续膨胀
   move_y=sin(radians(self.id*360/self.total))*self.initial_speed#粒子的y轴继续膨胀
   self.cv.move(self.cid, move_x, move_y)#根据id把画布上的粒子移动x和y个距离
   self.vx=move_x/(float(dt)*1000)
  #以自由落体坠落
  elif self.alive():#如果只是停留时间足够,说明膨胀到最大了,应该准备下坠
   move_x=cos(radians(self.id*360/self.total))#x轴继续膨胀
   self.cv.move(self.cid,self.vx+move_x,self.vy+GRAVITY*dt)#而y轴按照重力因素做落体运动,但实际上这个重力是v而不是a
   self.vy+=GRAVITY*dt#更新一下y轴

  elif self.cid is not None:#如果粒子的生命周期已过,就将其移除
   cv.delete(self.cid)#删除该粒子对象
   self.cid=None
 #定义膨胀效果的时间帧
 def expand(self):
  return self.age<=1.2#膨胀时间小于1.2s
 #检查粒子是否仍在生命周期内
 def alive(self):#已停留时间是不是比应该停留时间短
  return self.age<=self.lifespan
'''
烟花模拟回路:
递归调用来在画布上重复发出新的烟火
通过每个“部件”对象内部的更新协议,每次调用时都要在画布上创建并绘制列表(星列表,每个星列表成员都是粒子列表)来重复地在画布上发出新的焰火
'''
#生成新的一轮爆炸
def simulate(cv):
 t=time()#time()函数返回自1970年后经过的浮点秒数,精确到小数点后6位
 explode_points=[]#爆炸点列表--烟花列表
 wait_time=randint(10,100)#随机生成一个int n,10<=n<=100
 numb_explode=randint(6,10)#爆炸的个数是6~10
 #为所有模拟烟花绽放的全部例子创建一列列表
 for point in range(numb_explode):#遍历爆炸的个数
  objects=[]#这是每个点的爆炸粒子列表
  x_cordi=randint(50,550)#每个点的爆炸x轴
  y_cordi=randint(50,150)#爆炸y轴
  speed=uniform(0.5,1.5)#随机生成一个float speed,0.5<=speed<1.5
  size=uniform(0.5,3)#随机生成一个float size,0.5<=size<3
  color=choice(colors)#choice()是python内置函数,随机返回元组,列表,或字符串的一个成员
  explosion_speed=uniform(0.2,1)#爆炸的绽放速度也要随机出来
  total_particles=randint(10,50)#爆炸出来的粒子数半径也随机
  for i in range(1,total_particles):#同一个烟花爆炸出来的粒子的大小,速度,坐标都是相同的
   r = part(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#先把时间置0
 #在1.8秒时间帧内保持更新
 while total_time<1.8:
  sleep(0.01)#让画面暂停0.01s
  tnew=time()#刷新时间
  t,dt=tnew,tnew-t#时间等于新时间,与上次时间间隔为tnew-t
  for point in explode_points:#遍历烟花列表
   for item in point:#遍历烟花里的粒子列表
    item.update(dt)#更新时间
  cv.update()#刷新页面
  total_time+=dt#为上面的while循环增加时间
 root.after(wait_time,simulate,cv)#将组件置于其他组件之后,就是放在最顶层,覆盖下面的,这里递归第哦啊用了自己,形成新的一轮爆炸
def close(*ignore):
 #打开模拟循环并关闭窗口
 global root
 root.quit()

if __name__=="__main__":
 root=tk.Tk()
 cv=tk.Canvas(root,height=600,width=700)#绘制一个画布
 #绘制一个黑色背景
 #cv.create_rectangle(0,0,600,600,fill="black")
 #use a nice background image
 image=Image.open("L:\PyCharm Community Edition 2018.2.2\Python_program\image.jpg")
 photo=ImageTk.PhotoImage(image)
 cv.create_image(0,0,image=photo,anchor='nw')#在画板上绘制一张图片
 cv.pack()#把cv添加进去
 root.protocol("WM_DELETE_WINDOW",close)
 #在0.1s后才开始调用stimulate()
 root.after(100,simulate,cv)#调用stimulate生成一轮烟花绽放效果
 root.mainloop()#执行root,生成窗口
#让我更加明白了python的一切皆对象

五.结尾

这篇博客暂时就这样,如果有机会会重新做一遍,不过最大的可能就是遥遥无期了。先打算对项目做如下修改:

1.增加鼠标交互功能,用户鼠标点击位置额外生成烟花绽放;

2.增加文字绘制功能,在背景图中隐藏有字样,烟花绽放后落于字样中的粒子不消失,而是定格在窗口中。

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

Python 相关文章推荐
python中闭包Closure函数作为返回值的方法示例
Dec 17 Python
python使用json序列化datetime类型实例解析
Feb 11 Python
Python3.6笔记之将程序运行结果输出到文件的方法
Apr 22 Python
Python基于jieba库进行简单分词及词云功能实现方法
Jun 16 Python
Python爬虫之正则表达式基本用法实例分析
Aug 08 Python
Python模拟百度自动输入搜索功能的实例
Feb 14 Python
pandas DataFrame创建方法的方式
Aug 02 Python
python pyinstaller打包exe报错的解决方法
Nov 02 Python
python循环嵌套的多种使用方法解析
Nov 29 Python
给 TensorFlow 变量进行赋值的方式
Feb 10 Python
Python常见反爬虫机制解决方案
Jun 01 Python
pycharm软件实现设置自动保存操作
Jun 08 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
理想高通滤波实现Python opencv示例
Jan 30 #Python
You might like
在WINDOWS中设置计划任务执行PHP文件的方法
2011/12/19 PHP
解析PHP可变函数的经典用法
2013/06/20 PHP
php二维数组用键名分组相加实例函数
2013/11/06 PHP
php去除字符串中空字符的常用方法小结
2015/03/17 PHP
PHP+MySQL使用mysql_num_rows实现模糊查询图书信息功能
2018/05/31 PHP
PHP实现的支付宝支付功能示例
2019/03/26 PHP
jQuery初学:find()方法及children方法的区别分析
2011/01/31 Javascript
JavaScript中的运算符种类及其规则介绍
2013/09/26 Javascript
手写的一个兼容各种浏览器的javascript getStyle函数(获取元素的样式)
2014/06/06 Javascript
JavaScript中的方法调用详细介绍
2014/12/30 Javascript
jQuery实现带动画效果的多级下拉菜单代码
2015/09/08 Javascript
jQuery 局部div刷新和全局刷新方法总结
2016/10/05 Javascript
Angular 4.x+Ionic3踩坑之Ionic3.x pop反向传值详解
2018/03/13 Javascript
Python中的__new__与__init__魔术方法理解笔记
2014/11/08 Python
python抓取文件夹的所有文件
2018/02/27 Python
PyQt5每天必学之关闭窗口
2018/04/19 Python
python验证码识别教程之滑动验证码
2018/06/04 Python
PyCharm配置mongo插件的方法
2018/11/30 Python
在python中利用KNN实现对iris进行分类的方法
2018/12/11 Python
Django实现WebSSH操作物理机或虚拟机的方法
2019/11/06 Python
Python字典中的值为列表或字典的构造实例
2019/12/16 Python
Tensorflow安装问题: Could not find a version that satisfies the requirement tensorflow
2020/04/20 Python
python中前缀运算符 *和 **的用法示例详解
2020/05/28 Python
浅谈Python中的继承
2020/06/19 Python
Python爬虫实例——爬取美团美食数据
2020/07/15 Python
在 Windows 下搭建高效的 django 开发环境的详细教程
2020/07/27 Python
Python如何进行时间处理
2020/08/06 Python
护士检查书
2014/01/17 职场文书
高中生的自我鉴定范文
2014/01/24 职场文书
银行贷款承诺书
2014/03/29 职场文书
优秀大专毕业生求职信
2014/08/04 职场文书
医院领导班子整改方案
2014/10/01 职场文书
领导四风问题整改措施思想汇报
2014/10/13 职场文书
学校百日安全活动总结
2015/05/07 职场文书
Python 数据结构之十大经典排序算法一文通关
2021/10/16 Python
Android移动应用开发指南之六种布局详解
2022/09/23 Java/Android