core.widgets ============ .. py:module:: core.widgets .. autoapi-nested-parse:: Provides widgets. Widgets are small rendered pieces such as news messages or upcoming events. They are defined with XLS and can be combinded with simple XML. A widget is simply a class with a tag and template attribute and optionally a get_variables method:: class MyWidget: tag = 'my-widget' template = ( '' '
' ' ' '
' '
' ) def get_variables(self, layout): return {'foo': 'bar'} XML using the widget tags can then be transformed to chameleon HTML/XML. The widget variables need to be injected before rendering:: from onegov.core.templates import PageTemplate widgets = [MyWidget()] structure = 'Hello world!' template = PageTemplate(transform_structure(widgets, structure)) layout = None template.render(**inject_variables(widgets, layout, structure)) Attributes ---------- .. autoapisummary:: core.widgets.XSLT_BASE core.widgets.XML_BASE core.widgets.XML_LINE_OFFSET Classes ------- .. autoapisummary:: core.widgets.Widget Functions --------- .. autoapisummary:: core.widgets.parse_structure core.widgets.transform_structure core.widgets.inject_variables Module Contents --------------- .. py:class:: Widget Bases: :py:obj:`Protocol` Base class for protocol classes. Protocol classes are defined as:: class Proto(Protocol): def meth(self) -> int: ... Such classes are primarily used with static type checkers that recognize structural subtyping (static duck-typing). For example:: class C: def meth(self) -> int: return 0 def func(x: Proto) -> int: return x.meth() func(C()) # Passes static type check See PEP 544 for details. Protocol classes decorated with @typing.runtime_checkable act as simple-minded runtime protocols that check only the presence of given attributes, ignoring their type signatures. Protocol classes can be generic, they are defined as:: class GenProto(Protocol[T]): def meth(self) -> T: ... .. py:property:: tag :type: str .. py:property:: template :type: str .. py:data:: XSLT_BASE :value: Multiline-String .. raw:: html
Show Value .. code-block:: python """
{}
""" .. raw:: html
.. py:data:: XML_BASE :value: Multiline-String .. raw:: html
Show Value .. code-block:: python """ {} """ .. raw:: html
.. py:data:: XML_LINE_OFFSET :value: 6 .. py:function:: parse_structure(widgets: collections.abc.Collection[Widget], structure: str) -> lxml.etree._Element Takes the XML structure and returns the parsed etree xml. Raises a wtforms.ValidationError if there's an element which is not supported. We could do this with DTDs but we don't actually need to, since we only care to not include any unknown tags. Note that we *try* to make sure that this is no vector for remote code execution, but we ultimately do not guarantee it can't happen. This xml structure is meant to be static or possibly changeable by admins. Ordinary users should not be able to influence this structure. .. py:function:: transform_structure(widgets: collections.abc.Collection[Widget], structure: str) -> str Takes the XML structure and transforms it into a chameleon template. The app is required as it contains the available widgets. .. py:function:: inject_variables(widgets: collections.abc.Collection[Widget], layout: core.layout.Layout, structure: Literal[''] | None, variables: None = None, unique_variable_names: bool = True) -> None inject_variables(widgets: collections.abc.Collection[Widget], layout: core.layout.Layout, structure: Literal[''] | None, variables: core.types.RenderData, unique_variable_names: bool = True) -> core.types.RenderData inject_variables(widgets: collections.abc.Collection[Widget], layout: core.layout.Layout, structure: str, variables: None = None, unique_variable_names: bool = True) -> core.types.RenderData | None inject_variables(widgets: collections.abc.Collection[Widget], layout: core.layout.Layout, structure: str | None, variables: core.types.RenderData, unique_variable_names: bool = True) -> core.types.RenderData Takes the widgets, layout, structure and a dict of variables meant for the template engine and injects the variables required by the widgets, if the widgets are indeed in use.