Manim 动画制作中,TransformTransformFromCopyReplacementTransformRestore是四个通用的对象变换动画类。

这几个类能够实现从一个对象到另一个对象的平滑过渡复制并变换直接替换以及状态恢复等多种效果。

  1. Transform:将一个Mobject平滑地变换为另一个Mobject
  2. TransformFromCopy:从一个对象复制出一个副本,并将该副本变换为目标对象
  3. ReplacementTransform:将一个Mobject替换为另一个 Mobject,并且在变换过程中,原始对象会被目标对象完全替代
  4. Restore:将一个Mobject恢复到其之前保存的状态时使用

通过这些动画类,我们可以清晰地展示对象之间的关系、强调变换过程、突出替换逻辑,以及灵活地在不同状态之间切换。

它们在数学公式推导、图形变换、场景切换等场景中发挥着重要作用,为动画制作提供了强大的工具,帮助创作者以直观且富有表现力的方式呈现复杂的概念和过程。

1. 动画概述

1.1. Transform

Transform用于将一个Mobject平滑地变换为另一个Mobject。比如将一个正方形逐渐变为圆形,或者将一段文本替换为另一段文本。

它常用于数学公式推导、图形变换等场景,能够直观地展示从一种状态到另一种状态的过渡。

Transform 是一个通用的变换类,可以对形状、位置、颜色等多种属性进行变换。

变换过程中,原始对象和目标对象都会保留在场景中,直到动画结束。如果需要在变换后移除原始对象,需要手动操作。

它的参数主要有:

参数名称 类型 说明
mobject Mobject 要被变换的对象
target_mobject Mobject 变换的目标对象
path_arc float 如果使用圆形路径,指定点移动的弧度
path_func func 定义了 mobject 的点在变换过程中所走的路径。可以使用 Manim 提供的路径函数或自定义函数。
path_arc_axis np.ndarray 如果使用圆形路径,指定旋转的轴
path_arc_centers np.ndarray 如果设置了此参数且未设置path_func,则会生成一个沿圆形路径的路径函数。此参数定义了圆形路径的中心。
replace_mobject_with_target_in_scene bool 控制动画完成后,是否用target_mobject替换mobject

1.2. TransformFromCopy

当需要从一个对象复制出一个副本,并将该副本变换为目标对象时使用TransformFromCopy。例如,在讲解数学公式时,从一个已有的公式中复制一部分并将其变换为新的表达式。

TransformFromCopy适用于强调某个对象的复制和变换过程,使观众更清楚地看到变换的来源。

TransformFromCopy实际上是Transform的一个变体,它先对原始对象进行复制,然后对复制的对象执行变换动画。

原始对象在动画过程中保持不变,变换的是其副本,这使得动画效果更加清晰,避免了原始对象的直接改变。

它的参数类似Transform的参数。

1.3. ReplacementTransform

ReplacementTransform用于将一个Mobject替换为另一个Mobject,并且在变换过程中,原始对象会被目标对象完全替代。

例如,在展示数学公式推导时,将一个公式直接替换为另一个公式,强调两者的等价性或替换关系。

ReplacementTransform适用于需要明确表示对象被替换的场景,强调变换的“替代”性质。

ReplacementTransform继承自Transform类,但它在变换完成后会移除原始对象,只保留目标对象。

它更强调“替换”的效果,而不是像Transform那样保留原始对象和目标对象同时存在一段时间。

它的参数类似Transform的参数。

1.4. Restore

当需要将一个Mobject恢复到其之前保存的状态时使用Restore

例如,在对一个图形进行了一系列复杂的变换后,需要将其恢复到初始状态,或者在展示一个对象的多种变换后,恢复到某个特定的中间状态。

Restore适用于需要反复展示对象不同状态的场景,如对比不同变换效果、演示可逆变换等。

它需要先通过Mobject.save_state()方法保存对象的状态,然后才能使用Restore动画将其恢复。

它的参数主要有:

参数名称 类型 说明
mobject Mobject 要恢复的对象

2. 使用示例

下面通过示例来演示这几个转换动画的效果。

2.1. 通用转换Transform

这个示例中,将一个数学公式和一个图形平滑地变换为另一个公式和图形。

变换中,设置了path_arc参数,以一个弧形的路径变换为目标对象。

