Source code for mopidy.mixer

# ruff: noqa: ARG002

from __future__ import annotations

import logging
from typing import TYPE_CHECKING, Any, ClassVar

import pykka
from pykka.typing import ActorMemberMixin, proxy_field, proxy_method

from mopidy import listener

if TYPE_CHECKING:
    from mopidy.types import Percentage


logger = logging.getLogger(__name__)


[docs] class Mixer: """Audio mixer API. If the mixer has problems during initialization it should raise :exc:`mopidy.exceptions.MixerError` with a descriptive error message. This will make Mopidy print the error message and exit so that the user can fix the issue. :param config: the entire Mopidy configuration """ name: ClassVar[str] = "" """ Name of the mixer. Used when configuring what mixer to use. Should match the :attr:`~mopidy.ext.Extension.ext_name` of the extension providing the mixer. """ def __init__(self, config: dict) -> None: pass
[docs] def get_volume(self) -> Percentage | None: """Get volume level of the mixer on a linear scale from 0 to 100. Example values: 0: Minimum volume, usually silent. 100: Maximum volume. :class:`None`: Volume is unknown. *MAY be implemented by subclass.* """ return None
[docs] def set_volume(self, volume: Percentage) -> bool: """Set volume level of the mixer. *MAY be implemented by subclass.* Returns :class:`True` if successful, :class:`False` otherwise. :param volume: Volume in the range [0..100] """ return False
[docs] def trigger_volume_changed(self, volume: Percentage) -> None: """Send ``volume_changed`` event to all mixer listeners. This method should be called by subclasses when the volume is changed, either because of a call to :meth:`set_volume` or because of any external entity changing the volume. """ logger.debug("Mixer event: volume_changed(volume=%d)", volume) MixerListener.send("volume_changed", volume=volume)
[docs] def get_mute(self) -> bool | None: """Get mute state of the mixer. *MAY be implemented by subclass.* Returns :class:`True` if muted, :class:`False` if unmuted, and :class:`None` if unknown. """ return None
[docs] def set_mute(self, mute: bool) -> bool: """Mute or unmute the mixer. *MAY be implemented by subclass.* Returns :class:`True` if successful, :class:`False` otherwise. :param mute: :class:`True` to mute, :class:`False` to unmute """ return False
[docs] def trigger_mute_changed(self, mute: bool) -> None: """Send ``mute_changed`` event to all mixer listeners. This method should be called by subclasses when the mute state is changed, either because of a call to :meth:`set_mute` or because of any external entity changing the mute state. """ logger.debug("Mixer event: mute_changed(mute=%s)", mute) MixerListener.send("mute_changed", mute=mute)
[docs] def ping(self) -> bool: """Called to check if the actor is still alive.""" return True
[docs] class MixerListener(listener.Listener): """Marker interface for recipients of events sent by the mixer actor. Any Pykka actor that mixes in this class will receive calls to the methods defined here when the corresponding events happen in the mixer actor. This interface is used both for looking up what actors to notify of the events, and for providing default implementations for those listeners that are not interested in all events. """
[docs] @staticmethod def send(event: str, **kwargs: Any) -> None: """Helper to allow calling of mixer listener events.""" listener.send(MixerListener, event, **kwargs)
[docs] def volume_changed(self, volume: Percentage) -> None: """Called after the volume has changed. *MAY* be implemented by actor. :param volume: the new volume """
[docs] def mute_changed(self, mute: bool) -> None: """Called after the mute state has changed. *MAY* be implemented by actor. :param mute: :class:`True` if muted, :class:`False` if not muted :type mute: bool """
class MixerActor(pykka.ThreadingActor, Mixer): pass class MixerProxy(ActorMemberMixin, pykka.ActorProxy[MixerActor]): """Mixer wrapped in a Pykka actor proxy.""" name = proxy_field(MixerActor.name) get_volume = proxy_method(MixerActor.get_volume) set_volume = proxy_method(MixerActor.set_volume) trigger_volume_changed = proxy_method(MixerActor.trigger_volume_changed) get_mute = proxy_method(MixerActor.get_mute) set_mute = proxy_method(MixerActor.set_mute) trigger_mute_changed = proxy_method(MixerActor.trigger_mute_changed) ping = proxy_method(MixerActor.ping)