r"""
Convert a restructured text document to html.
Inline math markup can uses the *math* directive, or it can use latex
style *\$expression\$*. Math can be rendered using simple html and
unicode, or with mathjax.
"""
import re
from contextlib import contextmanager
# CRUFT: locale.getlocale() fails on some versions of OS X
# See https://bugs.python.org/issue18378
import locale
if hasattr(locale, '_parse_localename'):
try:
locale._parse_localename('UTF-8')
except ValueError:
_old_parse_localename = locale._parse_localename
def _parse_localename(localename):
code = locale.normalize(localename)
if code == 'UTF-8':
return None, code
else:
return _old_parse_localename(localename)
locale._parse_localename = _parse_localename
from docutils.core import publish_parts
from docutils.writers.html4css1 import HTMLTranslator
from docutils.nodes import SkipNode
# pylint: disable=unused-import
try:
from typing import Tuple
except ImportError:
pass
# pylint: enable=unused-import
[docs]def rst2html(rst, part="whole", math_output="mathjax"):
r"""
Convert restructured text into simple html.
Valid *math_output* formats for formulas include:
- html
- mathml
- mathjax
See `<http://docutils.sourceforge.net/docs/user/config.html#math-output>`_
for details.
The following *part* choices are available:
- whole: the entire html document
- html_body: document division with title and contents and footer
- body: contents only
There are other parts, but they don't make sense alone:
subtitle, version, encoding, html_prolog, header, meta,
html_title, title, stylesheet, html_subtitle, html_body,
body, head, body_suffix, fragment, docinfo, html_head,
head_prefix, body_prefix, footer, body_pre_docinfo, whole
"""
# Ick! mathjax doesn't work properly with math-output, and the
# others don't work properly with math_output!
if math_output == "mathjax":
# TODO: this is copied from docs/conf.py; there should be only one
mathjax_path = "https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.1/MathJax.js?config=TeX-MML-AM_CHTML"
settings = {"math_output": math_output + " " + mathjax_path}
else:
settings = {"math-output": math_output}
# TODO: support stylesheets
#html_root = "/full/path/to/_static/"
#sheets = [html_root+s for s in ["basic.css","classic.css"]]
#settings["embed_styesheet"] = True
#settings["stylesheet_path"] = sheets
# math2html and mathml do not support \frac12
rst = replace_compact_fraction(rst)
# mathml, html do not support \tfrac
if math_output in ("mathml", "html"):
rst = rst.replace(r'\tfrac', r'\frac')
rst = replace_dollar(rst)
with suppress_html_errors():
parts = publish_parts(source=rst, writer_name='html',
settings_overrides=settings)
return parts[part]
[docs]@contextmanager
def suppress_html_errors():
r"""
Context manager for keeping error reports out of the generated HTML.
Within the context, system message nodes in the docutils parse tree
will be ignored. After the context, the usual behaviour will be restored.
"""
visit_system_message = HTMLTranslator.visit_system_message
HTMLTranslator.visit_system_message = _skip_node
yield None
HTMLTranslator.visit_system_message = visit_system_message
[docs]def _skip_node(self, node):
raise SkipNode
_compact_fraction = re.compile(r"(\\[cdt]?frac)([0-9])([0-9])")
[docs]def replace_compact_fraction(content):
r"""
Convert \frac12 to \frac{1}{2} for broken latex parsers
"""
return _compact_fraction.sub(r"\1{\2}{\3}", content)
_dollar = re.compile(r"(?:^|(?<=\s|[-(]))[$]([^\n]*?)(?<![\\])[$](?:$|(?=\s|[-.,;:?\\)]))")
_notdollar = re.compile(r"\\[$]")
[docs]def replace_dollar(content):
r"""
Convert dollar signs to inline math markup in rst.
"""
content = _dollar.sub(r":math:`\1`", content)
content = _notdollar.sub("$", content)
return content
[docs]def test_dollar():
"""
Test substitution of dollar signs with equivalent RST math markup
"""
assert replace_dollar(u"no dollar") == u"no dollar"
assert replace_dollar(u"$only$") == u":math:`only`"
assert replace_dollar(u"$first$ is good") == u":math:`first` is good"
assert replace_dollar(u"so is $last$") == u"so is :math:`last`"
assert replace_dollar(u"and $mid$ too") == u"and :math:`mid` too"
assert replace_dollar(u"$first$, $mid$, $last$") == u":math:`first`, :math:`mid`, :math:`last`"
assert replace_dollar(u"dollar\\$ escape") == u"dollar$ escape"
assert replace_dollar(u"dollar \\$escape\\$ too") == u"dollar $escape$ too"
assert replace_dollar(u"spaces $in the$ math") == u"spaces :math:`in the` math"
assert replace_dollar(u"emb\\ $ed$\\ ed") == u"emb\\ :math:`ed`\\ ed"
assert replace_dollar(u"$first$a") == u"$first$a"
assert replace_dollar(u"a$last$") == u"a$last$"
assert replace_dollar(u"$37") == u"$37"
assert replace_dollar(u"($37)") == u"($37)"
assert replace_dollar(u"$37 - $43") == u"$37 - $43"
assert replace_dollar(u"($37, $38)") == u"($37, $38)"
assert replace_dollar(u"a $mid$dle a") == u"a $mid$dle a"
assert replace_dollar(u"a ($in parens$) a") == u"a (:math:`in parens`) a"
assert replace_dollar(u"a (again $in parens$) a") == u"a (again :math:`in parens`) a"
[docs]def load_rst_as_html(filename):
# type: (str) -> str
"""Load rst from file and convert to html"""
from os.path import expanduser
with open(expanduser(filename)) as fid:
rst = fid.read()
html = rst2html(rst)
return html
[docs]def wxview(html, url="", size=(850, 540)):
# type: (str, str, Tuple[int, int]) -> "wx.Frame"
"""View HTML in a wx dialog"""
import wx
from wx.html2 import WebView
frame = wx.Frame(None, -1, size=size)
view = WebView.New(frame)
view.SetPage(html, url)
frame.Show()
return frame
[docs]def view_html_wxapp(html, url=""):
# type: (str, str) -> None
"""HTML viewer app in wx"""
import wx # type: ignore
app = wx.App()
frame = wxview(html, url) # pylint: disable=unused-variable
app.MainLoop()
[docs]def view_url_wxapp(url):
# type: (str) -> None
"""URL viewer app in wx"""
import wx # type: ignore
from wx.html2 import WebView
app = wx.App()
frame = wx.Frame(None, -1, size=(850, 540))
view = WebView.New(frame)
view.LoadURL(url)
frame.Show()
app.MainLoop()
[docs]def qtview(html, url=""):
# type: (str, str) -> "QWebView"
"""View HTML in a Qt dialog"""
try:
from PyQt5.QtWebKitWidgets import QWebView
from PyQt5.QtCore import QUrl
except ImportError:
from PyQt4.QtWebKit import QWebView
from PyQt4.QtCore import QUrl
helpView = QWebView()
helpView.setHtml(html, QUrl(url))
helpView.show()
return helpView
[docs]def view_html_qtapp(html, url=""):
# type: (str, str) -> None
"""HTML viewer app in Qt"""
import sys
try:
from PyQt5.QtWidgets import QApplication
except ImportError:
from PyQt4.QtGui import QApplication
app = QApplication([])
frame = qtview(html, url) # pylint: disable=unused-variable
sys.exit(app.exec_())
[docs]def view_url_qtapp(url):
# type: (str) -> None
"""URL viewer app in Qt"""
import sys
try:
from PyQt5.QtWidgets import QApplication
except ImportError:
from PyQt4.QtGui import QApplication
app = QApplication([])
try:
from PyQt5.QtWebKitWidgets import QWebView
from PyQt5.QtCore import QUrl
except ImportError:
from PyQt4.QtWebKit import QWebView
from PyQt4.QtCore import QUrl
frame = QWebView()
frame.load(QUrl(url))
frame.show()
sys.exit(app.exec_())
# Set default html viewer
view_html = view_html_qtapp
[docs]def can_use_qt():
# type: () -> bool
"""
Return True if QWebView exists.
Checks first in PyQt5 then in PyQt4
"""
try:
from PyQt5.QtWebKitWidgets import QWebView # pylint: disable=unused-import
return True
except ImportError:
try:
from PyQt4.QtWebKit import QWebView # pylint: disable=unused-import
return True
except ImportError:
return False
[docs]def view_help(filename, qt=False):
# type: (str, bool) -> None
"""View rst or html file. If *qt* use q viewer, otherwise use wx."""
import os
if qt:
qt = can_use_qt()
url = "file:///"+os.path.abspath(filename).replace("\\", "/")
if filename.endswith('.rst'):
html = load_rst_as_html(filename)
if qt:
view_html_qtapp(html, url)
else:
view_html_wxapp(html, url)
else:
if qt:
view_url_qtapp(url)
else:
view_url_wxapp(url)
[docs]def main():
# type: () -> None
"""Command line interface to rst or html viewer."""
import sys
view_help(sys.argv[1], qt=False)
if __name__ == "__main__":
main()