Skip to content

Instantly share code, notes, and snippets.

@squentin
Created January 16, 2012 14:42
mpris2 plugin with patch for Net::DBus v1.0.0
# Copyright (C) 2011 Quentin Sculo <squentin@free.fr>
#
# This file is part of Gmusicbrowser.
# Gmusicbrowser is free software; you can redistribute it and/or modify
# it under the terms of the GNU General Public License version 3, as
# published by the Free Software Foundation
=for gmbplugin MPRIS2
name MPRIS v2
title MPRIS v2 support
desc Allows controlling gmusicbrowser via DBus using the MPRIS v2.0 standard
req perl(Net::DBus, libnet-dbus-perl perl-Net-DBus)
=cut
package GMB::Plugin::MPRIS2;
use strict;
use warnings;
use constant
{ OPT => 'PLUGIN_MPRIS2_',
};
use Net::DBus::Annotation 'dbus_call_async';
my $bus=$GMB::DBus::bus;
die "Requires DBus support to be active\n" unless $bus; #only requires this to use the hack in gmusicbrowser_dbus.pm so that Net::DBus::GLib is not required, else could do just : use Net::DBus::GLib; $bus=Net::DBus::GLib->session;
my @Objects;
sub Start
{ my $service= $bus->export_service('org.mpris.MediaPlayer2.gmusicbrowser');
push @Objects, GMB::DBus::MPRIS2->new($service);
}
sub Stop
{ ::UnWatch_all($_) for @Objects;
$_->disconnect for @Objects;
@Objects=();
}
sub prefbox
{ my $vbox=Gtk2::VBox->new(0,2);
my $blacklist= Gtk2::CheckButton->new(_"Show in sound menu");
soundmenu_button_update($blacklist);
$blacklist->signal_connect(toggled => \&soundmenu_toggle_cb);
$vbox->pack_start($blacklist,0,0,0);
return $vbox;
}
sub soundmenu_button_update
{ my $check=shift;
eval
{ my $service = $bus->get_service('com.canonical.indicators.sound');
my $object = $service->get_object('/com/canonical/indicators/sound/service');
my $asyncreply=$object->IsBlacklisted(dbus_call_async,'gmusicbrowser'); #called async, because it seems to trigger the calling of gmb DBus methods before replying
$check->{busy}=1;
$asyncreply->set_notify(sub { soundmenu_button_set($check, eval {$_[0]->get_result;}) });
};
soundmenu_button_set($check,undef) unless $check->{busy};
}
sub soundmenu_button_set
{ my ($check,$on)=@_;
$check->set_active(1) if !$on;
if (!defined $on)
{ $check->set_sensitive(0);
$check->set_tooltip_text(_"No sound menu found");
}
delete $check->{busy};
}
sub soundmenu_toggle_cb
{ my $check=shift;
return if $check->{busy};
my $on=$check->get_active;
eval
{ my $service = $bus->get_service('com.canonical.indicators.sound');
my $object = $service->get_object('/com/canonical/indicators/sound/service');
$object->BlacklistMediaPlayer('gmusicbrowser',!$on);
};
soundmenu_button_update($check);
}
package GMB::DBus::MPRIS2;
use base 'Net::DBus::Object';
use Net::DBus::Exporter 'org.mpris.MediaPlayer2';
use Net::DBus ':typing';
our %PropChanged;
# events watched by properties of org.mpris.MediaPlayer2.Player that send PropertiesChanged signal
# the functions associated with these properties must bless the return value with dbus_string() and friends
my %PropertiesWatch=
( PlaybackStatus => 'Playing',
LoopStatus => 'Lock Repeat',
Shuffle => 'Sort',
Metadata => 'CurSong',
Volume => 'Vol',
CanGoNext => 'Playlist Sort Queue Repeat',
CanGoPrevious => 'CurSongID',
CanPlay => 'CurSongID',
#CanSeek => 'CurSongID', # always true currently => no need to watch event
);
sub new
{ my ($class,$service) = @_;
my $self = $class->SUPER::new($service, '/org/mpris/MediaPlayer2');
bless $self, $class;
::Watch($self, Seek => \&Seeked);
#watchers for properties of org.mpris.MediaPlayer2.Player that send PropertiesChanged signal
my %events;
for my $prop (sort keys %PropertiesWatch)
{ push @{ $events{$_} }, $prop for split / /, $PropertiesWatch{$prop};
}
for my $event (keys %events)
{ my $props= $events{$event};
::Watch($self, $event =>
sub { my $self=shift;
$PropChanged{$_}=1 for @$props;
::IdleDo('2_MPRIS2_propchanged', 500, \&PropertiesChanged, $self);
});
}
return $self;
}
dbus_signal(PropertiesChanged => ['string',['dict','string',['variant']],['array','string']], 'org.freedesktop.DBus.Properties');
sub PropertiesChanged
{ my $self=shift;
my %changed;
for my $name (keys %PropChanged)
{ no strict "refs";
$changed{$name}= $name->();
}
%PropChanged=();
$self->emit_signal( PropertiesChanged => 'org.mpris.MediaPlayer2.Player', \%changed,[] );
}
dbus_method('Raise', [], [],{no_return=>1});
sub Raise
{ ::ShowHide(1);
}
dbus_method('Quit', [], [],{no_return=>1});
sub Quit
{ ::Quit();
}
dbus_property('CanQuit', 'bool', 'read');
sub CanQuit {dbus_boolean(1)}
dbus_property('CanRaise', 'bool', 'read');
sub CanRaise {dbus_boolean(1)}
dbus_property('HasTrackList', 'bool', 'read');
sub HasTrackList {dbus_boolean(0)}
dbus_property('Identity', 'string', 'read');
sub Identity { 'gmusicbrowser' }
dbus_property('DesktopEntry', 'string', 'read');
sub DesktopEntry { 'gmusicbrowser' }
dbus_property('SupportedUriSchemes', ['array','string'], 'read');
sub SupportedUriSchemes { return ['file']; }
dbus_property('SupportedMimeTypes', ['array','string'], 'read');
sub SupportedMimeTypes { return [qw(audio/mpeg application/ogg audio/x-flac audio/x-musepack audio/x-m4a)]; } #FIXME
dbus_method('Next', [], [], 'org.mpris.MediaPlayer2.Player', {no_return=>1});
sub Next { ::NextSong(); }
dbus_method('Previous', [], [], 'org.mpris.MediaPlayer2.Player', {no_return=>1});
sub Previous { ::PrevSong(); }
dbus_method('Pause', [], [], 'org.mpris.MediaPlayer2.Player', {no_return=>1});
sub Pause { ::Pause(); }
dbus_method('PlayPause',[], [], 'org.mpris.MediaPlayer2.Player', {no_return=>1});
sub PlayPause { ::PlayPause(); }
dbus_method('Stop', [], [], 'org.mpris.MediaPlayer2.Player', {no_return=>1});
sub Stop { ::Stop(); }
dbus_method('Play', [], [], 'org.mpris.MediaPlayer2.Player', {no_return=>1});
sub Play { ::Play(); }
dbus_method('Seek', ['int64'], [], 'org.mpris.MediaPlayer2.Player', {no_return=>1});
sub Seek
{ my $offset= $_[1]/1_000_000; #convert from microseconds
my $sec= $::PlayTime || 0;
return unless defined $::SongID;
if ($offset>0)
{ $sec+=$offset;
my $length= Songs::Get($::SongID,'length');
if ($sec>$length) { ::NextSong(); }
else { ::SkipTo($sec) }
}
elsif ($offset<0)
{ $sec+=$offset;
if ($sec<0) { ::PrevSong(); }
else { ::SkipTo($sec) }
}
}
dbus_method('SetPosition', [['struct','objectpath','int64']], [], 'org.mpris.MediaPlayer2.Player', {no_return=>1});
sub SetPosition
{ my ($ID,$position)= @{ $_[1] };
return unless defined $::SongID && $ID==$::SongID;
$position/=1_000_000;
my $length= Songs::Get($::SongID,'length');
return if $length<0 || $position>$length;
::SkipTo($position);
}
dbus_method('OpenUri', ['string'], [], 'org.mpris.MediaPlayer2.Player', {no_return=>1});
sub OpenUri
{ my $uri=$_[1];
my $IDs= ::Uris_to_IDs($uri);
my $ID= $IDs->[0];
::Select(song => $ID, play=>1) if defined $ID;
}
dbus_signal(Seeked => ['int64'], 'org.mpris.MediaPlayer2.Player');
sub Seeked
{ $_[0]->emit_signal( Seeked => $_[1]*1_000_000 );
}
dbus_property('PlaybackStatus', 'string', 'read', 'org.mpris.MediaPlayer2.Player');
sub PlaybackStatus
{ my $status= $::TogPlay ? 'Playing' : defined $::TogPlay ? 'Paused' : 'Stopped';
return dbus_string($status);
}
dbus_property('LoopStatus', 'string', 'readwrite', 'org.mpris.MediaPlayer2.Player');
sub LoopStatus
{ if (defined $_[1])
{ my $m=$_[1];
my $notrack;
if ($m eq 'None') { ::SetRepeat(0); $notrack=1; }
elsif ($m eq 'Track') { ::SetRepeat(1); ::ToggleLock('fullfilename',1); }
elsif ($m eq 'Playlist'){ ::SetRepeat(1); $notrack=1; }
if ($notrack && $::TogLock && $::TogLock eq 'fullfilename') { ::ToggleLock('fullfilename') }
}
else
{ my $r= !$::Options{Repeat} ? 'None' :
($::TogLock && $::TogLock eq 'fullfilename') ? 'Track' : 'Playlist';
return dbus_string($r);
}
}
dbus_property('Rate', 'double', 'readwrite', 'org.mpris.MediaPlayer2.Player');
sub Rate {dbus_double(1)}
dbus_property('MinimumRate', 'double', 'read', 'org.mpris.MediaPlayer2.Player');
sub MinimumRate {dbus_double(1)}
dbus_property('MaximumRate', 'double', 'read', 'org.mpris.MediaPlayer2.Player');
sub MaximumRate {dbus_double(1)}
dbus_property('Shuffle', 'bool', 'readwrite', 'org.mpris.MediaPlayer2.Player');
sub Shuffle
{ my $on= ($::RandomMode || $::Options{Sort}=~m/shuffle/) ? 1 : 0;
return dbus_boolean($on) if !defined $_[1];
::ToggleSort() if $_[1] xor $on;
}
dbus_property('Metadata', ['dict','string',['variant']], 'read', 'org.mpris.MediaPlayer2.Player');
sub Metadata
{ GetMetadata_from($::SongID);
}
dbus_property('Volume', 'double', 'readwrite', 'org.mpris.MediaPlayer2.Player');
sub Volume
{ if (defined $_[1]) { my $v=$_[1]; $v=0 if $v<0; ::ChangeVol($v); }
else { return dbus_double($::Volume/100); }
}
dbus_property('Position', 'int64', 'read', 'org.mpris.MediaPlayer2.Player');
sub Position
{ return dbus_int64( ($::PlayTime||0) *1_000_000 );
}
dbus_property('CanGoNext', 'bool', 'read', 'org.mpris.MediaPlayer2.Player');
sub CanGoNext
{ return dbus_boolean(1) if !defined $::Position && @$::ListPlay;
return dbus_boolean(1) if @$::Queue;
return dbus_boolean(0) unless @$::ListPlay>1;
return dbus_boolean(0) if !$::Options{Repeat} && $::Position==$#$::ListPlay;
return dbus_boolean(1);
}
dbus_property('CanGoPrevious', 'bool', 'read', 'org.mpris.MediaPlayer2.Player');
sub CanGoPrevious
{ return dbus_boolean( @$::Recent > ($::RecentPos||0) );
}
dbus_property('CanPlay', 'bool', 'read', 'org.mpris.MediaPlayer2.Player');
sub CanPlay
{ return dbus_boolean( defined $::SongID );
}
dbus_property('CanSeek', 'bool', 'read', 'org.mpris.MediaPlayer2.Player');
sub CanSeek
{ return dbus_boolean( defined $::SongID ); #will need to check if stream when supported
}
dbus_property('CanControl', 'bool', 'read', 'org.mpris.MediaPlayer2.Player');
sub CanControl {dbus_boolean(1)}
# 'org.mpris.MediaPlayer2.Player','Metadata'
sub GetMetadata_from
{ my $ID=shift;
# Net::DBus support for properties is incomplete, the following use undocumented functions to force it to use the correct data types for the returned values
my $type=
[ &Net::DBus::Binding::Message::TYPE_DICT_ENTRY,
[ &Net::DBus::Binding::Message::TYPE_STRING,
[ &Net::DBus::Binding::Message::TYPE_VARIANT,
[],
]]];
#my ($type)= Net::DBus::Binding::Introspector->_convert(['dict','string',['variant']]); #works too, not sure which one is best
return Net::DBus::Binding::Value->new($type,{}) unless defined $ID;
my %h;
$h{$_}=Songs::Get($ID,$_) for qw/title album artist comment length track disc year album_artist uri album_picture rating bitrate samprate genre playcount/;
my %r= #return values
( 'mpris:length' => dbus_int64($h{'length'}*1_000_000),
'mpris:trackid' => dbus_string($ID), #FIXME should contain a string that uniquely identifies the track within the scope of the playlist
'xesam:album' => dbus_string($h{album}),
'xesam:albumArtist' => dbus_array([ $h{album_artist} ]),
'xesam:artist' => dbus_array([ $h{artist} ]),
'xesam:comment' => ( $h{comment} ne '' ? dbus_array([$h{comment}]) : undef ),
'xesam:contentCreated' => ($h{year} ? dbus_string($h{year}) : undef), # ."-01-01T00:00Z" ?
'xesam:discNumber' => ($h{disc} ? dbus_int32($h{disc}) : undef),
'xesam:genre', => dbus_array([split /\x00/, $h{genre}]),
'xesam:lastUsed', => ($h{lastplay} ? dbus_string( ::strftime("%FT%RZ",gmtime($h{lastplay})) ) : undef),
'xesam:title', => dbus_string( $h{title} ),
'xesam:trackNumber' => ( $h{track} ? dbus_int32($h{track}) : undef),
'xesam:url' => dbus_string( $h{uri} ),
'xesam:useCount' => dbus_int32($h{playcount}),
# FIXME check if field exists
#'xesam:audioBPM' =>
#'xesam:composer' => dbus_array([ $h{composer} ]),
#'xesam:lyricist', => dbus_array([ $h{lyricist} ]),
);
my $rating=$h{rating};
if (defined $rating && length $rating) { $r{'xesam:userRating'}=dbus_double($rating/100); }
if (my $pic= $h{album_picture}) #FIXME use ~album.picture.uri when available
{ $r{'mpris:artUrl'}= dbus_string( 'file://'.::url_escape($pic) ) if $pic=~m/\.(?:jpe?g|png|gif)$/i; # ignore embedded pictures
}
delete $r{$_} for grep !defined $r{$_}, keys %r;
return Net::DBus::Binding::Value->new($type,\%r);
}
### patched version of Net::DBus::Object::_dispatch_all_prop_read v1.0.0 to support properties of different types
### Net::DBus::Object::_dispatch_all_prop_read was added in Net::DBus v1.0.0 to support the org.freedesktop.DBus.Properties.GetAll method
sub Net::DBus::Object::_dispatch_all_prop_read {
my $self = shift;
my $connection = shift;
my $message = shift;
my $ins = $self->_introspector;
if (!$ins) {
return $connection->make_error_message($message,
"org.freedesktop.DBus.Error.Failed",
"no introspection data exported for properties");
}
my ($pinterface) = $ins->decode($message, "methods", "Get", "params");
my %values = ();
foreach my $pname ($ins->list_properties($pinterface)) {
unless ($ins->is_property_readable($pinterface, $pname)) {
next; # skip write-only properties
}
$values{$pname} = eval {
$self->$pname;
};
if ($@) {
return $connection->make_error_message($message,
"org.freedesktop.DBus.Error.Failed",
"error reading '$pname' in interface '$pinterface': $@");
}
}
my $reply = $connection->make_method_return_message($message);
### patch : force variant type for values
my $Vtype=
[ &Net::DBus::Binding::Message::TYPE_DICT_ENTRY,
[ &Net::DBus::Binding::Message::TYPE_STRING,
[ &Net::DBus::Binding::Message::TYPE_VARIANT,
[],
]]];
my $values= Net::DBus::Binding::Value->new($Vtype,\%values);
$self->_introspector->encode($reply, "methods", "Get", "returns", $values);
###
### $self->_introspector->encode($reply, "methods", "Get", "returns", %\values);
### end of patch
return $reply;
}
1;
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment