Source code for ipypublish.sphinx.gls.transforms

import docutils.nodes
import docutils.transforms
import sphinx.util
from sphinx import addnodes
from typing import Any  # noqa: F401

from ipypublish.sphinx.gls.bibgloss import format_entries, docutils_citation_node
from ipypublish.sphinx.gls.nodes import BibGlossaryNode

logger = sphinx.util.logging.getLogger(__name__)


[docs]class OverrideCitationReferences(docutils.transforms.Transform): """ Replace citation references by pending_xref nodes before the default docutils transform tries to resolve them. overrides sphinx.transforms.CitationReferences, in order to propogate the classes of the citation_reference node to the pending_xref node """ # the default priority of sphinx.transforms.CitationReferences is 619 default_priority = 618
[docs] def apply(self, **kwargs): # type: (Any) -> None # mark citation labels as not smartquoted # for citation in self.document.traverse(nodes.citation): # label = cast(nodes.label, citation[0]) # label['support_smartquotes'] = False for citation_ref in self.document.traverse(docutils.nodes.citation_reference): cittext = citation_ref.astext() refnode = addnodes.pending_xref( cittext, refdomain="std", reftype="citation", reftarget=cittext, refwarn=True, support_smartquotes=False, ids=citation_ref["ids"], ) refnode.source = citation_ref.source or citation_ref.parent.source refnode.line = citation_ref.line or citation_ref.parent.line refnode += docutils.nodes.Text("[" + cittext + "]") for class_name in citation_ref.attributes.get("classes", []): refnode["classes"].append(class_name) citation_ref.parent.replace(citation_ref, refnode)
[docs]class HandleMissingCitesTransform(docutils.transforms.Transform): """ before sphinx.transforms.post_transforms.ReferencesResolver missing citations need to be handled (default_priority=10) """ default_priority = 9
[docs] def apply(self): # type: () -> None app = self.document.settings.env.app for node in self.document.traverse(addnodes.pending_xref): if "bibglossary" not in node.attributes.get("classes", []): continue text = node[0].astext() key = text[1:-1] try: app.env.bibgloss_cache.get_label_from_key(key) except KeyError: logger.warning( "could not relabel bibglossary reference [%s]" % key, type="bibgloss", subtype="relabel", ) # strip class (otherwise ReferencesResolver fails) node.attributes["classes"] = []
[docs]def node_text_transform(node, transform): """Apply transformation to all Text nodes within node.""" for child in node.children: if isinstance(child, docutils.nodes.Text): node.replace(child, transform(child)) else: node_text_transform(child, transform)
[docs]def transform_url_command(textnode): """Convert '\\\\url{...}' into a proper docutils hyperlink.""" text = textnode.astext() if "\\url" in text: text1, _, text = text.partition("\\url") text2, _, text3 = text.partition("}") text2 = text2.lstrip(" {") ref = docutils.nodes.reference(refuri=text2) ref += docutils.nodes.Text(text2) node = docutils.nodes.inline() node += transform_url_command(docutils.nodes.Text(text1)) node += ref node += transform_url_command(docutils.nodes.Text(text3)) return node else: return textnode
[docs]class BibGlossaryTransform(docutils.transforms.Transform): """A docutils transform to generate citation entries for bibglossary nodes. """ # transform must be applied before references are resolved default_priority = 10 """Priority of the transform. See http://docutils.sourceforge.net/docs/ref/transforms.html """
[docs] def apply(self): """Transform each :class:`~ipypublish.sphinx.gls.nodes.BibGlossaryNode` node into a list of glossary terms. """ env = self.document.settings.env docname = env.docname for bibnode in self.document.traverse(BibGlossaryNode): id_ = bibnode["ids"][0] bibcache = env.bibgloss_cache.get_bibliography_cache( docname=docname, id_=id_ ) entries = env.bibgloss_cache.get_bibliography_entries( docname=docname, id_=id_, warn=logger.warning ) # create citation nodes for all references nodes = docutils.nodes.paragraph() for styled_entry in format_entries( entries, style=bibcache.style, sort=not bibcache.unsorted ): citation = docutils_citation_node( styled_entry, self.document, convert_latex=env.app.config.bibgloss_convert_latex, ) # docutils_citation_node(...) uses entry.key # as citation label # we change it to entry.label later onwards # but we must note the entry.label now; # at this point, we also already prefix the label key = citation[0].astext() bibcache.labels[key] = styled_entry.label bibcache.plurals[key] = styled_entry.plural node_text_transform(citation, transform_url_command) nodes += citation bibnode.replace_self(nodes)