208 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
			
		
		
	
	
			208 lines
		
	
	
	
		
			6.6 KiB
		
	
	
	
		
			Python
		
	
	
	
	
	
# Copyright (C) 2006 Gustavo Carneiro <gjcarneiro AT gmail.com>
 | 
						|
#                    Nikos Kouremenos <kourem AT gmail.com>
 | 
						|
# Copyright (C) 2006-2014 Yann Leboulanger <asterix AT lagaule.org>
 | 
						|
# Copyright (C) 2008 Jean-Marie Traissard <jim AT lapin.org>
 | 
						|
#                    Jonathan Schleifer <js-gajim AT webkeks.org>
 | 
						|
#                    Stephan Erb <steve-e AT h3c.de>
 | 
						|
#
 | 
						|
# This file is part of Gajim.
 | 
						|
#
 | 
						|
# Gajim is free software; you can redistribute it and/or modify
 | 
						|
# it under the terms of the GNU General Public License as published
 | 
						|
# by the Free Software Foundation; version 3 only.
 | 
						|
#
 | 
						|
# Gajim is distributed in the hope that it will be useful,
 | 
						|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
						|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
						|
# GNU General Public License for more details.
 | 
						|
#
 | 
						|
# You should have received a copy of the GNU General Public License
 | 
						|
# along with Gajim. If not, see <http://www.gnu.org/licenses/>.
 | 
						|
 | 
						|
import logging
 | 
						|
 | 
						|
from gi.repository import GObject
 | 
						|
from gi.repository import Gio, GLib
 | 
						|
 | 
						|
log = logging.getLogger('gajim.c.dbus.music_track')
 | 
						|
 | 
						|
MPRIS_PLAYER_PREFIX = 'org.mpris.MediaPlayer2.'
 | 
						|
 | 
						|
 | 
						|
class MusicTrackInfo:
 | 
						|
    __slots__ = ['title', 'album', 'artist', 'duration', 'track_number',
 | 
						|
                 'paused']
 | 
						|
 | 
						|
 | 
						|
