Source code for reactpy.core.vdom

from __future__ import annotations

import json
from collections.abc import Mapping, Sequence
from functools import wraps
from typing import Any, Protocol, cast, overload

from fastjsonschema import compile as compile_json_schema

from reactpy._warnings import warn
from reactpy.config import REACTPY_CHECK_JSON_ATTRS, REACTPY_DEBUG_MODE
from reactpy.core._f_back import f_module_name
from reactpy.core.events import EventHandler, to_event_handler_function
from reactpy.core.types import (
    ComponentType,
    EventHandlerDict,
    EventHandlerType,
    ImportSourceDict,
    Key,
    VdomAttributes,
    VdomChild,
    VdomChildren,
    VdomDict,
    VdomDictConstructor,
    VdomJson,
)

VDOM_JSON_SCHEMA = {
    "$schema": "http://json-schema.org/draft-07/schema",
    "$ref": "#/definitions/element",
    "definitions": {
        "element": {
            "type": "object",
            "properties": {
                "tagName": {"type": "string"},
                "key": {"type": ["string", "number", "null"]},
                "error": {"type": "string"},
                "children": {"$ref": "#/definitions/elementChildren"},
                "attributes": {"type": "object"},
                "eventHandlers": {"$ref": "#/definitions/elementEventHandlers"},
                "importSource": {"$ref": "#/definitions/importSource"},
            },
            # The 'tagName' is required because its presence is a useful indicator of
            # whether a dictionary describes a VDOM model or not.
            "required": ["tagName"],
            "dependentSchemas": {
                # When 'error' is given, the 'tagName' should be empty.
                "error": {"properties": {"tagName": {"maxLength": 0}}}
            },
        },
        "elementChildren": {
            "type": "array",
            "items": {"$ref": "#/definitions/elementOrString"},
        },
        "elementEventHandlers": {
            "type": "object",
            "patternProperties": {
                ".*": {"$ref": "#/definitions/eventHandler"},
            },
        },
        "eventHandler": {
            "type": "object",
            "properties": {
                "target": {"type": "string"},
                "preventDefault": {"type": "boolean"},
                "stopPropagation": {"type": "boolean"},
            },
            "required": ["target"],
        },
        "importSource": {
            "type": "object",
            "properties": {
                "source": {"type": "string"},
                "sourceType": {"enum": ["URL", "NAME"]},
                "fallback": {
                    "type": ["object", "string", "null"],
                    "if": {"not": {"type": "null"}},
                    "then": {"$ref": "#/definitions/elementOrString"},
                },
                "unmountBeforeUpdate": {"type": "boolean"},
            },
            "required": ["source"],
        },
        "elementOrString": {
            "type": ["object", "string"],
            "if": {"type": "object"},
            "then": {"$ref": "#/definitions/element"},
        },
    },
}
"""JSON Schema describing serialized VDOM - see :ref:`VDOM` for more info"""


# we can't add a docstring to this because Sphinx doesn't know how to find its source
_COMPILED_VDOM_VALIDATOR = compile_json_schema(VDOM_JSON_SCHEMA)


