mirror of
https://github.com/3b1b/manim.git
synced 2025-04-13 09:47:07 +00:00

* Comment tweak * Directly print traceback Since the shell.showtraceback is giving some issues * Make InteracrtiveSceneEmbed into a class This way it can keep track of it's internal shell; use of get_ipython has a finicky relationship with reloading. * Move remaining checkpoint_paste logic into scene_embed.py This involved making a few context managers for Scene: temp_record, temp_skip, temp_progress_bar, which seem useful in and of themselves. * Change null key to be the empty string * Ensure temporary svg paths for Text are deleted * Remove unused dict_ops.py functions * Remove break_into_partial_movies from file_writer configuration * Rewrite guarantee_existence using Path * Clean up SceneFileWriter It had a number of vestigial functions no longer used, and some setup that could be made more organized. * Remove --save_pngs CLI arg (which did nothing) * Add --subdivide CLI arg * Remove add_extension_if_not_present * Remove get_sorted_integer_files * Have find_file return Path * Minor clean up * Clean up num_tex_symbols * Fix find_file * Minor cleanup for extract_scene.py * Add preview_frame_while_skipping option to scene config * Use shell.showtraceback function * Move keybindings to config, instead of in-place constants * Replace DEGREES -> DEG * Add arg to clear the cache * Separate out full_tex_to_svg from tex_to_svg And only cache to disk the results of full_tex_to_svg. Otherwise, making edits to the tex_templates would not show up without clearing the cache. * Bug fix in handling BlankScene * Make checkpoint_states an instance variable of CheckpointManager As per https://github.com/3b1b/manim/issues/2272 * Move resizing out of Window.focus, and into Window.init_for_scene * Make default output directory "." instead of "" To address https://github.com/3b1b/manim/issues/2261 * Remove input_file_path arg from SceneFileWriter * Use Dict syntax in place of dict for config more consistently across config.py * Simplify get_output_directory * Swap order of preamble and additional preamble * Minor stylistic tweak * Have UnitInterval pass on kwargs to NumberLine * Add simple get_dist function * Have TracedPath always update to the stroke configuration passed in * Have Mobject.match_points apply to all parts of data in pointlike_data_key * Always call Mobject.update upon adding an updater * Add Surface.uv_to_point * Make sure Surface.set_opacity takes in a recurse option * Update num_tex_symbols to account for \{ and \}
151 lines
4.2 KiB
Python
151 lines
4.2 KiB
Python
from __future__ import annotations
|
|
|
|
import os
|
|
import re
|
|
import yaml
|
|
import subprocess
|
|
from functools import lru_cache
|
|
|
|
from pathlib import Path
|
|
import tempfile
|
|
|
|
from manimlib.utils.cache import cache_on_disk
|
|
from manimlib.config import manim_config
|
|
from manimlib.config import get_manim_dir
|
|
from manimlib.logger import log
|
|
from manimlib.utils.simple_functions import hash_string
|
|
|
|
|
|
def get_tex_template_config(template_name: str) -> dict[str, str]:
|
|
name = template_name.replace(" ", "_").lower()
|
|
template_path = os.path.join(get_manim_dir(), "manimlib", "tex_templates.yml")
|
|
with open(template_path, encoding="utf-8") as tex_templates_file:
|
|
templates_dict = yaml.safe_load(tex_templates_file)
|
|
if name not in templates_dict:
|
|
log.warning(f"Cannot recognize template {name}, falling back to 'default'.")
|
|
name = "default"
|
|
return templates_dict[name]
|
|
|
|
|
|
@lru_cache
|
|
def get_tex_config(template: str = "") -> tuple[str, str]:
|
|
"""
|
|
Returns a compiler and preamble to use for rendering LaTeX
|
|
"""
|
|
template = template or manim_config.tex.template
|
|
config = get_tex_template_config(template)
|
|
return config["compiler"], config["preamble"]
|
|
|
|
|
|
def get_full_tex(content: str, preamble: str = ""):
|
|
return "\n\n".join((
|
|
"\\documentclass[preview]{standalone}",
|
|
preamble,
|
|
"\\begin{document}",
|
|
content,
|
|
"\\end{document}"
|
|
)) + "\n"
|
|
|
|
|
|
@lru_cache(maxsize=128)
|
|
def latex_to_svg(
|
|
latex: str,
|
|
template: str = "",
|
|
additional_preamble: str = "",
|
|
short_tex: str = "",
|
|
show_message_during_execution: bool = True,
|
|
) -> str:
|
|
"""Convert LaTeX string to SVG string.
|
|
|
|
Args:
|
|
latex: LaTeX source code
|
|
template: Path to a template LaTeX file
|
|
additional_preamble: String including any added "\\usepackage{...}" style imports
|
|
|
|
Returns:
|
|
str: SVG source code
|
|
|
|
Raises:
|
|
LatexError: If LaTeX compilation fails
|
|
NotImplementedError: If compiler is not supported
|
|
"""
|
|
if show_message_during_execution:
|
|
message = f"Writing {(short_tex or latex)[:70]}..."
|
|
else:
|
|
message = ""
|
|
|
|
compiler, preamble = get_tex_config(template)
|
|
|
|
preamble = "\n".join([preamble, additional_preamble])
|
|
full_tex = get_full_tex(latex, preamble)
|
|
return full_tex_to_svg(full_tex, compiler, message)
|
|
|
|
|
|
@cache_on_disk
|
|
def full_tex_to_svg(full_tex: str, compiler: str = "latex", message: str = ""):
|
|
if message:
|
|
print(message, end="\r")
|
|
|
|
if compiler == "latex":
|
|
dvi_ext = ".dvi"
|
|
elif compiler == "xelatex":
|
|
dvi_ext = ".xdv"
|
|
else:
|
|
raise NotImplementedError(f"Compiler '{compiler}' is not implemented")
|
|
|
|
# Write intermediate files to a temporary directory
|
|
with tempfile.TemporaryDirectory() as temp_dir:
|
|
tex_path = Path(temp_dir, "working").with_suffix(".tex")
|
|
dvi_path = tex_path.with_suffix(dvi_ext)
|
|
|
|
# Write tex file
|
|
tex_path.write_text(full_tex)
|
|
|
|
# Run latex compiler
|
|
process = subprocess.run(
|
|
[
|
|
compiler,
|
|
"-no-pdf",
|
|
"-interaction=batchmode",
|
|
"-halt-on-error",
|
|
f"-output-directory={temp_dir}",
|
|
tex_path
|
|
],
|
|
capture_output=True,
|
|
text=True
|
|
)
|
|
|
|
if process.returncode != 0:
|
|
# Handle error
|
|
error_str = ""
|
|
log_path = tex_path.with_suffix(".log")
|
|
if log_path.exists():
|
|
content = log_path.read_text()
|
|
error_match = re.search(r"(?<=\n! ).*\n.*\n", content)
|
|
if error_match:
|
|
error_str = error_match.group()
|
|
raise LatexError(error_str or "LaTeX compilation failed")
|
|
|
|
# Run dvisvgm and capture output directly
|
|
process = subprocess.run(
|
|
[
|
|
"dvisvgm",
|
|
dvi_path,
|
|
"-n", # no fonts
|
|
"-v", "0", # quiet
|
|
"--stdout", # output to stdout instead of file
|
|
],
|
|
capture_output=True
|
|
)
|
|
|
|
# Return SVG string
|
|
result = process.stdout.decode('utf-8')
|
|
|
|
if message:
|
|
print(" " * len(message), end="\r")
|
|
|
|
return result
|
|
|
|
|
|
class LatexError(Exception):
|
|
pass
|