manim边做边学–动画联动

今天介绍Manim中的动画联动的技巧,在数学动画中,动画联动是常用的功能,

比如讲解平面几何中三角形与圆的位置关系变化,通过动画联动可以让圆沿着三角形的边滚动,或者让三角形的顶点在圆上移动,从而直观地展示内切、外接等几何关系。

总之,通过动画联动,可以将复杂的概念、关系或变化过程以动态的方式展示出来。

这种动态展示比静态的图像或文字描述更具吸引力,能让观众更容易理解抽象的知识。

1. 联动原理

Manim中,每个Mobject对象都有一个add_updater函数,这个函数是实现动画联动的关键。

add_updater这个更新函数会在每一帧渲染之前被调用,用于更新Mobject的属性。

例如,你可以用它来改变一个图形的位置、颜色、大小等属性,从而创建动态的动画效果。

当你有多个Mobject并且希望它们的动画相互关联时,add_updater就非常有用。

比如,你有一个圆形和一个正方形,你想让正方形的位置始终跟随圆形的位置变化,就可以通过为正方形添加一个updater函数,在函数内部根据圆形的位置来更新正方形的位置。

add_updaterMobject类的一个方法。其基本函数原型如下:

add_updater(update_function, index=None, call_updater=False)

其中:

  1. update_function:这是最重要的参数,它是一个可调用对象(通常是一个函数),用于更新Mobject的属性
  2. index:当有多个updater联动函数时,index表示updater的调用顺序
  3. call_updater:是否在被调用后立即执行一次,而不仅仅是在每一帧渲染之前执行

2. 联动示例

示例是最好的学习资料,之前做尺规作图的动画时,模拟圆规动作的动画就是一个联动动画。

下面的示例主要介绍如何通过联动动画来完成其中的圆规动作。

圆规动画目的是画一个 圆弧 ,只是在绘制的圆弧的过程中还需要展示了圆弧的起点,终点,

以及绘制过程中动态更新起点终点之间的线。

首先,定义绘制函数:

```python
def ruler(sc: Scene, p1, p2, color=GREEN, angle=PI, axis=OUT):
    """
    圆规动画


    Parameters
    ---------
    sc
        绘制动画的场景
    p1
        代表圆规的针,绘制时不动的点
    p2
        代表圆规的笔芯,绘制圆弧的点
    color
        圆弧的颜色,默认绿色
    angle
        绘制圆弧的角度,默认PI,相当于绘制半个圆
    axis
        只有2个值 IN/OUT,分别表示顺时针还是逆时针作弧
    """
    # 省略。。。
    return arc
```

实现动画的思路如下:

  1. 构建3个元素,即:
    1. d1(根据参数中p1坐标绘制的点)
    2. d2(根据参数中p2坐标绘制的点)
    3. dl(连接p1和p2的虚线)
  2. 设置dl的动画,随着 d1d2变化不断重新绘制(这里 d1其实是不会变的)
  3. 再设置圆弧的动画,随着 d2的变动,不断绘制新的圆弧(圆弧就是起点到d2的弧)
  4. 通过 Manim自带的动画函数让 d2先动,其他动画随之运动
  5. 最后删除不必要的元素,只保留圆弧在 场景(sc)中

    ```python
    def ruler(sc: Scene, p1, p2, color=GREEN, angle=PI, axis=OUT):
    d1 = Dot(point=p1, color=RED)
    d2 = Dot(point=p2, color=color)
    dl = DashedLine(d1.get_center(), d2.get_center())

    r = np.linalg.norm(p2 - p1)
    arc = ArcBetweenPoints(p2, p2, stroke_width=2)
    
    dl.add_updater(lambda z: z.become(DashedLine(d1.get_center(), d2.get_center())))
    if np.array_equal(axis, OUT):
        arc.add_updater(
            lambda z: z.become(
                ArcBetweenPoints(
                    p2, d2.get_center(), radius=r, stroke_color=color, stroke_width=2
                )
            )
        )
    if np.array_equal(axis, IN):
        arc.add_updater(
            lambda z: z.become(
                ArcBetweenPoints(
                    d2.get_center(), p2, radius=r, stroke_color=color, stroke_width=2
                )
            )
        )
    
    sc.add(d1, d2, dl, arc)
    sc.play(
        Rotate(
            d2,
            about_point=d1.get_center(),
            axis=axis,
            angle=angle,
            rate_func=linear,
        )
    )
    
    arc.clear_updaters()
    dl.clear_updaters()
    sc.remove(d1, d2, dl)
    return arc
    

    ```

调用这个函数的效果如下:

