Source code for reactpy.core.events

from __future__ import annotations

import asyncio
from collections.abc import Sequence
from typing import Any, Callable, Literal, overload

from anyio import create_task_group

from reactpy.core.types import EventHandlerFunc, EventHandlerType


@overload
def event(
    function: Callable[..., Any],
    *,
    stop_propagation: bool = ...,
    prevent_default: bool = ...,
) -> EventHandler: ...


@overload
def event(
    function: Literal[None] = ...,
    *,
    stop_propagation: bool = ...,
    prevent_default: bool = ...,
) -> Callable[[Callable[..., Any]], EventHandler]: ...


[docs]def event( function: Callable[..., Any] | None = None, *, stop_propagation: bool = False, prevent_default: bool = False, ) -> EventHandler | Callable[[Callable[..., Any]], EventHandler]: """A decorator for constructing an :class:`EventHandler`. While you're always free to add callbacks by assigning them to an element's attributes .. code-block:: python element = reactpy.html.button({"onClick": my_callback}) You may want the ability to prevent the default action associated with the event from taking place, or stopping the event from propagating up the DOM. This decorator allows you to add that functionality to your callbacks. .. code-block:: python @event(stop_propagation=True, prevent_default=True) def my_callback(*data): ... element = reactpy.html.button({"onClick": my_callback}) Parameters: function: A function or coroutine responsible for handling the event. stop_propagation: Block the event from propagating further up the DOM. prevent_default: Stops the default actional associate with the event from taking place. """ def setup(function: Callable[..., Any]) -> EventHandler: return EventHandler( to_event_handler_function(function, positional_args=True), stop_propagation, prevent_default, ) if function is not None: return setup(function) else: return setup
[docs]class EventHandler: """Turn a function or coroutine into an event handler Parameters: function: The function or coroutine which handles the event. stop_propagation: Block the event from propagating further up the DOM. prevent_default: Stops the default action associate with the event from taking place. target: A unique identifier for this event handler (auto-generated by default) """ __slots__ = ( "__weakref__", "function", "prevent_default", "stop_propagation", "target", ) def __init__( self, function: EventHandlerFunc, stop_propagation: bool = False, prevent_default: bool = False, target: str | None = None, ) -> None: self.function = to_event_handler_function(function, positional_args=False) self.prevent_default = prevent_default self.stop_propagation = stop_propagation self.target = target def __eq__(self, other: object) -> bool: undefined = object() for attr in ( "function", "prevent_default", "stop_propagation", "target", ): if not attr.startswith("_"): if not getattr(other, attr, undefined) == getattr(self, attr): return False return True def __repr__(self) -> str: public_names = [name for name in self.__slots__ if not name.startswith("_")] items = ", ".join([f"{n}={getattr(self, n)!r}" for n in public_names]) return f"{type(self).__name__}({items})"
[docs]def to_event_handler_function( function: Callable[..., Any], positional_args: bool = True, ) -> EventHandlerFunc: """Make a :data:`~reactpy.core.proto.EventHandlerFunc` from a function or coroutine Parameters: function: A function or coroutine accepting a number of positional arguments. positional_args: Whether to pass the event parameters a positional args or as a list. """ if positional_args: if asyncio.iscoroutinefunction(function): async def wrapper(data: Sequence[Any]) -> None: await function(*data) else: async def wrapper(data: Sequence[Any]) -> None: function(*data) return wrapper elif not asyncio.iscoroutinefunction(function): async def wrapper(data: Sequence[Any]) -> None: function(data) return wrapper else: return function
[docs]def merge_event_handlers( event_handlers: Sequence[EventHandlerType], ) -> EventHandlerType: """Merge multiple event handlers into one Raises a ValueError if any handlers have conflicting :attr:`~reactpy.core.proto.EventHandlerType.stop_propagation` or :attr:`~reactpy.core.proto.EventHandlerType.prevent_default` attributes. """ if not event_handlers: msg = "No event handlers to merge" raise ValueError(msg) elif len(event_handlers) == 1: return event_handlers[0] first_handler = event_handlers[0] stop_propagation = first_handler.stop_propagation prevent_default = first_handler.prevent_default target = first_handler.target for handler in event_handlers: if ( handler.stop_propagation != stop_propagation or handler.prevent_default != prevent_default or handler.target != target ): msg = "Cannot merge handlers - 'stop_propagation', 'prevent_default' or 'target' mismatch." raise ValueError(msg) return EventHandler( merge_event_handler_funcs([h.function for h in event_handlers]), stop_propagation, prevent_default, target, )
[docs]def merge_event_handler_funcs( functions: Sequence[EventHandlerFunc], ) -> EventHandlerFunc: """Make one event handler function from many""" if not functions: msg = "No event handler functions to merge" raise ValueError(msg) elif len(functions) == 1: return functions[0] async def await_all_event_handlers(data: Sequence[Any]) -> None: async with create_task_group() as group: for func in functions: group.start_soon(func, data) return await_all_event_handlers