当前位置: 首页 > news >正文

气泡特效的核心在于BubbleEffect类,它继承自Manim的Animation类,通过重写关键方法来实现气泡的上升、变大和透明度变化效果。

核心类结构

BubbleEffect类的基本结构如下:

class BubbleEffect(Animation): """ 彩色气泡特效动画类 继承自Animation类,用于创建彩色气泡上升、变大、透明度变化的效果 """ def __init__( self, bubble_count=25, # 气泡数量 bubble_size_range=(0.1, 0.5), # 气泡大小范围 rise_speed_range=(0.6, 2.2), # 上升速度范围 growth_rate_range=(0.005, 0.015), # 生长速度范围 fade_rate_range=(0.02, 0.06), # 消失速度范围 colors=None, # 气泡颜色列表 **kwargs ): # 初始化代码... def create_bubble(self): # 创建单个气泡的代码... def interpolate_mobject(self, alpha): # 控制气泡动画效果的核心代码...

1.2. 气泡创建机制

__init__方法中,我们首先定义了一系列参数,如气泡数量、大小范围、上升速度等,然后创建一个VGroup来存放所有气泡:

# 创建气泡组 self.bubbles = VGroup() self.bubble_count = bubble_count self.bubble_size_range = bubble_size_range self.rise_speed_range = rise_speed_range self.growth_rate_range = growth_rate_range self.fade_rate_range = fade_rate_range self.colors = colors or [RED, BLUE, GREEN, YELLOW, PURPLE, ORANGE] # 保存运行时间,从kwargs中获取或使用默认值 self.run_time = kwargs.get("run_time", 5.0) # 初始化气泡 for _ in range(bubble_count): bubble = self.create_bubble() self.bubbles.add(bubble) # 调用父类初始化 super().__init__(self.bubbles, **kwargs)

每个气泡通过create_bubble方法创建,该方法随机设置气泡的大小、颜色和初始位置,并为每个气泡分配独立的上升速度、生长速度和消失速度:

def create_bubble(self): """创建单个彩色气泡""" # 随机大小 size = random.uniform(*self.bubble_size_range) # 随机颜色 color = random.choice(self.colors) # 创建圆形气泡 bubble = Circle(radius=size, color=color, fill_opacity=0.4, stroke_width=2) # 随机初始位置(底部区域) x_pos = random.uniform(-config.frame_width / 2 + 1, config.frame_width / 2 - 1) y_pos = random.uniform(-config.frame_height / 2, -config.frame_height / 2 + 2) bubble.move_to([x_pos, y_pos, 0]) # 存储气泡属性 bubble.rise_speed = random.uniform(*self.rise_speed_range) bubble.growth_rate = random.uniform(*self.growth_rate_range) bubble.fade_rate = random.uniform(*self.fade_rate_range) bubble.initial_radius = size return bubble

1.3. 动画插值实现

动画的核心在于interpolate_mobject方法,它控制着每个气泡在每一帧的状态变化:

def interpolate_mobject(self, alpha): """插值函数,控制彩色气泡的动画效果""" dt = 1 / config.frame_rate # 每帧的时间间隔 for bubble in self.bubbles: # 上升 bubble.shift(UP * bubble.rise_speed * dt) # 变大 bubble.scale(1 + bubble.growth_rate * dt) # 透明度变化 current_opacity = bubble.get_fill_opacity() new_opacity = current_opacity - bubble.fade_rate * dt # 如果气泡超出屏幕顶部或透明度降到0以下,则重置 if new_opacity <= 0 or bubble.get_y() > config.frame_height / 2: # 重置气泡 x_pos = random.uniform( -config.frame_width / 2 + 1, config.frame_width / 2 - 1 ) y_pos = random.uniform( -config.frame_height / 2, -config.frame_height / 2 + 2 ) bubble.move_to([x_pos, y_pos, 0]) bubble.set_fill(opacity=0.4) bubble.set_stroke(opacity=0.4) else: bubble.set_fill(opacity=new_opacity) bubble.set_stroke(opacity=new_opacity)

这个方法实现了三个关键效果:

  1. 上升:每个气泡以自己的速度向上移动
  2. 变大:每个气泡以自己的速度缓慢变大
  3. 透明度变化:每个气泡逐渐变得透明

特别值得注意的是,当气泡超出屏幕顶部或透明度降到0以下时,代码会将气泡重置到底部,从而实现循环不断的气泡效果。

2. 使用示例

代码提供了两个使用示例,分别展示了普通气泡效果和彩色气泡效果。

2.1. 普通气泡效果

SimpleBubbleEffectExample类展示了如何创建灰度的气泡效果:

class SimpleBubbleEffectExample(Scene): """普通气泡特效示例场景""" def construct(self): # 创建标题 title = Text("普通气泡特效演示", font_size=48) title.to_edge(UP) self.play(Write(title)) self.wait(0.5) bubble_effect = BubbleEffect( bubble_count=25, colors=[GRAY], # 设置为灰色 bubble_size_range=(0.1, 0.5), rise_speed_range=(0.6, 2.2), growth_rate_range=(0.1, 0.5), fade_rate_range=(0.02, 0.06), run_time=2, # 使用run_time而不是duration ) # 播放气泡特效 self.play(bubble_effect) self.wait()

2.2. 彩色气泡效果

ColorfulBubbleEffectExample类展示了如何创建彩色的气泡效果:

class ColorfulBubbleEffectExample(Scene): """彩色气泡特效示例场景""" def construct(self): # 创建标题 title = Text("彩色气泡特效演示", font_size=48) title.to_edge(UP) self.play(Write(title)) self.wait(0.5) # 创建彩色气泡特效动画 bubble_effect = BubbleEffect( bubble_count=25, bubble_size_range=(0.1, 0.5), rise_speed_range=(0.6, 2.2), growth_rate_range=(0.1, 0.5), fade_rate_range=(0.02, 0.06), run_time=2, # 使用run_time而不是duration ) # 播放彩色气泡特效
http://www.cnnetsun.cn/news/3075299.html

相关文章:

  • 一文搞懂巴别鸟版本管理:从历史回溯到冲突解决的完整攻略
  • 河南AI大模型人才培养观察:从通识普及到产业实战的多元路径
  • 快马AI三步搭建OpenClaw安卓自动化测试环境:告别手动配置噩梦
  • 别乱改!Multisim14.2三极管仿真参数修改的实战避坑指南(以2N3904为例)
  • 把 quicklink 的预加载思想搬到 API 层:我设计了一套‘懒请求调度器’,首屏并发从 9 降到了 2
  • 化学图像识别工具横评:DECIMER、Img2Mol、MolScribe,哪个更适合你的科研流水线?
  • 《Debezium + Kafka Connect 实战:从零搭建 MySQL CDC 数据管道,踩坑全记录》
  • M4Markets:技术架构的路径复盘
  • open harmony 项目实战:用 AppStorage 实现轻量级页面路由和状态管理
  • open harmony 项目实战:用 ArkTS 实现诗词收藏和阅读历史
  • 基于51/STM32单片机温湿度控制系统设计大棚检测成品恒温恒湿光照44(设计源文件+万字报告+讲解)(支持资料、图片参考_相
  • JavaScript Promise详解
  • Grid布局开发实践
  • C++虚函数工作原理
  • Angular基础开发教程
  • 阅读APP书源配置终极指南:一键解锁全网小说库的完整教程
  • PHP SQL注入检测实战:从原理到自动化工具实现
  • java+前端学习笔记
  • Python网站下载器:三步将整个网站完整保存到本地
  • 5个实用技巧:快速掌握Monitorian多显示器亮度调节
  • CAIWY 采购知识库(六)
  • Parsec虚拟显示器终极指南:如何实现零延迟的4K游戏串流体验
  • 6款论文降AIGC工具实测:AI率秒归安全区,学生党狂喜款
  • 计算机Java毕设实战-基于 SpringBoot 的大学生在线评教打分系统的设计与实现 基于 SpringBoot 的高校教学质量评价系统【完整源码+LW+部署说明+演示视频,全bao一条龙等】
  • 别再乱按复位键了!手把手教你搞懂STM32的三种复位方式(含独立/窗口看门狗详解)
  • Java基础快速入门: File类与IO流基础
  • 【AI大模型应用开发】【项目实战】9.基于GPT2搭建医疗问诊机器人
  • 版本兼容设计事件类预留版本字段:
  • C++ STL之互斥锁与条件变量详解
  • C++ STL 之随机数生成详解