# 创建初始公式
f1 = MathTex("a^2 + b^2 = c^2").shift(UP)
# 创建目标公式
f2 = MathTex("c = \\sqrt{a^2 + b^2}").shift(DOWN) # 创建一个正方形
square = Square(side_length=1, color=BLUE).shift(DOWN)
# 创建一个圆形
circle = Circle(radius=0.6).shift(UP) self.add(f1, square)
self.wait()
# 使用 Transform 将公式1变换为公式2,将正方形变换为圆形
self.play(
Transform(f1, f2, path_arc=PI),
Transform(square, circle, path_arc=PI),
)

2.2. 比较Replace和Copy的不同

通过这个示例,可以了解TransformFromCopyReplacementTransform两种动画效果的区别,帮助我们根据实际需求选择合适的动画类。

对于TransformFromCopy

  1. 从原始对象复制出一个副本,并对副本进行变换
  2. 原始对象保持不变,变换的是副本
  3. 适用于需要保留原始对象的场景

对于ReplacementTransform

  1. 直接将原始对象替换为目标对象
  2. 原始对象在动画结束后被移除,目标对象取代其位置
  3. 适用于需要明确表示对象被替换的场景

下面的示例中,TransformFromCopy每次转换都会保存原对象,而ReplacementTransform则只保留变换后的目标对象。

# 创建两个相同的正方形作为原对象
s1 = Square().shift(LEFT * 1.5)
s2 = Square().shift(RIGHT * 1.5) # 创建两个圆形作为目标对象
c1 = Circle().shift(LEFT * 1.5)
c2 = Circle().shift(RIGHT * 1.5) t1 = Triangle().shift(LEFT * 1.5)
t2 = Triangle().shift(RIGHT * 1.5) # 添加原对象到场景中
self.add(s1, s2)
self.wait(1) # 添加标题用于区分两种变换
title1 = Text(
"TransformFrom Copy",
t2g={"Copy": [BLUE, RED, YELLOW]},
font_size=25,
).next_to(s1, UP)
title2 = Text(
"Replacement Transform",
t2g={"Replacement": [BLUE, RED, YELLOW]},
font_size=25,
).next_to(s2, DOWN)
self.add(title1, title2)
self.wait() # 使用 TransformFromCopy 对左边的图形进行变换
# 使用 ReplacementTransform 对右边的图形进行变换
self.play(
TransformFromCopy(s1, c1),
ReplacementTransform(s2, c2),
)
self.wait()
self.play(
TransformFromCopy(c1, t1),
ReplacementTransform(c2, t2),
)

2.3. 恢复初始状态Restore

这个示例中,先创建一个六边形,先变换成星形,再经过一系列其他的变换,比如颜色,透明度,移动,放大,翻转等变换。

最后通过Restore直接恢复成原始状态。

# 创建一个六边形
hexagon = RegularPolygon(n=6)
# 保存六边形的初始状态
self.add(hexagon)
hexagon.save_state()
# 将六边形变换为一个星形
star = Star(color=RED)
self.play(Transform(hexagon, star), run_time=run_time)
self.wait()
self.play(Restore(hexagon)) self.play(hexagon.animate.set_color(YELLOW).set_opacity(0.4), run_time=run_time)
self.play(hexagon.animate.shift(RIGHT).scale(2), run_time=run_time)
self.play(hexagon.animate.rotate(PI * 0.75), run_time=run_time) # 使用 Restore 恢复六边形的初始状态
self.play(Restore(hexagon))

3. 附件

文中的代码只是关键部分的截取,完整的代码共享在网盘中(transform.py),

下载地址: 完整代码 (访问密码: 6872)

