【manim】含有add_updater更新函数become的物体移动方法
在manim社区版本中,
一、对于一般的物体,移动的方法分为 (瞬移) 和 (带动画移动)
1、瞬移
#直接对物体操作即可
obj.shift(LEFT)
#瞬间移动,LEFT,UP是单位方向向量
#以自身为参考
obj.move_to()
#瞬间移动
#以屏幕中心为参考
obj.next_to(circle,RIGHT)
#瞬间移动
#以传入对象为参考
obj.align_to(circle,DOWN)
#瞬间移动
#以传入对象的假想边界作为调整
#传入DOWN(0,-1),那么物体的纵坐标就会"对其"对象的下边界
#这个假想的边界一般以物体以中心为原点的第二象限区域
#此处可参考https://docs.manim.community/en/stable/tutorials/building_blocks.html#mobjectplacement
2、带动画移动
#此处需要使用animate方法
self.play(obj.animate.shift(LEFT))
#移动时间持续1秒
#以自身为参考
#。。。省略,同理
二、对于添加了 add_update()的物体obj,如果想移动实时更新的物体,则需要从更新函数中处理。
此时,直接对obj使用移动的函数是无效的。例如 obj.shift(LEFT)
add_update的逻辑是帧渲染,每一帧画面都会从 add_update()中传入的 方法参数 中生成新的obj。例如 obj.add_updater( lambda mob: mob.become(Function(lambda x : x + 1 )) )
manim在渲染的过程中每一帧画面的obj都会调用一次 lambda mob: mob.become(Function(lambda x : x + 1 )),作为这一帧的画面。
所以,如果我们要移动obj的位置,只有一种方法:
移动 Function(lambda x : x + 1 ) 的位置。
例如:我们要移动一个带updater的参数曲线,要怎么做呢?
效果:

于是只需要在become中取得上一帧画面的位置即可。

