Skip to content

Commit a70f2b6

Browse files
committed
Support glossary / term and sphinxcontrib-bibtex
1 parent cf1c8b2 commit a70f2b6

14 files changed

+196
-5
lines changed

docs/conf.py

+5
Original file line numberDiff line numberDiff line change
@@ -50,8 +50,11 @@
5050
'hoverxref.extension',
5151
'versionwarning.extension',
5252
'notfound.extension',
53+
'sphinxcontrib.bibtex',
5354
]
5455

56+
bibtex_bibfiles = ['refs.bib']
57+
5558
intersphinx_mapping = {
5659
'readthedocs': ('https://docs.readthedocs.io/en/stable/', None),
5760
'sphinx': ('https://www.sphinx-doc.org/en/master/', None),
@@ -95,6 +98,7 @@
9598
hoverxref_auto_ref = True
9699
hoverxref_roles = [
97100
'confval',
101+
'term',
98102
]
99103

100104
hoverxref_role_types = {
@@ -107,6 +111,7 @@
107111
}
108112
hoverxref_domains = [
109113
'py',
114+
'cite',
110115
]
111116
hoverxref_sphinxtabs = True
112117
hoverxref_mathjax = True

docs/configuration.rst

-4
Original file line numberDiff line numberDiff line change
@@ -65,10 +65,6 @@ These settings are global and have effect on both, tooltips and modal dialogues.
6565

6666
Description: List containing the Sphinx Domain's names where ``hoverxref`` has to be applied.
6767

68-
.. warning::
69-
70-
Only Python Domain (``py``) is currently supported.
71-
7268
Default: ``[]``
7369

7470
Type: list

docs/refs.bib

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
@Book{1987:nelson,
2+
author = {Edward Nelson},
3+
title = {Radically Elementary Probability Theory},
4+
publisher = {Princeton University Press},
5+
year = {1987}
6+
}

docs/requirements.txt

+1
Original file line numberDiff line numberDiff line change
@@ -6,3 +6,4 @@ sphinx-prompt==1.4.0
66
sphinx-version-warning==1.1.2
77
sphinx-notfound-page==0.7.1
88
sphinx-autobuild==2021.3.14
9+
sphinxcontrib-bibtex==2.4.1

docs/usage.rst

+34
Original file line numberDiff line numberDiff line change
@@ -113,6 +113,40 @@ To enable ``hoverxref`` on a domain, you need to use the config :confval:`hoverx
113113
indicating which are the domains you desire.
114114

115115

116+
Tooltip on glossary terms
117+
-------------------------
118+
119+
You can add tooltips to glossary terms:
120+
121+
.. code-block:: rst
122+
123+
See the :term:`sphinx:environment` definition in the glossary.
124+
125+
That will render to:
126+
127+
See the :term:`sphinx:environment` definition in the glossary.
128+
129+
To enable ``hoverxref`` on glossary terms, you need to add ``'term'`` to :confval:`hoverxref_roles`.
130+
131+
132+
Tooltip on sphinxcontrib-bibtex cites
133+
-------------------------------------
134+
135+
If you want to show a tooltip on `sphinxcontrib-bibtex <https://sphinxcontrib-bibtex.readthedocs.io/en/latest/>`_ cites,
136+
you just need to enable it in :confval:`hoverxref_domains` by adding ``'cite'`` to that list.
137+
Example:
138+
139+
.. code-block:: rst
140+
141+
See :cite:t:`1987:nelson` for an introduction to non-standard analysis.
142+
Non-standard analysis is fun :cite:p:`1987:nelson`.
143+
144+
See :cite:t:`1987:nelson` for an introduction to non-standard analysis.
145+
Non-standard analysis is fun :cite:p:`1987:nelson`.
146+
147+
.. bibliography::
148+
149+
116150
Tooltip with content that needs extra rendering steps
117151
-----------------------------------------------------
118152

hoverxref/domains.py

+37
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,5 @@
1+
import docutils
2+
13
from sphinx.util import logging
24

35
logger = logging.getLogger(__name__)
@@ -138,3 +140,38 @@ def _resolve_numref_xref(self, env, fromdocname, builder, typ, target, node, con
138140

139141
self._inject_hoverxref_data(env, refnode, typ)
140142
return refnode
143+
144+
145+
class HoverXRefBibtexDomainMixin(HoverXRefBaseDomain):
146+
"""
147+
Mixin for ``BibtexDomain`` to save the values after the xref resolution.
148+
149+
This class add the required ``hoverxref`` and ``modal``/``tooltip`` to tell
150+
the frontend to show a modal/tooltip on this element.
151+
152+
https://github.com/mcmtroffaes/sphinxcontrib-bibtex/blob/2.4.1/src/sphinxcontrib/bibtex/domain.py#L281
153+
"""
154+
155+
def resolve_xref(self, env, fromdocname, builder, typ, target, node, contnode):
156+
textnode = super().resolve_xref(env, fromdocname, builder, typ, target, node, contnode)
157+
if textnode is None:
158+
return textnode
159+
160+
if any([
161+
self._is_ignored_ref(env, target),
162+
not (env.config.hoverxref_auto_ref or typ in self.hoverxref_types or 'cite' in env.config.hoverxref_domains)
163+
]):
164+
return textnode
165+
166+
# The structure of the node generated by bibtex is between two
167+
# ``#text`` nodes and we need to add the classes into the ``reference``
168+
# node to get the ``href=`` attribute from it
169+
#
170+
# (Pdb++) textnode.children
171+
# [<#text: 'Nelson ['>, <reference: <#text: 'Nel87'>>, <#text: ']'>]
172+
refnode_index = textnode.first_child_matching_class(docutils.nodes.reference)
173+
if refnode_index:
174+
refnode = textnode.children[refnode_index]
175+
self._inject_hoverxref_data(env, refnode, typ)
176+
177+
return textnode

hoverxref/extension.py

+12
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@
1212
from . import version
1313
from .domains import (
1414
HoverXRefBaseDomain,
15+
HoverXRefBibtexDomainMixin,
1516
HoverXRefPythonDomainMixin,
1617
HoverXRefStandardDomainMixin,
1718
)
@@ -119,6 +120,17 @@ def setup_domains(app, config):
119120
)
120121
app.add_domain(domain, override=True)
121122

123+
if 'cite' in app.config.hoverxref_domains:
124+
domain = types.new_class(
125+
'HoverXRefBibtexDomain',
126+
(
127+
HoverXRefBibtexDomainMixin,
128+
app.registry.domains.get('cite'),
129+
),
130+
{}
131+
)
132+
app.add_domain(domain, override=True)
133+
122134

123135
def setup_sphinx_tabs(app, config):
124136
"""

tests/examples/bibtex-domain/conf.py

+11
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
# conf.py to run tests
2+
3+
master_doc = 'index'
4+
extensions = [
5+
'sphinx.ext.autodoc',
6+
'sphinx.ext.autosectionlabel',
7+
'hoverxref.extension',
8+
'sphinxcontrib.bibtex',
9+
]
10+
11+
bibtex_bibfiles = ['refs.bib']
+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
sphinxcontrib-bibtex Domain
2+
===========================
3+
4+
This is an example page with a sphinxcontrib-bibtex Domain role usage.
5+
6+
See :cite:t:`1987:nelson` for an introduction to non-standard analysis.
7+
Non-standard analysis is fun :cite:p:`1987:nelson`.
8+
9+
10+
.. bibliography::

tests/examples/bibtex-domain/refs.bib

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
@Book{1987:nelson,
2+
author = {Edward Nelson},
3+
title = {Radically Elementary Probability Theory},
4+
publisher = {Princeton University Press},
5+
year = {1987}
6+
}

tests/examples/default/glossary.rst

+25
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,25 @@
1+
Glossary
2+
========
3+
4+
Example page showing the usage of ``.. glossary`` and ``term``.
5+
6+
See definition :term:`builder` for more information.
7+
8+
.. copied from https://www.sphinx-doc.org/en/master/glossary.html
9+
10+
.. glossary::
11+
12+
builder
13+
A class (inheriting from :class:`~sphinx.builders.Builder`) that takes
14+
parsed documents and performs an action on them. Normally, builders
15+
translate the documents to an output format, but it is also possible to
16+
use builders that e.g. check for broken links in the documentation, or
17+
build coverage information.
18+
19+
See :doc:`/usage/builders/index` for an overview over Sphinx's built-in
20+
builders.
21+
22+
configuration directory
23+
The directory containing :file:`conf.py`. By default, this is the same as
24+
the :term:`source directory`, but can be set differently with the **-c**
25+
command-line option.

tests/test_htmltag.py

+41-1
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@
33
import sphinx
44
import textwrap
55

6-
from .utils import srcdir, prefixdocumentsrcdir, customobjectsrcdir, pythondomainsrcdir, intersphinxsrc
6+
from .utils import srcdir, prefixdocumentsrcdir, customobjectsrcdir, pythondomainsrcdir, intersphinxsrc, bibtexdomainsrcdir
77

88

99
@pytest.mark.sphinx(
@@ -121,6 +121,46 @@ def test_python_domain(app, status, warning):
121121
assert chunk in content
122122

123123

124+
@pytest.mark.sphinx(
125+
srcdir=bibtexdomainsrcdir,
126+
confoverrides={
127+
'hoverxref_domains': ['cite'],
128+
},
129+
)
130+
def test_bibtex_domain(app, status, warning):
131+
app.build()
132+
path = app.outdir / 'index.html'
133+
assert path.exists() is True
134+
content = open(path).read()
135+
136+
chunks = [
137+
'<p>See <span id="id1">Nelson [<a class="hoverxref tooltip reference internal" href="#id4" title="Edward Nelson. Radically Elementary Probability Theory. Princeton University Press, 1987.">Nel87</a>]</span> for an introduction to non-standard analysis.\nNon-standard analysis is fun <span id="id2">[<a class="hoverxref tooltip reference internal" href="#id4" title="Edward Nelson. Radically Elementary Probability Theory. Princeton University Press, 1987.">Nel87</a>]</span>.</p>',
138+
]
139+
140+
for chunk in chunks:
141+
assert chunk in content
142+
143+
144+
@pytest.mark.sphinx(
145+
srcdir=srcdir,
146+
confoverrides={
147+
'hoverxref_roles': ['term'],
148+
},
149+
)
150+
def test_glossary_term_domain(app, status, warning):
151+
app.build()
152+
path = app.outdir / 'glossary.html'
153+
assert path.exists() is True
154+
content = open(path).read()
155+
156+
chunks = [
157+
'<p>See definition <a class="hoverxref tooltip reference internal" href="#term-builder"><span class="xref std std-term">builder</span></a> for more information.</p>',
158+
]
159+
160+
for chunk in chunks:
161+
assert chunk in content
162+
163+
124164
@pytest.mark.sphinx(
125165
srcdir=srcdir,
126166
confoverrides={

tests/utils.py

+7
Original file line numberDiff line numberDiff line change
@@ -28,6 +28,13 @@
2828
'python-domain',
2929
)
3030

31+
# srcdir with ``:cite:`` call
32+
bibtexdomainsrcdir = os.path.join(
33+
os.path.dirname(os.path.abspath(__file__)),
34+
'examples',
35+
'bibtex-domain',
36+
)
37+
3138
# srcdir with intersphinx configured
3239
intersphinxsrc = os.path.join(
3340
os.path.dirname(os.path.abspath(__file__)),

tox.ini

+1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ envlist =
77
deps =
88
pytest
99
pdbpp
10+
sphinxcontrib-bibtex
1011
.
1112
sphinx18: sphinx~=1.8.0
1213
sphinx20: sphinx~=2.0.0

0 commit comments

Comments
 (0)