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 cs_events import Event

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 changed_handler(key: str, value: object) -> None:
    ...

changed += changed_handler
changed -= changed_handler

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

changed("state", 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.updated: Event[str] = Event()
        self.__value: str = ""

    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 cs_events import Event, events

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

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

public event EventHandler<T> Changed
{
    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 cs_events import accessors, event, EventHandler

class EventPropertyExample:
    @event[str, object]
    def 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_changed: Final = object()

    @event  # [str, object] is inferred
    def changed():  # -> accessors[str, object] is inferred
        def add(self: Self, value: EventHandler[str, object]) -> None:
            self.__events.add_handler(self.__event_changed, value)
        def remove(self: Self, value: EventHandler[str, object]) -> None:
            self.__events.remove_handler(self.__event_changed, value)
        return (add, remove)
    
    def __init__(self) -> None:
        self.__events = EventHandlerList()
    
    def perform_change(self, key: str, value: object) -> None:
        handler = self.__events[self.__event_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:
    changed: event[str, object]

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

    def perform_change(self, key: str, value: object) -> None:
        self.__events.invoke("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.3.0.tar.gz (10.8 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

cs_events-0.3.0-py3-none-any.whl (10.2 kB view details)

Uploaded Python 3

File details

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

File metadata

  • Download URL: cs_events-0.3.0.tar.gz
  • Upload date:
  • Size: 10.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/1.4.0 CPython/3.11.2 Windows/10

File hashes

Hashes for cs_events-0.3.0.tar.gz
Algorithm Hash digest
SHA256 c5e034dfa94c11231f9d37c7fe9958b2d3d04b42598a23f6fe18cd3ac03c9c95
MD5 b24197058c081822f19e3ebf79718d44
BLAKE2b-256 f2a0dd5cfe0a72af21bf4d0feb5b70bc07667ae5a62e956eabeca0d4d9e914ad

See more details on using hashes here.

File details

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

File metadata

  • Download URL: cs_events-0.3.0-py3-none-any.whl
  • Upload date:
  • Size: 10.2 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: poetry/1.4.0 CPython/3.11.2 Windows/10

File hashes

Hashes for cs_events-0.3.0-py3-none-any.whl
Algorithm Hash digest
SHA256 7a7971520af331b44b88f8b8e58b11124b52e465c49a51915547c2aaaa0db680
MD5 257f10c1aa5676e51c7244dbb77ee180
BLAKE2b-256 7d0c163d63a2f907436158a05f79c3f94fd02332958c1f1cdb0b6b4bbe8a42e3

See more details on using hashes here.

Supported by

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