add_update非常暴力,在某一帧中,我本来已经调用了shift动画了(也就是移动了一点),但是我的add_update又把它整回去了,因为add_update只认上一帧的位置,而这一切都是在帧与帧之间完成的。
所以目前只能暂时把updater去掉。
class Curve3(Scene):
def construct(self):
#initial value
A = 1
B = 1
C = 0
values = VGroup(*[
Tex(f"{t}=",tex_to_color_map={f"{t}":color}).scale(1.3)
for t,color in zip(["a","b","c"],[RED,BLUE,YELLOW])
])
values.arrange(DOWN,aligned_edge=LEFT,buff=1.2)
dn_kwargs = {"unit": r"^\circ"}
D_A = DecimalNumber(A,**dn_kwargs)
D_B = DecimalNumber(B,**dn_kwargs)
D_C = DecimalNumber(C,**dn_kwargs)
D_G = VGroup(D_A,D_B,D_C)
for d,s in zip(D_G,values):
d.next_to(s,aligned_edge=DOWN)
pc = self.get_param_func(A,B,C)
upfunc =lambda mob: mob.become(
self.get_param_func(
D_A.get_value(),
D_B.get_value(),
D_C.get_value()
).move_to(pc)
)
pc.add_updater(upfunc)
self.add(pc)
pc.shift(LEFT*3)
self.play(
ChangeDecimalToValue(D_B,15),
run_time=2,
rate_func=linear,
)
self.wait(0.3)
#暂时移除updater
pc.remove_updater(upfunc)
self.play(pc.animate.shift(RIGHT*6),run_time=1)
#平滑移动完,再添加回来
pc.add_updater(upfunc)
self.play(
ChangeDecimalToValue(D_A,10),
run_time=2,
rate_func=linear,
)
def get_param_func(self, a, b, c):
pc = ParametricFunction(
lambda t: np.array([
np.cos(a * t) + np.cos(b * t) / 2 + np.sin(c * t) / 3,
np.sin(a * t) + np.sin(b * t) / 2 + np.cos(c * t) / 3,
0
]),
t_range=[0,2*PI,0.007],
fill_opacity=0
).scale(2)\
.set_color(color=[RED,YELLOW,BLUE,RED])
return pc
或者使用一些曲线救国的方法实现平滑移动:
即添加一个参照物OBJECT,然后在更新函数中align_to(self.OBJECT)即可。当我们移动OBJECT的时候,我们的目标对象也会跟着移动。
class Curve3(Scene):
OBJECT = Text("1").shift(LEFT*6+UP*100)
def construct(self):
#initial value
A = 1
B = 1
C = 0
values = VGroup(*[
Tex(f"{t}=",tex_to_color_map={f"{t}":color}).scale(1.3)
for t,color in zip(["a","b","c"],[RED,BLUE,YELLOW])
])
values.arrange(DOWN,aligned_edge=LEFT,buff=1.2)
dn_kwargs = {"unit": r"^\circ"}
D_A = DecimalNumber(A,**dn_kwargs)
D_B = DecimalNumber(B,**dn_kwargs)
D_C = DecimalNumber(C,**dn_kwargs)
D_G = VGroup(D_A,D_B,D_C)
for d,s in zip(D_G,values):
d.next_to(s,aligned_edge=DOWN)
pc = self.get_param_func(A,B,C)
pc.add_updater(
lambda mob: mob.become(
self.get_param_func(
D_A.get_value(),
D_B.get_value(),
D_C.get_value()
)
).align_to(self.OBJECT,LEFT)
)
self.add(pc,self.OBJECT)
self.play(
ChangeDecimalToValue(D_B,15),
run_time=2,
rate_func=linear,
)
self.wait(0.3)
self.play(self.OBJECT.animate.shift(RIGHT*6))
self.play(
ChangeDecimalToValue(D_A,10),
run_time=2,
rate_func=linear,
)
def get_param_func(self, a, b, c):
pc = ParametricFunction(
lambda t: np.array([
np.cos(a * t) + np.cos(b * t) / 2 + np.sin(c * t) / 3,
np.sin(a * t) + np.sin(b * t) / 2 + np.cos(c * t) / 3,
0
]),
t_range=[0,2*PI,0.007],
fill_opacity=0
).scale(2)\
.set_color(color=[RED,YELLOW,BLUE,RED])
return pc
【manim】含有add_updater更新函数become的物体移动方法的更多相关文章
- Python3:读取配置dbconfig.ini(含有中文)显示乱码的解决方法
Python3:读取配置dbconfig.ini(含有中文)显示乱码的解决方法 一.原因 Python 3 中虽有encoding 参数,但是对于有BOM(如Windows下用记事本指定为utf-8) ...
- 题目一:编写一个类Computer,类中含有一个求n的阶乘的方法
作业:编写一个类Computer,类中含有一个求n的阶乘的方法.将该类打包,并在另一包中的Java文件App.java中引入包,在主类中定义Computer类的对象,调用求n的阶乘的方法(n值由参数决 ...
- 物体检测方法(1) - YOLO 详解
最近遇到一些卡证识别的检测问题,打算先把理论知识梳理一下,随后还会梳理一版代码注释. 以前的region+proposal来检测的框架,这一系列速度和精度不断提高,但是还是无法达到实时.存在的主要问题 ...
- 3DMax 物体选择方法
全选: Ctrl + A, 取消选择:Ctrl +D 加选:ctrl+鼠标左键:减选:alt+鼠标 窗口与交叉:下面红框内的右边的按钮, 是切换两种模式: 选择模式一:只要选框碰到物体边缘, 就可选中 ...
- Unity3D 角色(物体) 移动方法 合集
1. 简介 在Unity3D中,有多种方式可以改变物体的坐标,实现移动的目的,其本质是每帧修改物体的position. 2. 通过Transform组件移动物体 Transform 组件用于描述物体在 ...
- unity3d中控制物体移动方法有那些及区别
1. 利用GameObject的Translate,直接改变它的Transform,前提是需要你实现准备变换矩阵.2. 用MoveTo方法,你只要知道你的目标位置即可.3. 用Math的Lerp方法计 ...
- SQL列中含有换行符的查找和替换方法
最近在获取数据时,发现程序读取的字段中含有\r\n字符,检查数据库表中的数据,发现是varchar字符串中包含了换行符.导入数据导致了这一情况出现. 回车换行 不同系统的行结尾符号并不同,如下: li ...
- Unity3D 物体移动方法总结
1. 简介 在Unity3D中,有多种方式可以改变物体的坐标,实现移动的目的,其本质是每帧修改物体的position. 2. 通过Transform组件移动物体 Transform 组件用于描述物体在 ...
- laravel 迁移文件中修改含有enum字段的表报错解决方法
解决方法: 在迁移文件中up方法最上方加上下面这一行代码即可: Schema::getConnection()->getDoctrineSchemaManager()->getDataba ...
随机推荐
- JAVA用for循环打印*三角形
public class Sanjiaoxing { //本节为for循环的嵌套结构练习 public static void main(String[] args) { // TODO Auto-g ...
- 【Srping】事务的执行原理(一)
在使用事务的时候需要添加@EnableTransactionManagement注解来开启事务,那么就从@EnableTransactionManagement入手查看一下事务的执行原理. @Enab ...
- BUUCTF-qr
qr 签到题
- Java基础-并发篇
3.1. JAVA 并发知识库 3.2. JAVA 线程实现/创建方式 3.2.1. 继承 Thread 类 Thread 类本质上是实现了 Runnable 接口的一个实例,代表一个线程的实例. ...
- 自然常数e的由来以及计算机为什么是二进制
背景 昨晚我在看一本书,叫<数学极客>,看到第六章<e:不自然的自然数>,这个数最早开始接触应该是高一的时候,那时候问老师,这个数是怎么来的,老实说,和圆周率一样,是一个常 ...
- UiPath循环活动While的介绍和使用
一.While循环的介绍 先判断条件是否满足, 如果满足, 再执行循环体, 直到判断条件不满足,则跳出循环 二.While循环在UiPath中的使用 1. 打开设计器,在设计库中新建一个Flowcha ...
- java中常见的锁
1.悲观锁 认为别的线程都会修改数据,二话不说先锁上 synchronized 2.乐观锁 乐观豁达,起初不操作.最后修改的时候比对一下版本,不一致再上锁 3.可重入锁 外层锁了之后,内层仍可以直接使 ...
- Python实现12种概率分布(附代码)
今天给大家带来的这篇文章是关于机器学习的,机器学习有其独特的数学基础,我们用微积分来处理变化无限小的函数,并计算它们的变化:我们使用线性代数来处理计算过程:我们还用概率论与统计学建模不确定性. 在这其 ...
- 如何借助Chrome Tool Dev 帮助前端开发
11111111111111111111111111111111111111111111111111
- 文件的下载,HttpMessageConverter原理
HttpMessageConverter<T> 1) HttpMessageConverter<T> 是 Spring3.0 新添加的一个接口,负责将请求信息转换为一个对象(类 ...