Files
edx-platform/xmodule/capa/customrender.py
2026-01-07 16:39:11 +05:00

196 lines
6.0 KiB
Python

"""
This has custom renderers: classes that know how to render certain problem tags (e.g. <math> and
<solution>) to html.
These tags do not have state, so they just get passed the system (for access to render_template),
and the xml element.
"""
import logging
import re
from xml.sax import saxutils
from django.utils import html as html_escape
from lxml import etree
from .registry import TagRegistry
log = logging.getLogger(__name__)
registry = TagRegistry()
# -----------------------------------------------------------------------------
class MathRenderer: # pylint: disable=too-few-public-methods
"""
Renders <math> tags into MathJax-compatible HTML for displaying math expressions.
"""
tags = ["math"]
def __init__(self, system, xml):
r"""
Render math using latex-like formatting.
Examples:
<math>$\displaystyle U(r)=4 U_0 $</math>
<math>$r_0$</math>
We convert these to [mathjax]...[/mathjax] and [mathjaxinline]...[/mathjaxinline]
TODO: use shorter tags (but this will require converting problem XML files!)
"""
self.system = system
self.xml = xml
mathstr = re.sub(r"\$(.*)\$", r"[mathjaxinline]\1[/mathjaxinline]", xml.text)
mtag = "mathjax"
if r"\displaystyle" not in mathstr:
mtag += "inline"
else:
mathstr = mathstr.replace(r"\displaystyle", "")
self.mathstr = mathstr.replace("mathjaxinline]", f"{mtag}]")
def get_html(self):
"""
Return the contents of this tag, rendered to html, as an etree element.
"""
# TODO: why are there nested html tags here?? Why are there html tags at all, in fact?
html = f"<html><html>{self.mathstr}</html><html>{saxutils.escape(self.xml.tail or '')}</html></html>"
try:
xhtml = etree.XML(html)
except Exception as err: # pylint: disable=broad-exception-caught
if self.system.DEBUG:
msg = (
f"<html><div class='inline-error'>"
f"<p>Error {html_escape.escape(err)}</p>"
f"<p>Failed to construct math expression from <pre>{html_escape.escape(html)}</pre></p>"
f"</div></html>"
)
log.error(msg)
return etree.XML(msg)
raise
return xhtml
registry.register(MathRenderer)
# -----------------------------------------------------------------------------
class SolutionRenderer: # pylint: disable=too-few-public-methods
"""
A solution is just a <span>...</span> which is given an ID, that is used for displaying an
extended answer (a problem "solution") after "show answers" is pressed.
Note that the solution content is NOT rendered and returned in the HTML. It is obtained by an
ajax call.
"""
tags = ["solution"]
def __init__(self, system, xml):
self.system = system
self.id = xml.get("id")
def get_html(self):
"""Return the solution HTML rendered as an etree element."""
context = {"id": self.id}
html = self.system.render_template("solutionspan.html", context)
return etree.XML(html)
registry.register(SolutionRenderer)
# -----------------------------------------------------------------------------
class TargetedFeedbackRenderer: # pylint: disable=too-few-public-methods
"""
A targeted feedback is just a <span>...</span> that is used for displaying an
extended piece of feedback to students if they incorrectly answered a question.
"""
tags = ["targetedfeedback"]
def __init__(self, system, xml):
self.system = system
self.xml = xml
def get_html(self):
"""
Return the contents of this tag, rendered to html, as an etree element.
"""
# xss-lint: disable=python-wrap-html
html_str = (
f'<section class="targeted-feedback-span">'
f'<span>{etree.tostring(self.xml, encoding="unicode")}</span>'
f"</section>"
)
try:
xhtml = etree.XML(html_str)
except Exception as err: # pylint: disable=broad-exception-caught
if self.system.DEBUG:
# xss-lint: disable=python-wrap-html
msg = f"""
<html>
<div class="inline-error">
<p>Error {html_escape.escape(err)}</p>
<p>Failed to construct targeted feedback from <pre>{html_escape.escape(html_str)}</pre></p>
</div>
</html>
"""
log.error(msg)
return etree.XML(msg)
raise
return xhtml
registry.register(TargetedFeedbackRenderer)
# -----------------------------------------------------------------------------
class ClarificationRenderer: # pylint: disable=too-few-public-methods
"""
A clarification appears as an inline icon which reveals more information when the user
hovers over it.
e.g. <p>Enter the ROA <clarification>Return on Assets</clarification> for 2015:</p>
"""
tags = ["clarification"]
def __init__(self, system, xml):
self.system = system
# Get any text content found inside this tag prior to the first child tag. It may be a string or None type.
initial_text = xml.text if xml.text else ""
self.inner_html = initial_text + "".join(etree.tostring(element, encoding="unicode") for element in xml)
self.tail = xml.tail
def get_html(self):
"""
Return the contents of this tag, rendered to html, as an etree element.
"""
context = {"clarification": self.inner_html}
html = self.system.render_template("clarification.html", context)
xml = etree.XML(html)
# We must include any text that was following our original <clarification>...</clarification> XML node.:
xml.tail = self.tail
return xml
registry.register(ClarificationRenderer)