##// END OF EJS Templates
Merge pull request #1861 from tkf/dvipng...
Merge pull request #1861 from tkf/dvipng Use dvipng to format sympy.Matrix, enabling display of matrices in the Qt console with the sympy printing extension. Two changes are introduced: new dvipng backend is added to latex_to_png PNG formatter for sympy.Matrix uses the dvipng backend

File last commit:

r7330:b1cc95fc
r7409:7cd55e57 merge
Show More
latextools.py
181 lines | 4.9 KiB | text/x-python | PythonLexer
Brian Granger
More improvements to the display system....
r3279 # -*- coding: utf-8 -*-
"""Tools for handling LaTeX.
Authors:
* Brian Granger
"""
#-----------------------------------------------------------------------------
Takafumi Arakaki
Copyright fix on sympyprinting.py and latextools.py...
r7311 # Copyright (C) 2010 IPython Development Team.
Brian Granger
More improvements to the display system....
r3279 #
# Distributed under the terms of the Modified BSD License.
#
# The full license is in the file COPYING.txt, distributed with this software.
#-----------------------------------------------------------------------------
#-----------------------------------------------------------------------------
# Imports
#-----------------------------------------------------------------------------
from StringIO import StringIO
from base64 import encodestring
Takafumi Arakaki
Add dvipng backend to latex_to_png
r7306 import os
import tempfile
import shutil
import subprocess
Brian Granger
More improvements to the display system....
r3279
Takafumi Arakaki
Check if latex/dvipng exist before calling them
r7330 from IPython.utils.process import find_cmd, FindCmdError
Brian Granger
More improvements to the display system....
r3279 #-----------------------------------------------------------------------------
# Tools
#-----------------------------------------------------------------------------
Takafumi Arakaki
Add dvipng backend to latex_to_png
r7306 def latex_to_png(s, encode=False, backend='mpl'):
"""Render a LaTeX string to PNG.
Brian Granger
More improvements to the display system....
r3279
Parameters
----------
s : str
The raw string containing valid inline LaTeX.
encode : bool, optional
Should the PNG data bebase64 encoded to make it JSON'able.
Takafumi Arakaki
Add dvipng backend to latex_to_png
r7306 backend : {mpl, dvipng}
Backend for producing PNG data.
Takafumi Arakaki
latex_to_png returns None when the backend cannot be used...
r7309 None is returned when the backend cannot be used.
Brian Granger
More improvements to the display system....
r3279 """
Takafumi Arakaki
Add dvipng backend to latex_to_png
r7306 if backend == 'mpl':
f = latex_to_png_mpl
elif backend == 'dvipng':
f = latex_to_png_dvipng
else:
raise ValueError('No such backend {0}'.format(backend))
bin_data = f(s)
Takafumi Arakaki
latex_to_png returns None when the backend cannot be used...
r7309 if encode and bin_data:
Takafumi Arakaki
Add dvipng backend to latex_to_png
r7306 bin_data = encodestring(bin_data)
return bin_data
def latex_to_png_mpl(s):
Takafumi Arakaki
latex_to_png returns None when the backend cannot be used...
r7309 try:
from matplotlib import mathtext
except ImportError:
return None
Brian Granger
More improvements to the display system....
r3279
mt = mathtext.MathTextParser('bitmap')
f = StringIO()
mt.to_png(f, s, fontsize=12)
Takafumi Arakaki
Add dvipng backend to latex_to_png
r7306 return f.getvalue()
def latex_to_png_dvipng(s):
try:
Takafumi Arakaki
Check if latex/dvipng exist before calling them
r7330 find_cmd('latex')
find_cmd('dvipng')
except FindCmdError:
return None
try:
Takafumi Arakaki
Add dvipng backend to latex_to_png
r7306 workdir = tempfile.mkdtemp()
tmpfile = os.path.join(workdir, "tmp.tex")
dvifile = os.path.join(workdir, "tmp.dvi")
outfile = os.path.join(workdir, "tmp.png")
with open(tmpfile, "w") as f:
f.write(_latex_header)
f.write(s)
f.write(_latex_footer)
subprocess.check_call(
Takafumi Arakaki
Suppress output from latex and dvipng command
r7329 ["latex", "-halt-on-errror", tmpfile], cwd=workdir,
stdout=subprocess.PIPE, stderr=subprocess.PIPE)
Takafumi Arakaki
Add dvipng backend to latex_to_png
r7306
subprocess.check_call(
["dvipng", "-T", "tight", "-x", "1500", "-z", "9",
Takafumi Arakaki
Suppress output from latex and dvipng command
r7329 "-bg", "transparent", "-o", outfile, dvifile], cwd=workdir,
stdout=subprocess.PIPE, stderr=subprocess.PIPE)
Takafumi Arakaki
Add dvipng backend to latex_to_png
r7306
with open(outfile) as f:
bin_data = f.read()
finally:
shutil.rmtree(workdir)
Brian Granger
More improvements to the display system....
r3279 return bin_data
Brian Granger
Added mdboom's math_to_image from matplotlib.
r3881
Takafumi Arakaki
Add dvipng backend to latex_to_png
r7306 _latex_header = r'''
\documentclass{article}
\usepackage{amsmath}
\usepackage{amsthm}
\usepackage{amssymb}
\usepackage{bm}
\pagestyle{empty}
\begin{document}
'''
_latex_footer = r'\end{document}'
Brian Granger
More improvements to the display system....
r3279 _data_uri_template_png = """<img src="data:image/png;base64,%s" alt=%s />"""
def latex_to_html(s, alt='image'):
"""Render LaTeX to HTML with embedded PNG data using data URIs.
Parameters
----------
s : str
The raw string containing valid inline LateX.
alt : str
The alt text to use for the HTML.
"""
base64_data = latex_to_png(s, encode=True)
Takafumi Arakaki
latex_to_png returns None when the backend cannot be used...
r7309 if base64_data:
return _data_uri_template_png % (base64_data, alt)
Brian Granger
More improvements to the display system....
r3279
Brian Granger
Added mdboom's math_to_image from matplotlib.
r3881
# From matplotlib, thanks to mdboom. Once this is in matplotlib releases, we
# will remove.
def math_to_image(s, filename_or_obj, prop=None, dpi=None, format=None):
"""
Given a math expression, renders it in a closely-clipped bounding
box to an image file.
*s*
A math expression. The math portion should be enclosed in
dollar signs.
*filename_or_obj*
A filepath or writable file-like object to write the image data
to.
*prop*
If provided, a FontProperties() object describing the size and
style of the text.
*dpi*
Override the output dpi, otherwise use the default associated
with the output format.
*format*
The output format, eg. 'svg', 'pdf', 'ps' or 'png'. If not
provided, will be deduced from the filename.
"""
from matplotlib import figure
# backend_agg supports all of the core output formats
from matplotlib.backends import backend_agg
from matplotlib.font_manager import FontProperties
from matplotlib.mathtext import MathTextParser
if prop is None:
prop = FontProperties()
parser = MathTextParser('path')
width, height, depth, _, _ = parser.parse(s, dpi=72, prop=prop)
fig = figure.Figure(figsize=(width / 72.0, height / 72.0))
fig.text(0, depth/height, s, fontproperties=prop)
backend_agg.FigureCanvasAgg(fig)
fig.savefig(filename_or_obj, dpi=dpi, format=format)
return depth