[docs]def validate_vdom_json(value: Any) -> VdomJson: """Validate serialized VDOM - see :attr:`VDOM_JSON_SCHEMA` for more info""" _COMPILED_VDOM_VALIDATOR(value) return cast(VdomJson, value)
[docs]def is_vdom(value: Any) -> bool: """Return whether a value is a :class:`VdomDict` This employs a very simple heuristic - something is VDOM if: 1. It is a ``dict`` instance 2. It contains the key ``"tagName"`` 3. The value of the key ``"tagName"`` is a string .. note:: Performing an ``isinstance(value, VdomDict)`` check is too restrictive since the user would be forced to import ``VdomDict`` every time they needed to declare a VDOM element. Giving the user more flexibility, at the cost of this check's accuracy, is worth it. """ return ( isinstance(value, dict) and "tagName" in value and isinstance(value["tagName"], str) )
@overload def vdom(tag: str, *children: VdomChildren) -> VdomDict: ... @overload def vdom(tag: str, attributes: VdomAttributes, *children: VdomChildren) -> VdomDict: ...
[docs]def vdom( tag: str, *attributes_and_children: Any, **kwargs: Any, ) -> VdomDict: """A helper function for creating VDOM elements. Parameters: tag: The type of element (e.g. 'div', 'h1', 'img') attributes_and_children: An optional attribute mapping followed by any number of children or iterables of children. The attribute mapping **must** precede the children, or children which will be merged into their respective parts of the model. key: A string indicating the identity of a particular element. This is significant to preserve event handlers across updates - without a key, a re-render would cause these handlers to be deleted, but with a key, they would be redirected to any newly defined handlers. event_handlers: Maps event types to coroutines that are responsible for handling those events. import_source: (subject to change) specifies javascript that, when evaluated returns a React component. """ if kwargs: # nocov if "key" in kwargs: if attributes_and_children: maybe_attributes, *children = attributes_and_children if _is_attributes(maybe_attributes): attributes_and_children = ( {**maybe_attributes, "key": kwargs.pop("key")}, *children, ) else: attributes_and_children = ( {"key": kwargs.pop("key")}, maybe_attributes, *children, ) else: attributes_and_children = ({"key": kwargs.pop("key")},) warn( "An element's 'key' must be declared in an attribute dict instead " "of as a keyword argument. This will error in a future version.", DeprecationWarning, ) if kwargs: msg = f"Extra keyword arguments {kwargs}" raise ValueError(msg) model: VdomDict = {"tagName": tag} if not attributes_and_children: return model attributes, children = separate_attributes_and_children(attributes_and_children) key = attributes.pop("key", None) attributes, event_handlers = separate_attributes_and_event_handlers(attributes) if attributes: if REACTPY_CHECK_JSON_ATTRS.current: json.dumps(attributes) model["attributes"] = attributes if children: model["children"] = children if key is not None: model["key"] = key if event_handlers: model["eventHandlers"] = event_handlers return model
[docs]def make_vdom_constructor( tag: str, allow_children: bool = True, import_source: ImportSourceDict | None = None ) -> VdomDictConstructor: """Return a constructor for VDOM dictionaries with the given tag name. The resulting callable will have the same interface as :func:`vdom` but without its first ``tag`` argument. """ def constructor(*attributes_and_children: Any, **kwargs: Any) -> VdomDict: model = vdom(tag, *attributes_and_children, **kwargs) if not allow_children and "children" in model: msg = f"{tag!r} nodes cannot have children." raise TypeError(msg) if import_source: model["importSource"] = import_source return model # replicate common function attributes constructor.__name__ = tag constructor.__doc__ = ( "Return a new " f"`<{tag}> <https://developer.mozilla.org/en-US/docs/Web/HTML/Element/{tag}>`__ " "element represented by a :class:`VdomDict`." ) module_name = f_module_name(1) if module_name: constructor.__module__ = module_name constructor.__qualname__ = f"{module_name}.{tag}" return cast(VdomDictConstructor, constructor)
[docs]def custom_vdom_constructor(func: _CustomVdomDictConstructor) -> VdomDictConstructor: """Cast function to VdomDictConstructor""" @wraps(func) def wrapper(*attributes_and_children: Any) -> VdomDict: attributes, children = separate_attributes_and_children(attributes_and_children) key = attributes.pop("key", None) attributes, event_handlers = separate_attributes_and_event_handlers(attributes) return func(attributes, children, key, event_handlers) return cast(VdomDictConstructor, wrapper)
def separate_attributes_and_children( values: Sequence[Any], ) -> tuple[dict[str, Any], list[Any]]: if not values: return {}, [] attributes: dict[str, Any] children_or_iterables: Sequence[Any] if _is_attributes(values[0]): attributes, *children_or_iterables = values else: attributes = {} children_or_iterables = values children: list[Any] = [] for child in children_or_iterables: if _is_single_child(child): children.append(child) else: children.extend(child) return attributes, children def separate_attributes_and_event_handlers( attributes: Mapping[str, Any], ) -> tuple[dict[str, Any], EventHandlerDict]: separated_attributes = {} separated_event_handlers: dict[str, EventHandlerType] = {} for k, v in attributes.items(): handler: EventHandlerType if callable(v): handler = EventHandler(to_event_handler_function(v)) elif ( # isinstance check on protocols is slow - use function attr pre-check as a # quick filter before actually performing slow EventHandlerType type check hasattr(v, "function") and isinstance(v, EventHandlerType) ): handler = v else: separated_attributes[k] = v continue separated_event_handlers[k] = handler return separated_attributes, dict(separated_event_handlers.items()) def _is_attributes(value: Any) -> bool: return isinstance(value, Mapping) and "tagName" not in value def _is_single_child(value: Any) -> bool: if isinstance(value, (str, Mapping)) or not hasattr(value, "__iter__"): return True if REACTPY_DEBUG_MODE.current: _validate_child_key_integrity(value) return False def _validate_child_key_integrity(value: Any) -> None: if hasattr(value, "__iter__") and not hasattr(value, "__len__"): warn( f"Did not verify key-path integrity of children in generator {value} " "- pass a sequence (i.e. list of finite length) in order to verify" ) else: for child in value: if isinstance(child, ComponentType) and child.key is None: warn(f"Key not specified for child in list {child}", UserWarning) elif isinstance(child, Mapping) and "key" not in child: # remove 'children' to reduce log spam child_copy = {**child, "children": _EllipsisRepr()} warn(f"Key not specified for child in list {child_copy}", UserWarning) class _CustomVdomDictConstructor(Protocol): def __call__( self, attributes: VdomAttributes, children: Sequence[VdomChild], key: Key | None, event_handlers: EventHandlerDict, ) -> VdomDict: ... class _EllipsisRepr: def __repr__(self) -> str: return "..."