Transform

注意

只要读者看到这一部分的动画,就会知道 Transform 对文字是有多么炸裂。因此对于文字的动画,建议还是使用其他的方式。

Transform

class manimlib.animation.transform.Transform(mobject: Mobject, target_mobject: Mobject | None = None, **kwargs)
get_all_mobjects() list[Mobject]

Ordering must match the ording of arguments to interpolate_submobject

TransformExample
class TransformExample(Scene):
    def construct(self):
        A = Text("Text-A").scale(3)
        B = Text("Text-B").scale(3)
        C = Text("C-Text").scale(3)

        self.add(A)
        self.wait()
        self.play(Transform(A, B))
        self.wait()
        self.play(Transform(A, C)) # notice here
        self.wait()

ReplacementTransform

class manimlib.animation.transform.ReplacementTransform(mobject: Mobject, target_mobject: Mobject | None = None, **kwargs)
ReplacementTransformExample
class ReplacementTransformExample(Scene):
    def construct(self):
        A = Text("Text-A").scale(3)
        B = Text("Text-B").scale(3)
        C = Text("C-Text").scale(3)

        self.add(A)
        self.wait()
        self.play(ReplacementTransform(A, B))
        self.wait()
        self.play(ReplacementTransform(B, C)) # notice here
        self.wait()

Transform和ReplacementTransform的区别

  1. Transform(A, B) 在画面上 A 变成了 B 的样子,但是画面上的物体名字还叫 A

  2. ReplacementTransform(A, B) 在画面上 A 变成了 B 的样子,并且画面上的物体名字叫 B

所以以下两个效果相同

self.play(Transform(A, B))
self.play(Transform(A, C))
self.play(ReplacementTransform(A, B))
self.play(ReplacementTransform(B, C))

TransformFromCopy

class manimlib.animation.transform.TransformFromCopy(mobject: Mobject, target_mobject: Mobject, **kwargs)

Performs a reversed Transform

TransformFromCopyExample
class TransformFromCopyExample(Scene):
    def construct(self):
        A = Text("Text-A").scale(3)
        B = Text("Text-B").scale(3).shift(UP*2)

        self.add(A)
        self.wait()
        self.play(TransformFromCopy(A, B))
        self.wait()

ClockwiseTransform

class manimlib.animation.transform.ClockwiseTransform(mobject: Mobject, target_mobject: Mobject | None = None, **kwargs)
ClockwiseTransformExample
class ClockwiseTransformExample(Scene):
    def construct(self):
        A = Text("Text-A").scale(3)
        B = Text("Text-B").scale(3).shift(UP*2)

        self.add(A)
        self.wait()
        self.play(ClockwiseTransform(A, B))
        self.wait()

CounterclockwiseTransform

class manimlib.animation.transform.CounterclockwiseTransform(mobject: Mobject, target_mobject: Mobject | None = None, **kwargs)
CounterclockwiseTransformExample
class CounterclockwiseTransformExample(Scene):
    def construct(self):
        A = Text("Text-A").scale(3)
        B = Text("Text-B").scale(3).shift(UP*2)

        self.add(A)
        self.wait()
        self.play(CounterclockwiseTransform(A, B))
        self.wait()

MoveToTarget

class manimlib.animation.transform.MoveToTarget(mobject: Mobject, **kwargs)
MoveToTargetExample
class MoveToTargetExample(Scene):
    def construct(self):
        A = Text("Text-A").to_edge(LEFT)
        A.generate_target()  # copyA自身形成A的target属性
        A.target.scale(3).shift(RIGHT*7+UP*2) # 操作A的target

        self.add(A)
        self.wait()
        self.play(MoveToTarget(A))
        self.wait()

另外,直接使用 self.play(mob.method, ...) 相当于给mob创建target,然后根据method 操作target。并且这样能对同一个mobject多次操作。当然,也可以用 self.play(mob.animate.shift(UP).scale(2)) 进行链式操作(而 ApplyMethod 不可以,见下)

SelfPlayExample
class SelfPlayExample(Scene):
    def construct(self):
        A = TextMobject("Text-A").to_edge(LEFT)

        self.add(A)
        self.wait()
        # self.play(
        #     A.scale, 3,
        #     A.shift, RIGHT*7+UP*2,
        # )
        # 上面为旧版用法,在 Manim Community 引入 animate 方法后,我们更青睐下面的方法
        self.play(
              A.animate.scale(3).shift(RIGHT*7+UP*2)
        )
        self.wait()

ApplyMethod

class manimlib.animation.transform.ApplyMethod(method: Callable, *args, **kwargs)

method is a method of Mobject, *args are arguments for that method. Key word arguments should be passed in as the last arg, as a dict, since **kwargs is for configuration of the transform itself

Relies on the fact that mobject methods return the mobject

ApplyMethodExample
class ApplyMethodExample(Scene):
    def construct(self):
        A = Text("Text-A").to_edge(LEFT)

        self.add(A)
        self.wait()
        self.play(
            ApplyMethod(A.scale, 3), # 这个不会执行
            ApplyMethod(A.shift, RIGHT*7+UP*2),
        )
        self.wait()

因为 ApplyMethod 是Transform的子类,所以每次只能对同一个物体执行一次操作(最后一次)

ApplyPointwiseFunction

class manimlib.animation.transform.ApplyPointwiseFunction(function: Callable[[np.ndarray], np.ndarray], mobject: Mobject, **kwargs)

method is a method of Mobject, *args are arguments for that method. Key word arguments should be passed in as the last arg, as a dict, since **kwargs is for configuration of the transform itself

Relies on the fact that mobject methods return the mobject

