# copyright 2003-2024 LOGILAB S.A. (Paris, FRANCE), all rights reserved.
# contact https://www.logilab.fr/ -- mailto:contact@logilab.fr
#
# This file is part of CubicWeb.
#
# CubicWeb is free software: you can redistribute it and/or modify it under the
# terms of the GNU Lesser General Public License as published by the Free
# Software Foundation, either version 2.1 of the License, or (at your option)
# any later version.
#
# CubicWeb is distributed in the hope that it will be useful, but WITHOUT
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
# FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
# details.
#
# You should have received a copy of the GNU Lesser General Public License along
# with CubicWeb. If not, see <https://www.gnu.org/licenses/>.
"""inline help system, rendering ReST files in the `wdoc` subdirectory of
CubicWeb and cubes
"""
from itertools import chain
from os.path import join
from xml.etree.ElementTree import parse
from logilab.common.registry import yes
from cubicweb import _
from cubicweb.predicates import match_form_params
from cubicweb.uilib import rest_publish
from cubicweb_web.view import StartupView
from cubicweb_web import NotFound, action
# table of content management #################################################
def build_toc_index(node, parent, index):
try:
nodeidx = node.attrib["resource"]
assert nodeidx not in index, nodeidx
index[nodeidx] = node, parent
except KeyError:
pass
for child in node:
build_toc_index(child, node, index)
def get_insertion_point(section, index):
if section.attrib.get("insertafter"):
snode, node = index[section.attrib["insertafter"]]
idx = list(node).index(snode) + 1
elif section.attrib.get("insertbefore"):
snode, node = index[section.attrib["insertbefore"]]
idx = node.getchildren().index(snode)
elif "appendto" in section.attrib:
node, _ = index[section.attrib["appendto"]]
idx = None
else:
node, idx = None, None
return node, idx
def build_toc(config):
alltocfiles = reversed(tuple(config.locate_all_files("toc.xml")))
maintoc = parse(next(alltocfiles)).getroot()
index = {}
build_toc_index(maintoc, None, index)
# insert component documentation into the tree according to their toc.xml
# file
for fpath in alltocfiles:
toc = parse(fpath).getroot()
for section in toc:
node, idx = get_insertion_point(section, index)
if node is None:
continue
if idx is None:
node.append(section)
else:
node.insert(idx, section)
build_toc_index(section, node, index)
return index
def title_for_lang(node, lang):
fallback_title = None
for title in node.findall("title"):
title_lang = title.attrib["{http://www.w3.org/XML/1998/namespace}lang"]
if title_lang == lang:
return title.text
if title_lang == "en":
fallback_title = title.text
return fallback_title
def subsections(node):
return [child for child in node if child.tag == "section"]
# help views ##################################################################
[docs]class InlineHelpView(StartupView):
__select__ = match_form_params("fid")
__regid__ = "wdoc"
title = _("site documentation")
def call(self):
fid = self._cw.form["fid"]
vreg = self._cw.vreg
for lang in chain(
(self._cw.lang, vreg.property_value("ui.language")),
vreg.config.available_languages(),
):
rid = f"{fid}_{lang}.rst"
resourcedir = vreg.config.locate_doc_file(rid)
if resourcedir:
break
else:
raise NotFound
self.tocindex = build_toc(vreg.config)
try:
node, parent = self.tocindex[fid]
except KeyError:
node, parent = None, None
else:
self.navigation_links(node, parent)
self.w('<div class="hr"></div>')
self.w("<h1>%s</h1>", title_for_lang(node, self._cw.lang))
with open(join(resourcedir, rid)) as f:
self.w(rest_publish(self, f.read()))
if node is not None:
self.subsections_links(node)
self.w('<div class="hr"></div>')
self.navigation_links(node, parent)
def navigation_links(self, node, parent):
if parent is None:
return
brothers = subsections(parent)
self.w('<div class="docnav">\n')
previousidx = brothers.index(node) - 1
if previousidx >= 0:
self.navsection(brothers[previousidx], "prev")
self.navsection(parent, "up")
nextidx = brothers.index(node) + 1
if nextidx < len(brothers):
self.navsection(brothers[nextidx], "next")
self.w("</div>\n")
navinfo = {
"prev": ("", "data/previous.png", _("i18nprevnext_previous")),
"next": ("", "data/next.png", _("i18nprevnext_next")),
"up": ("", "data/up.png", _("i18nprevnext_up")),
}
def navsection(self, node, navtype):
htmlclass, imgpath, msgid = self.navinfo[navtype]
self.w('<span class="%s">', htmlclass)
self.w("%s : ", self._cw._(msgid))
self.w(
'<a href="%s">%s</a>',
self._cw.build_url("doc/" + node.attrib["resource"]),
title_for_lang(node, self._cw.lang),
)
self.w("</span>\n")
def subsections_links(self, node, first=True):
sub = subsections(node)
if not sub:
return
if first:
self.w('<div class="hr"></div>')
self.w('<ul class="docsum">')
for child in sub:
self.w(
'<li><a href="%s">%s</a>',
self._cw.build_url("doc/" + child.attrib["resource"]),
title_for_lang(child, self._cw.lang),
)
self.subsections_links(child, False)
self.w("</li>")
self.w("</ul>\n")
class InlineHelpImageView(StartupView):
__regid__ = "wdocimages"
__select__ = match_form_params("fid")
binary = True
templatable = False
content_type = "image/png"
def call(self):
fid = self._cw.form["fid"]
for lang in chain(
(self._cw.lang, self._cw.vreg.property_value("ui.language")),
self._cw.vreg.config.available_languages(),
):
rid = join("images", f"{fid}_{lang}.png")
resourcedir = self._cw.vreg.config.locate_doc_file(rid)
if resourcedir:
break
else:
raise NotFound
with open(join(resourcedir, rid)) as f:
self.w(f.read())
[docs]class HelpAction(action.Action):
__regid__ = "help"
__select__ = yes()
category = "footer"
order = 0
title = _("Help")
def url(self):
return self._cw.build_url("doc/main")
[docs]class AboutAction(action.Action):
__regid__ = "about"
__select__ = yes()
category = "footer"
order = 2
title = _("About this site")
def url(self):
return self._cw.build_url("doc/about")