manim边学边做--通用变换的更多相关文章

  1. 学EE做硬件找工作不如学CS做软件,为什么会这样?

    学EE做硬件找工作不如学CS做软件,为什么会这样? 电子工程(EE)就业最好的方向居然是转计算机,也许让有的人觉得很不公平,EE也是很重要的学科,我们学习也很努力,为什么就业会不如CS?也有的人好奇, ...

  2. 牛腩学Kotlin做Android应用

    牛腩学Kotlin做Android应用,蹭热度视频,边学边做, 01-kotlin插件安装及hello world 02-kotlin基础语法速览 哔哩哔哩观看地址:http://www.bilibi ...

  3. php实现记忆化递归--以斐波那契数列为例(还是以边学边做为主,注重练习)

    php实现记忆化递归--以斐波那契数列为例(还是以边学边做为主,注重练习) 一.总结 1.递归不优化的话,30层开外就有点吃力了 2.php因为定义变量的时候不用定义变量类型,所以数组里面的类型也是p ...

  4. html5-3 html5标签(热点地图如何实现)(边学边做)

    html5-3 html5标签(热点地图如何实现)(边学边做) 一.总结 一句话总结:热点地图用绝对定位实现. 1.自定义列表怎么弄? dl  自定义列表dt  自定义标题dd  自定义列表内容 2. ...

  5. 第一份开发工作,边学边做android

    我刚刚毕业,在培训学校学的Java web开发,虽然学的没有大学生那么丰富细致,没有他们理论基础扎实,但是这是我学习软件开发的唯一方式了. 从小学我学习就是倒数2.3等,所有人都认为我是个没法学习的孩 ...

  6. 如何通过 Vue+Webpack 来做通用的前端组件化架构设计

    目录:   1. 架构选型     2. 架构目录介绍     3. 架构说明     4. 招聘消息 目前如果要说比较流行的前端架构哪家强,屈指可数:reactjs.angularjs.emberj ...

  7. java利用注解及反射做通用的入参校验

    一.原理: 1.做一个field注解,注解有两个参数:是否必填.toString之后的最大长度 2.对某个request类(或基类),使用注解标记某个字段的校验详情 3.通用的static方法,利用反 ...

  8. 边学边做,简单的 GraphQL 实例

    项目中有功能要调用 API,对方 API 用的是 GraphQL 实现,就简单学了下,感叹技术进步真快,Facebook 发明的这玩意儿咋这么牛逼,使前端开发人员变得主动起来,想要什么接口.返回什么结 ...

  9. thinkphp5高亮当前页(仅针对个人项目记录,不做通用参考)

    <div class="navbg"> <ul class="menu"> <li> <a href="/& ...

  10. LabVIEW Actor Framwork (2)________ 边学边做server&client

    回顾下初始需求: 现在要做一个类似聊天的demo,一个server端,若干个client端:首先是server启动,通过server可以打开若干个client端,然后每个client可以独立给serv ...

随机推荐

  1. Git之实战命令

    相关概念: HEAD: 说简单一点,HEAD就是当前活跃分支的游标. 形象的记忆就是:你现在在哪儿,HEAD就指向哪儿,所以Git才知道你在那儿! 不过HEAD并非只能指向分支的最顶端(时间节点距今最 ...

  2. 简单端口映射、转发、重定向工具之Rinetd

    ◆一.概述 Rinetd是为在一个Unix和Linux操作系统中为重定向传输控制协议(TCP)连接的一个工具.将 TCP 连接从一个 IP 地址和端口重定向到另一个.它处理文件中/etc/rinetd ...

  3. 简述GoLang优势与生态

    开门见山,一睹golang的风采 性能优势 部署运维成本低 编码格式统一:测试简单 1. 性能优势 Go 语言被称为是:"21世纪的C语言",虽然这个帽子戴的有点高,不妨这里给大家 ...

  4. 【集成-Jedis】SpringBoot集成Jedis

    将jedis的依赖放进Maven <dependency> <groupId>redis.clients</groupId> <artifactId>j ...

  5. Linux清理内存,清理储存

    因为工作中项目部署服务器后更新迭代或者服务器使用时间长后会出现内存/储存爆满,所以整合了一下,方便以后使用: 清理虚拟内存 查看内存 free -h 清理缓存 输入命令释放内存 0 – 不释放 1 – ...

  6. 树莓派设置CPU运行的核心数为3,保留核心4号

    具体步骤:1.打开终端,输入 sudo nano /boot/cmdline.txt2.在第一行最后空一格加上isolcpus=33.保存

  7. grpc与http2的关系

    https://nullget.sourceforge.io/?q=node/895 grpc与http2的关系 grpc client 发送包到原生的http2 server client收到报错: ...

  8. Qt/C++音视频开发48-推流到rtsp服务器

    一.前言 之前已经打通了rtmp的推流,理论上按照同样的代码,只要将rtmp推流地址换成rtsp推流地址,然后格式将flv换成rtsp就行,无奈直接遇到协议不支持的错误提示,网上说要换成rtp,换了也 ...

  9. Qt编写可视化大屏电子看板系统20-横向分组图

    一.前言 横向分组图是柱状分组图的横向展示,有了横向柱状图加上分组图的绘制经验,这个做起来就比较简单了,横向的设置规则按照横向柱状图来,分组的规则按照柱状分组图的算法来,在横向的柱子中要绘制对应的值, ...

  10. IM跨平台技术学习(十二):万字长文详解QQ Linux端实时音视频背后的跨平台实践

    本文由QQ音视频团队贺坤分享原题"Linux QQ能打语音视频了!一文详解背后技术实现!",下文进行了排版和内容优化等. 1.引言 2024年6月6日,QQ For Linux 3 ...