Source code for reactpy.core.types

from __future__ import annotations

import sys
from collections import namedtuple
from collections.abc import Mapping, Sequence
from types import TracebackType
from typing import (
    TYPE_CHECKING,
    Any,
    Callable,
    Generic,
    Literal,
    NamedTuple,
    Protocol,
    TypeVar,
    overload,
    runtime_checkable,
)

from typing_extensions import TypeAlias, TypedDict

_Type = TypeVar("_Type")


if TYPE_CHECKING or sys.version_info < (3, 9) or sys.version_info >= (3, 11):

[docs] class State(NamedTuple, Generic[_Type]): value: _Type set_value: Callable[[_Type | Callable[[_Type], _Type]], None]
else: # nocov State = namedtuple("State", ("value", "set_value")) ComponentConstructor = Callable[..., "ComponentType"] """Simple function returning a new component""" RootComponentConstructor = Callable[[], "ComponentType"] """The root component should be constructed by a function accepting no arguments.""" Key: TypeAlias = "str | int" _OwnType = TypeVar("_OwnType")
[docs]@runtime_checkable class ComponentType(Protocol): """The expected interface for all component-like objects""" key: Key | None """An identifier which is unique amongst a component's immediate siblings""" type: Any """The function or class defining the behavior of this component This is used to see if two component instances share the same definition. """
[docs] def render(self) -> VdomDict | ComponentType | str | None: """Render the component's view model."""
_Render_co = TypeVar("_Render_co", covariant=True) _Event_contra = TypeVar("_Event_contra", contravariant=True)
[docs]@runtime_checkable class LayoutType(Protocol[_Render_co, _Event_contra]): """Renders and delivers, updates to views and events to handlers, respectively"""
[docs] async def render(self) -> _Render_co: """Render an update to a view"""
[docs] async def deliver(self, event: _Event_contra) -> None: """Relay an event to its respective handler"""
async def __aenter__(self) -> LayoutType[_Render_co, _Event_contra]: """Prepare the layout for its first render""" async def __aexit__( self, exc_type: type[Exception], exc_value: Exception, traceback: TracebackType, ) -> bool | None: """Clean up the view after its final render"""
VdomAttributes = Mapping[str, Any] """Describes the attributes of a :class:`VdomDict`""" VdomChild: TypeAlias = "ComponentType | VdomDict | str | None | Any" """A single child element of a :class:`VdomDict`""" VdomChildren: TypeAlias = "Sequence[VdomChild] | VdomChild" """Describes a series of :class:`VdomChild` elements""" class _VdomDictOptional(TypedDict, total=False): key: Key | None children: Sequence[ComponentType | VdomChild] attributes: VdomAttributes eventHandlers: EventHandlerDict importSource: ImportSourceDict class _VdomDictRequired(TypedDict, total=True): tagName: str
[docs]class VdomDict(_VdomDictRequired, _VdomDictOptional): """A :ref:`VDOM` dictionary"""
[docs]class ImportSourceDict(TypedDict): source: str fallback: Any sourceType: str unmountBeforeUpdate: bool
class _OptionalVdomJson(TypedDict, total=False): key: Key error: str children: list[Any] attributes: dict[str, Any] eventHandlers: dict[str, _JsonEventTarget] importSource: _JsonImportSource class _RequiredVdomJson(TypedDict, total=True): tagName: str
[docs]class VdomJson(_RequiredVdomJson, _OptionalVdomJson): """A JSON serializable form of :class:`VdomDict` matching the :data:`VDOM_JSON_SCHEMA`"""
class _JsonEventTarget(TypedDict): target: str preventDefault: bool stopPropagation: bool class _JsonImportSource(TypedDict): source: str fallback: Any EventHandlerMapping = Mapping[str, "EventHandlerType"] """A generic mapping between event names to their handlers""" EventHandlerDict: TypeAlias = "dict[str, EventHandlerType]" """A dict mapping between event names to their handlers"""
[docs]class EventHandlerFunc(Protocol): """A coroutine which can handle event data""" async def __call__(self, data: Sequence[Any]) -> None: ...
[docs]@runtime_checkable class EventHandlerType(Protocol): """Defines a handler for some event""" prevent_default: bool """Whether to block the event from propagating further up the DOM""" stop_propagation: bool """Stops the default action associate with the event from taking place.""" function: EventHandlerFunc """A coroutine which can respond to an event and its data""" target: str | None """Typically left as ``None`` except when a static target is useful. When testing, it may be useful to specify a static target ID so events can be triggered programmatically. .. note:: When ``None``, it is left to a :class:`LayoutType` to auto generate a unique ID. """
[docs]class VdomDictConstructor(Protocol): """Standard function for constructing a :class:`VdomDict`""" @overload def __call__( self, attributes: VdomAttributes, *children: VdomChildren ) -> VdomDict: ... @overload def __call__(self, *children: VdomChildren) -> VdomDict: ... @overload def __call__( self, *attributes_and_children: VdomAttributes | VdomChildren ) -> VdomDict: ...
[docs]class LayoutUpdateMessage(TypedDict): """A message describing an update to a layout""" type: Literal["layout-update"] """The type of message""" path: str """JSON Pointer path to the model element being updated""" model: VdomJson """The model to assign at the given JSON Pointer path"""
[docs]class LayoutEventMessage(TypedDict): """Message describing an event originating from an element in the layout""" type: Literal["layout-event"] """The type of message""" target: str """The ID of the event handler.""" data: Sequence[Any] """A list of event data passed to the event handler."""
[docs]class Context(Protocol[_Type]): """Returns a :class:`ContextProvider` component""" def __call__( self, *children: Any, value: _Type = ..., key: Key | None = ..., ) -> ContextProviderType[_Type]: ...
[docs]class ContextProviderType(ComponentType, Protocol[_Type]): """A component which provides a context value to its children""" type: Context[_Type] """The context type""" @property def value(self) -> _Type: "Current context value"