【manim动画教程】–相机

yizhihongxing

相机(Camera)在二维的场景下使用不多,一般在3D场景中提及的比较多。
相机相当于我们看动画的视角,简单来理解的话,相当于我们的眼睛(实际情况会复杂一些,相机还有其他一些辅助功能)。

默认的相机焦点在屏幕的中心位置,相机默认是以俯视的视角查看所有的元素。
之前的介绍的常用动画效果高级动画效果,都没有对相机进行调整过,
所以,是通过移动和变换各个元素来实现动画效果。

本篇介绍的相机,则是另一种制作动画的方式,它不改变元素在屏幕或者说在坐标系中的位置,
通过改变相机的位置和角度来实现动画效果。

举个现实世界中的简单例子,如果有个杯子,我们可以通过转动杯子来从各个角度观察杯子;
而如果是一栋楼的话,我们无法移动它,只能围着楼走一圈来观察它,这个过程就相当于移动相机。

下面通过一些实例来看看移动相机带来的不一样的动画效果。

1. 相机移动

相机移动常用的两种方式:

  1. 移动焦点:改变相机的焦点,焦点在那个元素,那个元素就会在屏幕中心
  2. 改变视野:改变相机与元素的距离,离得越远,物体越小。

1.1 移动焦点

manim中移动焦点要继承 MovingCameraScene类,
然后通过 self.camera.frame.animate.move_to函数来移动焦点。

下面的示例构造了一个正方形,一个三角形,然后通过改变焦点来形成元素移动的动画。
实际上元素并没有移动,它们的坐标始终没变,变化的是相机的焦点。

class CameraSample1(MovingCameraScene):
    def _move_focus(self):
        s = Square(color=RED, fill_opacity=0.5)
        t = Triangle(color=GREEN, fill_opacity=0.5)
        vg = VGroup(s, t)
        vg.arrange(RIGHT, buff=MED_LARGE_BUFF)
        self.add(vg)

        self.play(self.camera.frame.animate.move_to(s))
        self.play(self.camera.frame.animate.move_to(t))
        self.play(self.camera.frame.animate.move_to(vg))

    def construct(self):
        self._move_focus()

        self.wait()

运行效果:
相机移动-移动焦点.gif

1.2 改变视野

改变视野通过 self.camera.frame.animate.set方法,通过这个方法设置视野的宽度,可以形成缩放元素的效果。

下面的示例,通过改变视野的宽度,让元素出现放大和缩小的现象,实际上元素并没有变化。
变化的是相机到元素的距离。

class CameraSample1(MovingCameraScene):
    def _scale(self):
        s = Square(color=RED, fill_opacity=0.5)
        self.add(s)

        self.camera.frame.save_state()
        self.play(self.camera.frame.animate.set(width=s.width * 2))
        self.wait(0.3)
        self.play(self.camera.frame.animate.set(width=s.width * 8))
        self.wait(0.3)
        self.play(Restore(self.camera.frame))

    def construct(self):
        self._scale()

        self.wait()

运行效果:
相机移动-改变视野.gif

2. 鹰眼效果

鹰眼的效果是通过两个相机来实现的,两个相机的焦点一样,但是视野不一样。
鹰眼效果一般用在提供全局视图的场合,特别是当元素特别多的时候。

下面示例中,设置了两个相机,缩放的参数为3,zoom_factor=3

class CameraSample2(ZoomedScene):
    def __init__(self, **kwargs):
        ZoomedScene.__init__(
            self,
            zoom_factor=3,
            zoomed_display_height=1,
            zoomed_display_width=2,
            image_frame_stroke_width=5,
            zoomed_camera_config={
                "default_frame_stroke_width": 3,
            },
            **kwargs
        )

    def construct(self):
        s = Square(color=RED, fill_opacity=0.5, side_length=1.5)
        t = Triangle(color=GREEN, fill_opacity=0.5).scale(0.5)
        vg = VGroup(s, t)
        vg.arrange(RIGHT, buff=SMALL_BUFF)
        self.add(vg)
        self.activate_zooming(animate=False)
        self.play(s.animate.shift(LEFT), t.animate.shift(RIGHT))
        self.play(s.animate.rotate(2 * PI / 3), t.animate.rotate(PI / 2))
        self.play(s.animate.shift(RIGHT), t.animate.shift(LEFT))
        self.wait()

运行效果:
鹰眼效果.gif

3. 追踪物体

追踪物体就是将相机的焦点定位在移动的物体上,就像我们坐在火车上的感觉一样,那时,我们觉得火车没动,而是车外的风景不断向后移动。

下面的示例是一个点沿着正弦曲线运动,我们将相机焦点定位在这个点上,感觉就像是曲线在移动。

class CameraSample3(MovingCameraScene):
    def construct(self):
        self.camera.frame.save_state()

        graph = FunctionGraph(
            lambda x: np.sin(x),
            x_range=[-3, 3],
            color=RED,
        )
        d = dot(graph.get_start())
        self.add(graph, d)

        self.play(self.camera.frame.animate.scale(0.5).move_to(d))

        def update_curve(mob):
            mob.move_to(d.get_center())

        self.camera.frame.add_updater(update_curve)
        self.play(MoveAlongPath(d, graph), rate_func=linear, run_time=2)
        self.camera.frame.remove_updater(update_curve)

        self.play(Restore(self.camera.frame))

运行效果:
追踪物体.gif

4. 3D 场景

相机其实主要就是应用在3D场景中的,所以 manim的3D场景类 ThreeDScene中提供了一个非常方便的移动相机的方法 move_camera

