python实现的可爱卸载动画

打印 上一主题 下一主题

主题 931|帖子 931|积分 2793


在逛掘金时,掘金用户在B站看到的灵感举行的一个卸载窗口的动画效果的实用案例。人类是一种不断在学习的动物,并且是一种模仿能力学习能里比较强的动物。我这里是第三波的学习实践者咯!

相对VUE构建动画效果窗口,我更加喜欢用python来举行实现可爱卸载动画效果。其实,类似的软件卸载窗口,我们可以在一些常用的软件平台上举行看到,他们做的会更加的精美,同时在操作的过程当中也会流畅和丝滑太多。软件在用户在体验过程中举行末了的一次挽留用户的一次机遇。
我们自己做的过程当中更多的是对功能的一个实现以及在实现过程当中的学习拓展,搞定后的满满的成就感。我们始终在孜孜不倦,同时又在乐此不疲。希望能够在不同的维度上为你带来一些灵感。
  1. import tkinter as tk
  2. from tkinter import ttk
  3. class EmojiDialog:
  4.     def __init__(self, parent):
  5.         self.top = tk.Toplevel(parent)
  6.         self.top.title("表情弹窗")
  7.         self.top.geometry("640x480")
  8.         self.top.configure(bg='#f5f5f7')  # 苹果风格背景颜色
  9.         
  10.         # 初始化状态变量
  11.         self.hue_deg = 49
  12.         self.cheek_alpha = 1.0
  13.         self.eye_scale = 1.0
  14.         self.mouth_radius = [5, 5, 5, 5]
  15.         self.eye_pos = [90, 80, 10, 20]
  16.         self.is_leaving = False
  17.         # 创建画布
  18.         self.canvas = tk.Canvas(self.top, width=600, height=400, bg='#f5f5f7', highlightthickness=0)
  19.         self.canvas.pack(pady=20)
  20.         # 绘制初始元素
  21.         self.draw_emoji()
  22.         # 绑定事件
  23.         self.canvas.bind("<Motion>", self.on_mouse_move)
  24.         self.canvas.bind("<Leave>", self.on_mouse_leave)
  25.         # 自定义按钮样式
  26.         self.style = ttk.Style()
  27.         self.style.theme_use('clam')  # 使用'clam'主题,允许更多样式定制
  28.         self.style.configure('TButton',
  29.                              background='#eaeaea',
  30.                              foreground='black',
  31.                              font=('Helvetica', 12, 'bold'),
  32.                              padding=10,
  33.                              relief='flat')
  34.         self.style.map('TButton',
  35.                        background=[('active', '#d0d0d0')])
  36.         
  37.         # 按钮容器
  38.         button_frame = ttk.Frame(self.top, style='TFrame')
  39.         button_frame.pack(pady=20)
  40.         ttk.Button(button_frame, text="保留", command=self.some_command).pack(side=tk.LEFT, padx=10)
  41.         ttk.Button(button_frame, text="卸载", command=self.some_command).pack(side=tk.LEFT, padx=10)
  42.     def draw_emoji(self):
  43.         self.canvas.delete("all")
  44.         self.draw_head()
  45.         self.draw_eyes()
  46.         self.draw_mouth()
  47.         self.draw_cheeks()
  48.     def draw_head(self):
  49.         base_color = self.hsl_to_rgb(self.hue_deg, 100, 65.29)
  50.         color_hex = f"#{base_color[0]:02x}{base_color[1]:02x}{base_color[2]:02x}"
  51.         self.canvas.create_oval(200, 80, 400, 280, outline="#e6d706", width=3, fill=color_hex)
  52.     def draw_eyes(self):
  53.         self.draw_eye(250, 140, self.eye_pos[0], self.eye_pos[1])  # 左眼
  54.         self.draw_eye(350, 140, self.eye_pos[2], self.eye_pos[3])  # 右眼
  55.     def draw_eye(self, x, y, dx, dy):
  56.         self.canvas.create_oval(x - 25, y - 25, x + 25, y + 25, fill="white", outline="")
  57.         scale_factor = 1 + (self.eye_scale - 1) / 3
  58.         eye_size = 22 * scale_factor
  59.         eye_x = x + (dx / 100) * 25 - eye_size / 2
  60.         eye_y = y + (dy / 100) * 25 - eye_size / 2
  61.         self.canvas.create_oval(eye_x, eye_y, eye_x + eye_size, eye_y + eye_size, fill="#5f0303", outline="")
  62.     def draw_mouth(self):
  63.         x, y = 300, 200
  64.         r = self.mouth_radius
  65.         self.canvas.create_rectangle(
  66.             x - 45 + r[0], y - 20 + r[2],
  67.             x + 45 - r[1], y + 30 - r[3],
  68.             fill="#ad2424", outline="#ac0c0c", width=2, activedash=(5, 2)
  69.         )
  70.     def draw_cheeks(self):
  71.         cheek_color = self.rgba_to_hex(250, 147, 147, self.cheek_alpha)
  72.         self.canvas.create_oval(150, 200, 200, 250, fill=cheek_color, outline="")
  73.         self.canvas.create_oval(400, 200, 450, 250, fill=cheek_color, outline="")
  74.     def on_mouse_move(self, event):
  75.         if self.is_leaving:
  76.             return
  77.         x = max(200, min(event.x, 400)) - 200
  78.         y = max(80, min(event.y, 280)) - 80
  79.         width, height = 200, 200
  80.         self.hue_deg = 49 + 91 * (x / width)
  81.         self.cheek_alpha = max(0.1, 1 - (x / (width - 50)))
  82.         self.eye_scale = 1 + (x / width) / 3
  83.         self.eye_pos = [
  84.             (x / width) * 100, (y / height) * 100,
  85.             (x / width) * 100, (y / height) * 100
  86.         ]
  87.         mr = x / width
  88.         self.mouth_radius = [
  89.             max(5, mr * 50), max(5, mr * 50),
  90.             max(5, 50 - mr * 45), max(5, 50 - mr * 45)
  91.         ]
  92.         self.draw_emoji()
  93.     def on_mouse_leave(self, event):
  94.         self.is_leaving = True
  95.         self.reset_state()
  96.         self.draw_emoji()
  97.         self.is_leaving = False
  98.     def reset_state(self):
  99.         self.hue_deg = 49
  100.         self.cheek_alpha = 1.0
  101.         self.eye_scale = 1.0
  102.         self.mouth_radius = [5, 5, 5, 5]
  103.         self.eye_pos = [90, 80, 10, 20]
  104.     def hsl_to_rgb(self, h, s, l):
  105.         h /= 360
  106.         s /= 100
  107.         l /= 100
  108.         if s == 0:
  109.             rgb = (l * 255, l * 255, l * 255)
  110.         else:
  111.             def hue_to_rgb(p, q, t):
  112.                 t += 1 if t < 0 else 0
  113.                 t -= 1 if t > 1 else 0
  114.                 if t < 1 / 6: return p + (q - p) * 6 * t
  115.                 if t < 1 / 2: return q
  116.                 if t < 2 / 3: return p + (q - p) * (2 / 3 - t) * 6
  117.                 return p
  118.             q = l * (1 + s) if l < 0.5 else l + s - l * s
  119.             p = 2 * l - q
  120.             r = hue_to_rgb(p, q, h + 1 / 3)
  121.             g = hue_to_rgb(p, q, h)
  122.             b = hue_to_rgb(p, q, h - 1 / 3)
  123.             rgb = (r * 255, g * 255, b * 255)
  124.         return tuple(int(max(0, min(255, c))) for c in rgb)
  125.     def rgba_to_hex(self, r, g, b, a):
  126.         r = int(r * a + 255 * (1 - a))
  127.         g = int(g * a + 255 * (1 - a))
  128.         b = int(b * a + 255 * (1 - a))
  129.         return f"#{r:02x}{g:02x}{b:02x}"
  130.     def some_command(self):
  131.         print("命令执行")
  132. class MainApp:
  133.     def __init__(self, root):
  134.         self.root = root
  135.         self.root.geometry("300x200")
  136.         self.root.configure(bg='#f5f5f7')  # 设置主窗口背景颜色
  137.         ttk.Button(self.root, text="打开弹窗", command=self.open_dialog).pack(expand=True)
  138.     def open_dialog(self):
  139.         EmojiDialog(self.root)
  140. if __name__ == "__main__":
  141.     root = tk.Tk()
  142.     app = MainApp(root)
  143.     root.mainloop()
复制代码


免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。

本帖子中包含更多资源

您需要 登录 才可以下载或查看,没有账号?立即注册

x
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

涛声依旧在

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表