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.