```python
ruler(self, np.array([-1, 0, 0]), np.array([-1, 1, 0]))
ruler(self, np.array([1, 0, 0]), np.array([1, 1, 0]), axis=IN)
```

manim边做边学--动画联动

画两个圆弧的交叉效果:

```python
ruler(self, np.array([-1, -1, 0]), np.array([1.1, -0.2222, 0]), angle=PI / 3)
ruler(
    self,
    np.array([1, -1, 0]),
    np.array([-1.1, -0.2222, 0]),
    axis=IN,
    angle=PI / 3,
)
```

manim边做边学--动画联动

3. 注意事项

使用add_updater函数时,首先需要注意的是性能问题 ,由于add_updater函数会在每一帧渲染前被调用,所以要避免在updater函数中进行复杂的计算。

例如,在一个场景中有大量的Mobject,如果在updater函数中进行高复杂度的矩阵运算或者嵌套循环来更新每个Mobject的位置,会导致动画渲染速度变慢,甚至可能出现卡顿。

其次,当多个Mobject之间存在复杂的依赖关系并且都使用add_updater方法时,要确保更新函数的逻辑正确。

要注意更新函数内部不要出现无意的无限循环或递归情况。

最后,当动画的某个阶段不再需要updater函数来更新Mobject时,要及时使用remove_updater方法移除更新器。

如果不及时移除不再需要的更新器,可能会导致内存泄漏和资源浪费。

文章整理自互联网,只做测试使用。发布者:Lomu,转转请注明出处:https://www.it1024doc.com/6682.html

(0)
LomuLomu
上一篇 2025 年 1 月 16 日 上午8:33
下一篇 2025 年 1 月 16 日

相关推荐

  • 架构-初识BFF

    引言 在最近的一次公司技术分享会上,我们深入探讨了公司的项目架构。核心议题是BFF架构,这是一种在微服务架构之上增加的额外层级。此外,我们还讨论了DDD(领域驱动设计)理念,它在订单、用户等业务中台中扮演着关键角色。 这是我对架构领域的初步探索,虽然理解尚浅,但我还是尝试着将所学内容进行了整理。 BFF 定义 BFF,即Backend For Fronten…

    2024 年 12 月 26 日
    58100
  • Java中的网络基础认知(如果想知道Java中有关网络基础的知识,那么只看这一篇就足够了!)

    前言:网络基础是现代通信和信息技术的基石,涉及数据传输、网络协议、路由、交换、网络设备以及网络安全等多个方面,深入了解网络基础,不仅能提升技术能力,还能为更复杂的网络架构与应用打下坚实的基础。 ✨✨✨ 这里是秋刀鱼不做梦的BLOG 目录 网络发展史简介 独立模式与网络互连 局域网(LAN) 广域网(WAN) 网络通信基础 —— IP和端口号 IP地址 端口号…

    2024 年 12 月 28 日
    36200
  • Java中String类常用的各种方法

    Java中String类常见的方法 以下介绍字符串常见的几个方法。 介绍String类 在 Java 中,String 类是一个代表字符串的类,具有以下特性: 不可变性 :String 对象一旦被创建就是不可变的,即它们的值在创建后不能被更改。任何对 String 对象的修改操作实际上会创建一个新的 String 对象。 字符串池 :Java 中的字符串池是…

    未分类 2025 年 1 月 17 日
    34100
  • 深入解析Java中的JSONObject操作指南

    文章大纲 Java中JSONObject的全面解析 第一部分:概述 第二部分:JSONObject的基础应用 1、初始化JSONObject 2、数据插入方法 3、数值提取技巧 第三部分:JSONObject进阶功能 1、对象遍历实现 2、字符串转换JSON 3、JSONArray协同操作 4、数据修改与移除 第四部分:异常处理机制 1. 键值验证 2. 安…

    2025 年 5 月 19 日
    61200
  • JDK、IDEA安装教程 IDEA安装2023年10月 最新最详细!免费、亲测成功!!!

    系列文章目录(Intellij IDEA2023年10月 最新教程) Java学习之IDEA的安装教程和使用 以下是第一章的内容:IntelliJ IDEA安装教程 文章目录 前言 准备工作 步骤(敲详细!!!) 前言 IDEA的安装是Java课程的入门,本人台式机于2023年7月份进行了系统重装,恰巧最近需要重新安装IDEA进行使用,借此机会给同学们分享我…

    未分类 2025 年 5 月 13 日
    31100

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

联系我们

400-800-8888

在线咨询: QQ交谈

邮件:admin@example.com

工作时间:周一至周五,9:30-18:30,节假日休息

关注微信