# Copyright: Ankitects Pty Ltd and contributors
# License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
from __future__ import annotations
import html
import os
import re
import shutil
from typing import Any, Dict, Optional
import anki
from anki import hooks
from anki.lang import _
from anki.types import NoteType, QAData
from anki.utils import call, checksum, isMac, namedtmp, stripHTML, tmpdir
pngCommands = [
["latex", "-interaction=nonstopmode", "tmp.tex"],
["dvipng", "-D", "200", "-T", "tight", "tmp.dvi", "-o", "tmp.png"],
]
svgCommands = [
["latex", "-interaction=nonstopmode", "tmp.tex"],
["dvisvgm", "--no-fonts", "--exact", "-Z", "2", "tmp.dvi", "-o", "tmp.svg"],
]
build = True # if off, use existing media but don't create new
regexps = {
"standard": re.compile(r"\[latex\](.+?)\[/latex\]", re.DOTALL | re.IGNORECASE),
"expression": re.compile(r"\[\$\](.+?)\[/\$\]", re.DOTALL | re.IGNORECASE),
"math": re.compile(r"\[\$\$\](.+?)\[/\$\$\]", re.DOTALL | re.IGNORECASE),
}
# add standard tex install location to osx
if isMac:
os.environ["PATH"] += ":/usr/texbin:/Library/TeX/texbin"
def stripLatex(text) -> Any:
for match in regexps["standard"].finditer(text):
text = text.replace(match.group(), "")
for match in regexps["expression"].finditer(text):
text = text.replace(match.group(), "")
for match in regexps["math"].finditer(text):
text = text.replace(match.group(), "")
return text
# media code and some add-ons depend on the current name
def mungeQA(
html: str,
type: str,
fields: Dict[str, str],
model: NoteType,
data: QAData,
col: anki.storage._Collection,
) -> str:
"Convert TEXT with embedded latex tags to image links."
for match in regexps["standard"].finditer(html):
html = html.replace(match.group(), _imgLink(col, match.group(1), model))
for match in regexps["expression"].finditer(html):
html = html.replace(
match.group(), _imgLink(col, "$" + match.group(1) + "$", model)
)
for match in regexps["math"].finditer(html):
html = html.replace(
match.group(),
_imgLink(
col,
"\\begin{displaymath}" + match.group(1) + "\\end{displaymath}",
model,
),
)
return html
def _imgLink(col, latex: str, model: NoteType) -> str:
"Return an img link for LATEX, creating if necesssary."
txt = _latexFromHtml(col, latex)
if model.get("latexsvg", False):
ext = "svg"
else:
ext = "png"
# is there an existing file?
fname = "latex-%s.%s" % (checksum(txt.encode("utf8")), ext)
link = '' % fname
if os.path.exists(fname):
return link
# building disabled?
if not build:
return "[latex]%s[/latex]" % latex
err = _buildImg(col, txt, fname, model)
if err:
return err
else:
return link
def _latexFromHtml(col, latex: str) -> str:
"Convert entities and fix newlines."
latex = re.sub("
|
" + html.escape(log) + "" except: msg += _("Have you installed latex and dvipng/dvisvgm?") return msg # setup q/a filter - type ignored due to import cycle hooks.rendered_card_template_filter.append(mungeQA) # type: ignore