class MusicTrackListener(GObject.GObject):
 | 
						|
    __gsignals__ = {
 | 
						|
        'music-track-changed': (GObject.SignalFlags.RUN_LAST, None, (object,)),
 | 
						|
    }
 | 
						|
 | 
						|
    _instance = None
 | 
						|
 | 
						|
    @classmethod
 | 
						|
    def get(cls):
 | 
						|
        if cls._instance is None:
 | 
						|
            cls._instance = cls()
 | 
						|
        return cls._instance
 | 
						|
 | 
						|
    def __init__(self):
 | 
						|
        super().__init__()
 | 
						|
        self.players = {}
 | 
						|
 | 
						|
    def start(self):
 | 
						|
        proxy = Gio.DBusProxy.new_for_bus_sync(
 | 
						|
            Gio.BusType.SESSION,
 | 
						|
            Gio.DBusProxyFlags.NONE,
 | 
						|
            None,
 | 
						|
            'org.freedesktop.DBus',
 | 
						|
            '/org/freedesktop/DBus',
 | 
						|
            'org.freedesktop.DBus',
 | 
						|
            None)
 | 
						|
 | 
						|
        self.connection = proxy.get_connection()
 | 
						|
        self.connection.signal_subscribe(
 | 
						|
            'org.freedesktop.DBus',
 | 
						|
            'org.freedesktop.DBus',
 | 
						|
            'NameOwnerChanged',
 | 
						|
            '/org/freedesktop/DBus',
 | 
						|
            None,
 | 
						|
            Gio.DBusSignalFlags.NONE,
 | 
						|
            self._signal_name_owner_changed)
 | 
						|
 | 
						|
        try:
 | 
						|
            result = proxy.call_sync(
 | 
						|
                'ListNames',
 | 
						|
                None,
 | 
						|
                Gio.DBusCallFlags.NONE,
 | 
						|
                -1,
 | 
						|
                None)
 | 
						|
        except GLib.Error as error:
 | 
						|
            if error.domain == 'g-dbus-error-quark':
 | 
						|
                log.debug("Could not list names: %s", error.message)
 | 
						|
                return
 | 
						|
            raise
 | 
						|
 | 
						|
        for name in result[0]:
 | 
						|
            if name.startswith(MPRIS_PLAYER_PREFIX):
 | 
						|
                self._add_player(name)
 | 
						|
 | 
						|
    def stop(self):
 | 
						|
        for name in list(self.players):
 | 
						|
            if name.startswith(MPRIS_PLAYER_PREFIX):
 | 
						|
                self._remove_player(name)
 | 
						|
 | 
						|
    def _signal_name_owner_changed(self, connection, sender_name, object_path,
 | 
						|
                         interface_name, signal_name, parameters, *user_data):
 | 
						|
        name, oldOwner, newOwner = parameters
 | 
						|
        if name.startswith(MPRIS_PLAYER_PREFIX):
 | 
						|
            if newOwner and not oldOwner:
 | 
						|
                self._add_player(name)
 | 
						|
            else:
 | 
						|
                self._remove_player(name)
 | 
						|
 | 
						|
    def _add_player(self, name):
 | 
						|
        '''Set up a listener for music player signals'''
 | 
						|
        log.info('%s appeared', name)
 | 
						|
 | 
						|
        if name in self.players:
 | 
						|
            return
 | 
						|
 | 
						|
        self.players[name] = self.connection.signal_subscribe(
 | 
						|
            name,
 | 
						|
            'org.freedesktop.DBus.Properties',
 | 
						|
            'PropertiesChanged',
 | 
						|
            '/org/mpris/MediaPlayer2',
 | 
						|
            None,
 | 
						|
            Gio.DBusSignalFlags.NONE,
 | 
						|
            self._signal_received,
 | 
						|
            name)
 | 
						|
 | 
						|
        info = self.get_playing_track(name)
 | 
						|
        if info is not None:
 | 
						|
            self.emit('music-track-changed', info)
 | 
						|
 | 
						|
    def _remove_player(self, name):
 | 
						|
        log.info('%s vanished', name)
 | 
						|
        if name in self.players:
 | 
						|
            self.connection.signal_unsubscribe(
 | 
						|
                self.players[name])
 | 
						|
            self.players.pop(name)
 | 
						|
 | 
						|
            self.emit('music-track-changed', None)
 | 
						|
 | 
						|
    def _signal_received(self, connection, sender_name, object_path,
 | 
						|
                         interface_name, signal_name, parameters, *user_data):
 | 
						|
        '''Signal handler for PropertiesChanged event'''
 | 
						|
 | 
						|
        if 'PlaybackStatus' not in parameters[1]:
 | 
						|
            return
 | 
						|
 | 
						|
        log.info('Signal received: %s - %s', interface_name, parameters)
 | 
						|
 | 
						|
        info = self.get_playing_track(user_data[0])
 | 
						|
 | 
						|
        self.emit('music-track-changed', info)
 | 
						|
 | 
						|
    def _properties_extract(self, properties):
 | 
						|
        meta = properties.get('Metadata')
 | 
						|
        if meta is None or not meta:
 | 
						|
            return None
 | 
						|
 | 
						|
        info = MusicTrackInfo()
 | 
						|
        info.title = meta.get('xesam:title')
 | 
						|
        info.album = meta.get('xesam:album')
 | 
						|
        artist = meta.get('xesam:artist')
 | 
						|
        if artist:
 | 
						|
            info.artist = artist[0]
 | 
						|
        else:
 | 
						|
            info.artist = None
 | 
						|
        info.duration = float(meta.get('mpris:length', 0))
 | 
						|
        info.track_number = meta.get('xesam:trackNumber', 0)
 | 
						|
 | 
						|
        status = properties.get('PlaybackStatus')
 | 
						|
        info.paused = status is not None and status == 'Paused'
 | 
						|
 | 
						|
        return info
 | 
						|
 | 
						|
    def get_playing_track(self, name):
 | 
						|
        '''Return a MusicTrackInfo for the currently playing
 | 
						|
        song, or None if no song is playing'''
 | 
						|
        proxy = Gio.DBusProxy.new_for_bus_sync(
 | 
						|
            Gio.BusType.SESSION,
 | 
						|
            Gio.DBusProxyFlags.NONE,
 | 
						|
            None,
 | 
						|
            name,
 | 
						|
            '/org/mpris/MediaPlayer2',
 | 
						|
            'org.freedesktop.DBus.Properties',
 | 
						|
            None)
 | 
						|
 | 
						|
        try:
 | 
						|
            result = proxy.call_sync(
 | 
						|
                "GetAll",
 | 
						|
                GLib.Variant('(s)', ('org.mpris.MediaPlayer2.Player',)),
 | 
						|
                Gio.DBusCallFlags.NONE,
 | 
						|
                -1,
 | 
						|
                None)
 | 
						|
        except GLib.Error as error:
 | 
						|
            if error.domain == 'g-dbus-error-quark':
 | 
						|
                log.debug("Could not enable music listener: %s", error.message)
 | 
						|
                return None
 | 
						|
            raise
 | 
						|
        else:
 | 
						|
            info = self._properties_extract(result[0])
 | 
						|
            return info
 | 
						|
 | 
						|
 | 
						|
# here we test :)
 | 
						|
if __name__ == '__main__':
 | 
						|
    def music_track_change_cb(_listener, music_track_info):
 | 
						|
        if music_track_info is None or music_track_info.paused:
 | 
						|
            print('Stop!')
 | 
						|
        else:
 | 
						|
            print(music_track_info.title)
 | 
						|
    listener = MusicTrackListener.get()
 | 
						|
    listener.connect('music-track-changed', music_track_change_cb)
 | 
						|
    listener.start()
 | 
						|
    GLib.MainLoop().run()
 |