下面的示例中,我们用 move_camera方法来改变视角和调整视野。
示例中的球其实一直没动,也就是球上各点的坐标没有改变过。

class CameraSample4(ThreeDScene):
    def construct(self):
        axes = ThreeDAxes()
        sphere = Surface(
            lambda u, v: np.array(
                [
                    1.5 * np.cos(u) * np.cos(v),
                    1.5 * np.cos(u) * np.sin(v),
                    1.5 * np.sin(u),
                ]
            ),
            v_range=[0, TAU],
            u_range=[-PI / 2, PI / 2],
            checkerboard_colors=[BLUE_D, BLUE_E],
            resolution=(15, 32),
        )
        self.add(axes, sphere)

        self.move_camera(phi=75 * DEGREES, theta=30 * DEGREES)
        self.move_camera(zoom=1.5)
        self.move_camera(zoom=0.5)
        self.wait()

运行效果:
3D场景.gif

5. 总结回顾

本篇介绍从另一种角度来实现动画的方式,也就是不改变物体本身,而是改变观察物体的方式。

manim中能够操作相机的类主要有:

  1. MovingCameraScene:改变相机的焦点和视野
  2. ZoomedScene:增加多个相机,多个相机可以从不同的视角同时观察物体
  3. ThreeDScene:3D场景下的相机操作

本文关联的微信视频号短视频:
manim-相机-视频号.png

原文链接:https://www.cnblogs.com/wang_yb/p/17334029.html

本站文章如无特殊说明,均为本站原创,如若转载,请注明出处:【manim动画教程】–相机 - Python技术站

(0)
上一篇 2023年4月19日
下一篇 2023年4月20日

相关文章

  • Python中TypeError:unhashable type:’dict’错误的解决办法

    当我们在使用Python进行开发时,有时候会遇到 “TypeError:unhashabletype:’dict’” 错误,这个错误一般是由于我们将一个字典作为某些操作函数的输入参数,并将这个字典作为空间的 key 进行 hash 计算导致的。下面我将为大家介绍解决这个错误的方法。 1. 错误原因 在 Python 中,一般而言我们需要将某些函数的输入数据进…

    python 2023年5月13日
    00
  • python变量命名的7条建议

    以下是Python变量命名的7条建议的详细攻略: 1. 变量名应当有意义,易于理解 在选择变量名时应该避免使用单个字母或缩写,而应该使用能够清晰表达变量用途的单词或短语,这有助于提高代码的可读性和易于理解程度。如: # 不好的变量名 a = 1 b = 2 # 好的变量名 width = 1 height = 2 2. 变量名应该遵循命名规范 Python有…

    python 2023年6月3日
    00
  • Python HTTP客户端自定义Cookie实现实例

    Python HTTP 客户端自定义 Cookie 实现实例 在 Python 中,可以使用 requests 模块发送 HTTP 请求,并自定义 Cookie。以下是 Python HTTP 客户端自定义 Cookie 实现实例。 1. 使用 cookies 参数 在使用 requests 模块发送 HTTP 请求时,可以使用 cookies 参数来自定义…

    python 2023年5月15日
    00
  • Python利用Pillow(PIL)库实现验证码图片的全过程

    下面是关于“Python利用Pillow(PIL)库实现验证码图片的全过程”的攻略: Pillow(PIL)库简介 Pillow(PIL)是Python的一个图像处理库,可以对图片进行基础的操作,比如打开、保存、裁剪、旋转、缩放、加文字等处理。本文将示范如何使用Pillow库生成验证码图片。 生成验证码图片的过程 1. 导入Pillow库相关模块 from …

    python 2023年5月18日
    00
  • Python 中打印字典中的所有键值对的示例详解

    下面我来为您详细讲解“Python 中打印字典中的所有键值对的示例详解”的完整攻略。 1. 简介 Python 字典(dictionary)是一种非常常用的数据类型,它包含多个键值对,即将一些键和它们对应的数据值联系在一起。在 Python 中,我们可以使用 for 循环语句来遍历字典中的所有键值对,并将它们依次打印出来。下面,我们就来看看具体的实现方法。 …

    python 2023年5月13日
    00
  • 详解Python中的编码问题(encoding与decode、str与bytes)

    详解Python中的编码问题 在Python中,经常会用到编码相关的操作,因此理解编码问题是很重要的。本篇攻略将详解Python中的编码问题,包括编码与解码(encoding与decode)、str与bytes等主要内容。 编码与解码 编码通常指将字符串转换为字节序列,解码则是将字节序列转换为字符串。在Python中,有两个内置的函数,可以进行编码与解码的操…

    python 2023年5月20日
    00
  • MongoDB安装使用并实现Python操作数据库

    MongoDB是一个开源的、高性能稳定的NoSQL数据库,支持跨平台,提供了丰富的数据结构和查询方式,被广泛应用于大数据存储和处理中。本文将详细讲解如何安装、使用MongoDB,并使用Python编写操作MongoDB的代码。 安装MongoDB MongoDB提供了Windows、Linux、MacOS等多个平台的安装包,可以在官网下载最新版本的安装包。以…

    python 2023年5月14日
    00
  • 如何让python的运行速度得到提升

    提升Python运行速度的攻略: 使用更高效的算法和数据结构 对于相同的问题,使用不同的算法和数据结构可以对 Python 的运行速度有显著的影响。任何时候,当我们需要处理大量数据时,都需要牢记这一点。以下这些算法和数据结构可以帮助提高 Python 的程序的运行速度: 二分查找:二分查找比线性查找要快得多,因为它的时间复杂度是O(log n)。在输入数据量…

    python 2023年5月18日
    00
合作推广
合作推广
分享本页
返回顶部