ApplyPointwiseFunctionExample
class ApplyPointwiseFunctionExample(Scene):
    def construct(self):
        def trans(p: np.ndarray) -> np.ndarray:
            return np.array([
                p[0] + np.sin(p[1]),
                p[1] + np.sin(p[0]),
                p[2]
            ])
        plane = NumberPlane()
        self.add(plane)
        self.wait()
        plane.prepare_for_nonlinear_transform()
        self.play(ApplyPointwiseFunction(trans, plane))
        self.wait()

ApplyPointwiseFunctionToCenter

class manimlib.animation.transform.ApplyPointwiseFunctionToCenter(function: Callable[[np.ndarray], np.ndarray], mobject: Mobject, **kwargs)

method is a method of Mobject, *args are arguments for that method. Key word arguments should be passed in as the last arg, as a dict, since **kwargs is for configuration of the transform itself

Relies on the fact that mobject methods return the mobject

FadeToColor

class manimlib.animation.transform.FadeToColor(mobject: Mobject, color: ManimColor, **kwargs)

method is a method of Mobject, *args are arguments for that method. Key word arguments should be passed in as the last arg, as a dict, since **kwargs is for configuration of the transform itself

Relies on the fact that mobject methods return the mobject

ScaleInPlace

class manimlib.animation.transform.ScaleInPlace(mobject: Mobject, scale_factor: npt.ArrayLike, **kwargs)

method is a method of Mobject, *args are arguments for that method. Key word arguments should be passed in as the last arg, as a dict, since **kwargs is for configuration of the transform itself

Relies on the fact that mobject methods return the mobject

ShrinkToCenter

class manimlib.animation.transform.ShrinkToCenter(mobject: Mobject, **kwargs)

method is a method of Mobject, *args are arguments for that method. Key word arguments should be passed in as the last arg, as a dict, since **kwargs is for configuration of the transform itself

Relies on the fact that mobject methods return the mobject

ShrinkToCenterExample
class ShrinkToCenterExample(Scene):
    def construct(self):
        mobjects = VGroup(
            Square(),
            RegularPolygon(fill_opacity=1, color=GREEN),
            Text("Text").scale(2)
        )
        mobjects.scale(1.5)
        mobjects.arrange(RIGHT, buff=2)

        self.play(
            *[ShrinkToCenter(mob) for mob in mobjects]
        )

        self.wait()

Restore

class manimlib.animation.transform.Restore(mobject: Mobject, **kwargs)

method is a method of Mobject, *args are arguments for that method. Key word arguments should be passed in as the last arg, as a dict, since **kwargs is for configuration of the transform itself

Relies on the fact that mobject methods return the mobject

RestoreExample
class RestoreExample(Scene):
    def construct(self):
        A = Text("Text-A").to_edge(LEFT)
        A.save_state()  # 记录下现在状态,restore会回到此时
        A.scale(3).shift(RIGHT*7+UP*2)

        self.add(A)
        self.wait()
        self.play(Restore(A))
        self.wait()

ApplyFunction

class manimlib.animation.transform.ApplyFunction(function: Callable[[Mobject], Mobject], mobject: Mobject, **kwargs)
ApplyFunctionExample
class ApplyFunctionExample(Scene):
    def construct(self):
        A = Text("Text-A").to_edge(LEFT)
        def function(mob):
            return mob.scale(3).shift(RIGHT*7+UP*2)
            # 需要return一个mobject

        self.add(A)
        self.wait()
        self.play(ApplyFunction(function, A))
        self.wait()

ApplyMatrix

class manimlib.animation.transform.ApplyMatrix(matrix: npt.ArrayLike, mobject: Mobject, **kwargs)

method is a method of Mobject, *args are arguments for that method. Key word arguments should be passed in as the last arg, as a dict, since **kwargs is for configuration of the transform itself

Relies on the fact that mobject methods return the mobject

ApplyMatrixExample
class ApplyMatrixExample(Scene):
    def construct(self):
        A = Text("Text-A").scale(2)
        mat = np.array([
            [1, 3, 1],
            [0.5, 1, 1],
            [1, 1, 1]
        ])

        self.add(A)
        self.wait()
        self.play(ApplyMatrix(mat, A))
        self.wait()

ApplyComplexFunction

class manimlib.animation.transform.ApplyComplexFunction(function: Callable[[complex], complex], mobject: Mobject, **kwargs)

method is a method of Mobject, *args are arguments for that method. Key word arguments should be passed in as the last arg, as a dict, since **kwargs is for configuration of the transform itself

Relies on the fact that mobject methods return the mobject

ApplyComplexFunctionExample
class ApplyComplexFunctionExample(Scene):
    def construct(self):
        A = NumberPlane()
        A.prepare_for_nonlinear_transform()

        def complex_func(z: complex) -> complex:
            return z**3

        self.add(A)
        self.wait()
        self.play(ApplyComplexFunction(complex_func, A), run_time=3)
        self.wait()

CyclicReplace

class manimlib.animation.transform.CyclicReplace(*mobjects: Mobject, **kwargs)
CyclicReplaceExample
class CyclicReplaceExample(Scene):
    def construct(self):
        A = Text("Text-A").scale(3)
        B = Text("Text-B").scale(3)
        VGroup(A, B).arrange(RIGHT)

        self.add(A, B)
        self.wait()
        self.play(CyclicReplace(A, B)) # 或Swap(A, B)
        self.wait()

Swap

class manimlib.animation.transform.Swap(*mobjects: Mobject, **kwargs)

与上面的 CyclicReplace 一样

TransformAnimations

class manimlib.animation.transform.TransformAnimations(start_anim: Animation, end_anim: Animation, **kwargs)

这个不常用,甚至Grant都怀疑它存在的意义