Skip to main content

C#-style event handling mechanism for Python

Project description

C#-Style Event Handling Mechanism for Python

pypi status downloads python license
build issues pull requests

C# provides a very simple syntax using the observer pattern for its event handling system. The aim of this project is to implement the pattern in python as similarly as possible.

In C#, an "event" is a field or a property of the delegate type EventHandler. Since delegates in C# can be combined and removed with += and -= operators, event handlers can easily subscribe to or unsubscribe from the event using those operators.

Python does not support an addition of two Callable types. So the Event[**TArgs] class is provided to mimic delegates:

from events import Event

item_changed = Event[str, object]()

C# naming convention prefers present/past participles (changing/changed) instead of on+infinitive (on_change) for events.

Handlers can subscribe to and unsubscribe from the event with the same syntax:

def item_changed_handler(key: str, value: object) -> None:
    ...

item_changed += item_changed_handler
item_changed -= item_changed_handler

An event can be raised by simply invoking it with the arguments:

item_changed("info", obj)

Since Event acts just like a delegate from C#, it is not required to be bound to a class or an instance object. This is the major difference to other packages that try to implement the C#-style event system, which usually revolve around a container object for events.

An example class with event fields may look like this:

class EventExample:
    def __init__(self) -> None:
        self.__value = ""
        self.updated: Event[str] = Event()

    def update(self, value: str) -> None:
        if self.__value != value:
            self.__value = value
            self.updated(value)

obj = EventExample()
obj.updated += lambda value: print(f"obj.{value=}")
obj.update("new value")

A class decorator @events is provided as a shortcut for event fields:

from events import Event, events

@events
class EventFieldsExample:
    item_added: Event[object]
    item_removed: Event[object]
    item_updated: Event[str, object]

C# also provides event properties with add and remove accessors:

public event EventHandler<ItemChangedEventArgs> ItemChanged
{
    add { ... }
    remove { ... }
}

This feature is useful for classes that do not actually own the events, but need to forward the subscriptions to the underlying object that do own the events.

The @event[**TArgs] decorator and the accessors[**TArgs] type are provided to support this feature:

from events import accessors, event, EventHandler

class EventPropertyExample:
    @event[str, object]
    def item_changed() -> accessors[str, object]:
        def add(self: Self, value: EventHandler[str, object]) -> None: ...
        def remove(self: Self, value: EventHandler[str, object]) -> None: ...
        return (add, remove)

Furthermore, the EventHandlerCollection interface is provided to support the functionalities of System.ComponentModel.EventHandlerList class from C#, along with the two implementations EventHandlerList and EventHandlerDict using a linked list and a dictionary respectively. The behaviour of EventHandlerList is exactly the same as of its counterpart from C#.

A typical usage of EventHandlerList in C# can be translated directly into the python code:

class EventPropertyExample:
    __event_item_changed: Final = object()

    def __init__(self) -> None:
        self.__events = EventHandlerList()

    @event  # [str, object] is inferred
    def item_changed():  # -> accessors[str, object] is inferred
        def add(self: Self, value: EventHandler[str, object]) -> None:
            self.__events.add_handler(self.__event_item_changed, value)

        def remove(self: Self, value: EventHandler[str, object]) -> None:
            self.__events.remove_handler(self.__event_item_changed, value)

        return (add, remove)

    def _on_item_changed(self, key: str, value: object) -> None:
        handler = self.__events[self.__event_item_changed]
        if handler:
            handler(key, value)

The class decorator @events also provides a shortcut for event properties. The above code can be shortened to:

@events(collection="__events")
class EventPropertyExample:
    item_changed: event[str, object]

    def __init__(self) -> None:
        self.__events = EventHandlerList()

    def _on_item_changed(self, key: str, value: object) -> None:
        self.__events.invoke("item_changed", key, value)

Installation

Install using pip:

pip install cs-events

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

cs_events-0.5.0.tar.gz (12.2 kB view details)

Uploaded Source

Built Distribution

cs_events-0.5.0-py3-none-any.whl (12.5 kB view details)

Uploaded Python 3

File details

Details for the file cs_events-0.5.0.tar.gz.

File metadata

  • Download URL: cs_events-0.5.0.tar.gz
  • Upload date:
  • Size: 12.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/1.8.2 CPython/3.12.3 Windows/11

File hashes

Hashes for cs_events-0.5.0.tar.gz
Algorithm Hash digest
SHA256 ba9183ccda2e7c22e8556fa6185f449085e9ba75ab444b0115c9539b341bdc34
MD5 72854bc1cae436710a39f5d44508d9e0
BLAKE2b-256 c606b0c092039e877f85a4db8e6524a17a5fa426f03fc3370d52990a913c6020

See more details on using hashes here.

File details

Details for the file cs_events-0.5.0-py3-none-any.whl.

File metadata

  • Download URL: cs_events-0.5.0-py3-none-any.whl
  • Upload date:
  • Size: 12.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/1.8.2 CPython/3.12.3 Windows/11

File hashes

Hashes for cs_events-0.5.0-py3-none-any.whl
Algorithm Hash digest
SHA256 851bb8a4c3adf4841622ae561d40e7098ee82c86fedc3b235157a007228714ef
MD5 cf769685e5fa7b34f7a2dc2dbaa08710
BLAKE2b-256 ae10136d9eaa4e465bef83edd166fa53ddf02211e2f6dcea7be9a49c25da6a25

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page