七夕来袭!是时候展现专属于程序员的浪漫了!你打算怎么给心爱的人表达爱意?鲜花礼物?代码表白?还是创意DIY?或者…无论那种形式,快来秀我们一脸吧!
让我们一起记忆这美好的今天,愿我们年年有今日,岁岁有今朝。祝天下的情人终成眷属。
很多人说工科生不懂浪漫,程序员就更别提了,铁男一枚,钢铁大直男,可程序员其实也是动得浪漫🌹的。
下面我们用python写出我们程序员的浪漫吧!以程序员的方式撒狗粮,专业浪漫,值得拥有!
这个动态效果是由 Tkinter
库来完成的,属于Python的GUI编程部分。Python提供了多个图形开发界面的库,常用的有Tkinter
,xwPython
,Jython
。Tkinter是Python的标准GUI库,内置在Python中,不需要额外安装,对于一些简单的图形界面可以轻松实现。
下面是七夕节烟花效果的代码实现,首先导入所有需要的库:
然后定义一个通用的烟花颗粒的类(part),烟花颗粒的属性如下:
然后在这个类中定义了烟花颗粒的一些类方法:
- # 设置重力参数
- GRAVITY = 0.05
- # 设置随机的颜色列表
- colors = ['red', 'blue', 'yellow', 'white', 'green', 'orange', 'purple', 'seagreen', 'indigo', 'cornflowerblue']
-
- 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
- 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))
-
- 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
上面完成了一个通用的烟花颗粒类的实现,下面就开始烟花燃放的模拟循环过程:通过递归不断循地在背景中产生新的烟花。
首先定义一个 simulate
模拟的函数,在函数中定了一些参数:
然后在所有的烟花数量中循环创建所有的烟花颗粒类,当然在每次循环中颗粒类都需要设置一定的属性参数,参数多是随机产生:
有了这些参数,我们就可以定义循环产生每个颗粒对象了,并将每个烟花的所有颗粒对象储存在objects
中。也就是说explore_points是列表中套列表,内层列表是每个烟花的所有颗粒对象,外层列表是所有烟花。
所有的颗粒对象完成后,就开始对每个颗粒的生命时间进行更新,且总时间设定在1.8秒
以内。最后通过root
递归使烟花可以一直在背景中燃放。
- 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 = 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
- # 保持在1.8秒内进行更新
- 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()
以上代码部分均与Tkinter
无关,只是定义了颗粒对象以及模拟颗粒生命周期的全过程,下面将使用Tkinter
完成最终的效果。
然后将在画布对象上创建一个图像(使用定义的photo
对象作为参数),最后调用Tkinter
对象root
进行持续不断地simulate模拟过程。
- if __name__ == '__main__':
- root = tk.Tk()
- cv = tk.Canvas(root, height=600, width=600)
- # 自己选择一个好的图像背景填充画布
- 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()
联系我获取完整代码 ,祝君开心度过一个美好浪漫的情人节
注意:背景图片可根据自己的喜好进行更换,还不赶紧定制一个属于自己的烟花秀?
完整代码可以到下面的链接中下载