mirror of
https://github.com/jeertmans/manim-slides.git
synced 2025-05-22 04:56:24 +08:00

* fix(lib): Manim fixes, bump to >= 0.18, and tests * chore(ci): tests and happy mypy * chore(deps): fix override * fix(tests): correct skipping * fix(ci): coverage * fix(docs): dead links * fix(tests): deps fixes * fix(deps): add missing override * fix(tests): correctly ignore * [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci * chore(tests): no filterwarning * chore(tests): add a check to see if we can install package * [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci * chore(ci): typo * fix(ci): typo * oops * [pre-commit.ci] auto fixes from pre-commit.com hooks for more information, see https://pre-commit.ci * fix(ci): double quote instead of single * chore(ci): add OSes requirements * chore(tests): removed `full-gl` extra * chore(ci): automatically cancel jobs * fix(docs): typo --------- Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com>
412 lines
12 KiB
Python
412 lines
12 KiB
Python
import random
|
|
import shutil
|
|
import sys
|
|
from pathlib import Path
|
|
from typing import Any, Union
|
|
|
|
import pytest
|
|
from click.testing import CliRunner
|
|
from manim import (
|
|
BLUE,
|
|
DOWN,
|
|
LEFT,
|
|
ORIGIN,
|
|
RIGHT,
|
|
UP,
|
|
Circle,
|
|
Dot,
|
|
FadeIn,
|
|
GrowFromCenter,
|
|
Text,
|
|
)
|
|
from manim.renderer.opengl_renderer import OpenGLRenderer
|
|
|
|
from manim_slides.config import PresentationConfig
|
|
from manim_slides.defaults import FOLDER_PATH
|
|
from manim_slides.render import render
|
|
from manim_slides.slide.manim import Slide as CESlide
|
|
|
|
|
|
class CEGLSlide(CESlide):
|
|
def __init__(self, *args: Any, **kwargs: Any) -> None:
|
|
super().__init__(*args, renderer=OpenGLRenderer(), **kwargs)
|
|
|
|
|
|
if sys.version_info >= (3, 12):
|
|
|
|
class _GLSlide:
|
|
pass
|
|
|
|
GLSlide = pytest.param(_GLSlide, marks=pytest.mark.skip())
|
|
else:
|
|
from manim_slides.slide.manimlib import Slide as GLSlide
|
|
|
|
SlideType = Union[type[CESlide], type[GLSlide], type[CEGLSlide]]
|
|
Slide = Union[CESlide, GLSlide, CEGLSlide]
|
|
|
|
|
|
@pytest.mark.parametrize(
|
|
"renderer",
|
|
[
|
|
"--CE",
|
|
pytest.param(
|
|
"--GL",
|
|
marks=pytest.mark.skipif(
|
|
sys.version_info >= (3, 12),
|
|
reason="ManimGL requires numpy<1.25, which is outdated and Python < 3.12",
|
|
),
|
|
),
|
|
],
|
|
)
|
|
def test_render_basic_slide(
|
|
renderer: str,
|
|
slides_file: Path,
|
|
presentation_config: PresentationConfig,
|
|
manimgl_config: Path,
|
|
) -> None:
|
|
runner = CliRunner()
|
|
|
|
with runner.isolated_filesystem() as tmp_dir:
|
|
shutil.copy(manimgl_config, tmp_dir)
|
|
results = runner.invoke(
|
|
render, [renderer, str(slides_file), "BasicSlide", "-ql"]
|
|
)
|
|
|
|
assert results.exit_code == 0, results
|
|
|
|
local_slides_folder = (Path(tmp_dir) / "slides").resolve(strict=True)
|
|
|
|
local_config_file = (local_slides_folder / "BasicSlide.json").resolve(
|
|
strict=True
|
|
)
|
|
|
|
local_presentation_config = PresentationConfig.from_file(local_config_file)
|
|
|
|
assert len(local_presentation_config.slides) == len(presentation_config.slides)
|
|
|
|
assert (
|
|
local_presentation_config.background_color
|
|
== presentation_config.background_color
|
|
)
|
|
|
|
assert (
|
|
local_presentation_config.background_color
|
|
== presentation_config.background_color
|
|
)
|
|
|
|
assert local_presentation_config.resolution == presentation_config.resolution
|
|
|
|
|
|
def init_slide(cls: SlideType) -> Slide:
|
|
if issubclass(cls, CESlide):
|
|
return cls()
|
|
elif issubclass(cls, GLSlide):
|
|
from manimlib.config import get_configuration, parse_cli
|
|
from manimlib.extract_scene import get_scene_config
|
|
|
|
args = parse_cli()
|
|
config = get_configuration(args)
|
|
scene_config = get_scene_config(config)
|
|
return cls(**scene_config)
|
|
|
|
raise ValueError(f"Unsupported class {cls}")
|
|
|
|
|
|
parametrize_base_cls = pytest.mark.parametrize(
|
|
"base_cls", (CESlide, GLSlide, CEGLSlide), ids=("CE", "GL", "CE(GL)")
|
|
)
|
|
|
|
|
|
def assert_constructs(cls: SlideType) -> None:
|
|
init_slide(cls).construct()
|
|
|
|
|
|
def assert_renders(cls: SlideType) -> None:
|
|
init_slide(cls).render()
|
|
|
|
|
|
class TestSlide:
|
|
def test_default_properties(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
assert self._output_folder == FOLDER_PATH
|
|
assert len(self._slides) == 0
|
|
assert self._current_slide == 1
|
|
assert self._start_animation == 0
|
|
assert len(self._canvas) == 0
|
|
assert self._wait_time_between_slides == 0.0
|
|
|
|
@parametrize_base_cls
|
|
def test_frame_height(self, base_cls: SlideType) -> None:
|
|
@assert_constructs
|
|
class _(base_cls): # type: ignore
|
|
def construct(self) -> None:
|
|
assert self._frame_height > 0 and isinstance(self._frame_height, float)
|
|
|
|
@parametrize_base_cls
|
|
def test_frame_width(self, base_cls: SlideType) -> None:
|
|
@assert_constructs
|
|
class _(base_cls): # type: ignore
|
|
def construct(self) -> None:
|
|
assert self._frame_width > 0 and isinstance(self._frame_width, float)
|
|
|
|
@parametrize_base_cls
|
|
def test_resolution(self, base_cls: SlideType) -> None:
|
|
@assert_constructs
|
|
class _(base_cls): # type: ignore
|
|
def construct(self) -> None:
|
|
pw, ph = self._resolution
|
|
assert isinstance(pw, int) and pw > 0
|
|
assert isinstance(ph, int) and ph > 0
|
|
|
|
@parametrize_base_cls
|
|
def test_backround_color(self, base_cls: SlideType) -> None:
|
|
@assert_constructs
|
|
class _(base_cls): # type: ignore
|
|
def construct(self) -> None:
|
|
assert self._background_color in ["#000000", "#000"] # DEFAULT
|
|
|
|
def test_multiple_animations_in_last_slide(self) -> None:
|
|
@assert_renders
|
|
class _(CESlide):
|
|
"""Check against solution for issue #161."""
|
|
|
|
def construct(self) -> None:
|
|
circle = Circle(color=BLUE)
|
|
dot = Dot()
|
|
|
|
self.play(GrowFromCenter(circle))
|
|
self.play(FadeIn(dot))
|
|
self.next_slide()
|
|
|
|
self.play(dot.animate.move_to(RIGHT))
|
|
self.play(dot.animate.move_to(UP))
|
|
self.play(dot.animate.move_to(LEFT))
|
|
self.play(dot.animate.move_to(DOWN))
|
|
|
|
def test_file_too_long(self) -> None:
|
|
@assert_renders
|
|
class _(CESlide):
|
|
"""Check against solution for issue #123."""
|
|
|
|
def construct(self) -> None:
|
|
circle = Circle(radius=3, color=BLUE)
|
|
dot = Dot()
|
|
self.play(GrowFromCenter(circle), run_time=0.1)
|
|
|
|
for _ in range(30):
|
|
direction = (random.random() - 0.5) * LEFT + (
|
|
random.random() - 0.5
|
|
) * UP
|
|
self.play(dot.animate.move_to(direction), run_time=0.1)
|
|
self.play(dot.animate.move_to(ORIGIN), run_time=0.1)
|
|
|
|
def test_loop(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
text = Text("Some text")
|
|
|
|
self.add(text)
|
|
|
|
assert not self._base_slide_config.loop
|
|
|
|
self.next_slide(loop=True)
|
|
self.play(text.animate.scale(2))
|
|
|
|
assert self._base_slide_config.loop
|
|
|
|
self.next_slide(loop=False)
|
|
|
|
assert not self._base_slide_config.loop
|
|
|
|
def test_auto_next(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
text = Text("Some text")
|
|
|
|
self.add(text)
|
|
|
|
assert not self._base_slide_config.auto_next
|
|
|
|
self.next_slide(auto_next=True)
|
|
self.play(text.animate.scale(2))
|
|
|
|
assert self._base_slide_config.auto_next
|
|
|
|
self.next_slide(auto_next=False)
|
|
|
|
assert not self._base_slide_config.auto_next
|
|
|
|
def test_loop_and_auto_next_succeeds(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
text = Text("Some text")
|
|
|
|
self.add(text)
|
|
|
|
self.next_slide(loop=True, auto_next=True)
|
|
self.play(text.animate.scale(2))
|
|
|
|
self.next_slide()
|
|
|
|
def test_playback_rate(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
text = Text("Some text")
|
|
|
|
self.add(text)
|
|
|
|
assert self._base_slide_config.playback_rate == 1.0
|
|
|
|
self.next_slide(playback_rate=2.0)
|
|
self.play(text.animate.scale(2))
|
|
|
|
assert self._base_slide_config.playback_rate == 2.0
|
|
|
|
def test_reversed_playback_rate(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
text = Text("Some text")
|
|
|
|
self.add(text)
|
|
|
|
assert self._base_slide_config.reversed_playback_rate == 1.0
|
|
|
|
self.next_slide(reversed_playback_rate=2.0)
|
|
self.play(text.animate.scale(2))
|
|
|
|
assert self._base_slide_config.reversed_playback_rate == 2.0
|
|
|
|
def test_notes(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
text = Text("Some text")
|
|
|
|
self.add(text)
|
|
|
|
assert self._base_slide_config.notes == ""
|
|
|
|
self.next_slide(notes="test")
|
|
self.play(text.animate.scale(2))
|
|
|
|
assert self._base_slide_config.notes == "test"
|
|
|
|
def test_wipe(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
text = Text("Some text")
|
|
bye = Text("Bye")
|
|
|
|
self.add(text)
|
|
|
|
assert text in self.mobjects
|
|
assert bye not in self.mobjects
|
|
|
|
self.wipe([text], [bye])
|
|
|
|
assert text not in self.mobjects
|
|
assert bye in self.mobjects
|
|
|
|
def test_zoom(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
text = Text("Some text")
|
|
bye = Text("Bye")
|
|
|
|
self.add(text)
|
|
|
|
assert text in self.mobjects
|
|
assert bye not in self.mobjects
|
|
|
|
self.zoom([text], [bye])
|
|
|
|
assert text not in self.mobjects
|
|
assert bye in self.mobjects
|
|
|
|
def test_animation_count(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
assert self._current_animation == 0
|
|
circle = Circle(color=BLUE)
|
|
dot = Dot()
|
|
|
|
self.play(GrowFromCenter(circle))
|
|
assert self._current_animation == 1
|
|
self.play(FadeIn(dot))
|
|
assert self._current_animation == 2
|
|
|
|
def test_wait_time_between_slides(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
self._wait_time_between_slides = 1.0
|
|
assert self._current_animation == 0
|
|
circle = Circle(color=BLUE)
|
|
self.play(GrowFromCenter(circle))
|
|
assert self._current_animation == 1
|
|
self.next_slide()
|
|
assert self._current_animation == 2 # self.wait = +1
|
|
|
|
def test_next_slide(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
assert self._current_slide == 1
|
|
self.next_slide()
|
|
assert self._current_slide == 1
|
|
circle = Circle(color=BLUE)
|
|
self.play(GrowFromCenter(circle))
|
|
self.next_slide()
|
|
assert self._current_slide == 2
|
|
self.next_slide()
|
|
assert self._current_slide == 2
|
|
|
|
def test_canvas(self) -> None:
|
|
@assert_constructs
|
|
class _(CESlide):
|
|
def construct(self) -> None:
|
|
text = Text("Some text")
|
|
bye = Text("Bye")
|
|
|
|
assert len(self.canvas) == 0
|
|
|
|
self.add(text)
|
|
|
|
assert len(self.canvas) == 0
|
|
|
|
self.add_to_canvas(text=text)
|
|
|
|
assert len(self.canvas) == 1
|
|
|
|
self.add(bye)
|
|
|
|
assert len(self.canvas) == 1
|
|
|
|
assert text not in self.mobjects_without_canvas
|
|
assert bye in self.mobjects_without_canvas
|
|
|
|
self.remove(text)
|
|
|
|
assert len(self.canvas) == 1
|
|
|
|
self.add_to_canvas(bye=bye)
|
|
|
|
assert len(self.canvas) == 2
|
|
|
|
self.remove_from_canvas("text", "bye")
|
|
|
|
assert len(self.canvas) == 0
|
|
|
|
with pytest.raises(KeyError):
|
|
self.remove_from_canvas("text")
|