在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