2020-06-25 17:23:45 +00:00
|
|
|
"""Displays the current song being played and allows pausing, skipping ahead, and skipping back.
|
|
|
|
|
|
|
|
Requires the following library:
|
|
|
|
* python-dbus
|
|
|
|
|
|
|
|
Parameters:
|
2020-06-25 17:53:53 +00:00
|
|
|
* spotify.format: Format string (defaults to '{artist} - {title}')
|
2020-06-25 17:23:45 +00:00
|
|
|
Available values are: {album}, {title}, {artist}, {trackNumber}
|
2020-06-25 17:53:53 +00:00
|
|
|
* spotify.layout: Comma-separated list to change order of widgets (defaults to song, previous, pause, next)
|
|
|
|
Widget names are: spotify.song, spotify.prev, spotify.pause, spotify.next
|
2020-10-04 14:36:24 +00:00
|
|
|
* spotify.concise_controls: When enabled, allows spotify to be controlled from just the spotify.song widget.
|
|
|
|
Concise controls are: Left Click: Toggle Pause; Wheel Up: Next; Wheel Down; Previous.
|
2020-06-25 18:00:05 +00:00
|
|
|
|
|
|
|
contributed by `yvesh <https://github.com/yvesh>`_ - many thanks!
|
|
|
|
|
|
|
|
added controls by `LtPeriwinkle <https://github.com/LtPeriwinkle>`_ - many thanks!
|
2020-06-30 05:20:54 +00:00
|
|
|
|
|
|
|
fixed icons and layout parameter by `gkeep <https://github.com/gkeep>`_ - many thanks!
|
2020-06-25 17:23:45 +00:00
|
|
|
"""
|
2020-04-25 08:49:02 +00:00
|
|
|
|
2020-06-25 18:00:57 +00:00
|
|
|
import sys
|
|
|
|
import dbus
|
|
|
|
|
|
|
|
import core.module
|
|
|
|
import core.widget
|
|
|
|
import core.input
|
|
|
|
import core.decorators
|
|
|
|
import util.format
|
|
|
|
|
2020-09-06 12:27:17 +00:00
|
|
|
import logging
|
|
|
|
|
2020-04-25 08:51:40 +00:00
|
|
|
class Module(core.module.Module):
|
2020-04-26 14:39:24 +00:00
|
|
|
def __init__(self, config, theme):
|
2020-06-25 17:23:45 +00:00
|
|
|
super().__init__(config, theme, [])
|
2020-04-25 08:49:02 +00:00
|
|
|
|
2020-09-09 12:15:16 +00:00
|
|
|
self.background = True
|
|
|
|
|
2020-06-30 05:20:54 +00:00
|
|
|
self.__layout = util.format.aslist(
|
|
|
|
self.parameter(
|
|
|
|
"layout", "spotify.song,spotify.prev,spotify.pause,spotify.next",
|
|
|
|
)
|
2020-06-25 17:23:45 +00:00
|
|
|
)
|
2020-05-03 09:15:52 +00:00
|
|
|
|
2020-09-13 09:16:57 +00:00
|
|
|
self.__bus = dbus.SessionBus()
|
2020-05-03 09:15:52 +00:00
|
|
|
self.__song = ""
|
2020-06-25 17:23:45 +00:00
|
|
|
self.__pause = ""
|
2020-05-03 09:15:52 +00:00
|
|
|
self.__format = self.parameter("format", "{artist} - {title}")
|
2020-04-25 08:49:02 +00:00
|
|
|
|
2020-06-25 17:23:45 +00:00
|
|
|
self.__cmd = "dbus-send --session --type=method_call --dest=org.mpris.MediaPlayer2.spotify \
|
2020-05-03 09:15:52 +00:00
|
|
|
/org/mpris/MediaPlayer2 org.mpris.MediaPlayer2.Player."
|
2020-04-25 08:49:02 +00:00
|
|
|
|
2020-09-06 12:00:32 +00:00
|
|
|
widget_map = {}
|
|
|
|
for widget_name in self.__layout:
|
|
|
|
widget = self.add_widget(name=widget_name)
|
|
|
|
if widget_name == "spotify.prev":
|
|
|
|
widget_map[widget] = {
|
|
|
|
"button": core.input.LEFT_MOUSE,
|
|
|
|
"cmd": self.__cmd + "Previous",
|
|
|
|
}
|
|
|
|
widget.set("state", "prev")
|
|
|
|
elif widget_name == "spotify.pause":
|
|
|
|
widget_map[widget] = {
|
|
|
|
"button": core.input.LEFT_MOUSE,
|
|
|
|
"cmd": self.__cmd + "PlayPause",
|
|
|
|
}
|
|
|
|
elif widget_name == "spotify.next":
|
|
|
|
widget_map[widget] = {
|
|
|
|
"button": core.input.LEFT_MOUSE,
|
|
|
|
"cmd": self.__cmd + "Next",
|
|
|
|
}
|
|
|
|
widget.set("state", "next")
|
|
|
|
elif widget_name == "spotify.song":
|
2020-10-04 14:36:24 +00:00
|
|
|
if util.format.asbool(self.parameter("concise_controls", "false")):
|
|
|
|
widget_map[widget] = [
|
|
|
|
{
|
|
|
|
"button": core.input.LEFT_MOUSE,
|
|
|
|
"cmd": self.__cmd + "PlayPause",
|
|
|
|
}, {
|
|
|
|
"button": core.input.WHEEL_UP,
|
|
|
|
"cmd": self.__cmd + "Next",
|
|
|
|
}, {
|
|
|
|
"button": core.input.WHEEL_DOWN,
|
|
|
|
"cmd": self.__cmd + "Previous",
|
|
|
|
}
|
|
|
|
]
|
2020-09-06 12:00:32 +00:00
|
|
|
else:
|
|
|
|
raise KeyError(
|
|
|
|
"The spotify module does not have a {widget_name!r} widget".format(
|
|
|
|
widget_name=widget_name
|
|
|
|
)
|
|
|
|
)
|
2020-10-04 14:36:24 +00:00
|
|
|
# is there any reason the inputs can't be directly registered above?
|
2020-09-06 12:00:32 +00:00
|
|
|
for widget, callback_options in widget_map.items():
|
2020-10-04 14:36:24 +00:00
|
|
|
if isinstance(callback_options, dict):
|
|
|
|
core.input.register(widget, **callback_options)
|
|
|
|
|
|
|
|
elif isinstance(callback_options, list): # used by concise_controls
|
|
|
|
for opts in callback_options:
|
|
|
|
core.input.register(widget, **opts)
|
2020-09-06 12:00:32 +00:00
|
|
|
|
|
|
|
|
2020-04-25 08:49:02 +00:00
|
|
|
def hidden(self):
|
2020-05-03 09:15:52 +00:00
|
|
|
return self.string_song == ""
|
2020-04-25 08:49:02 +00:00
|
|
|
|
2020-06-25 17:53:53 +00:00
|
|
|
def __get_song(self):
|
2020-09-13 09:16:57 +00:00
|
|
|
bus = self.__bus
|
2020-06-25 17:53:53 +00:00
|
|
|
spotify = bus.get_object(
|
|
|
|
"org.mpris.MediaPlayer2.spotify", "/org/mpris/MediaPlayer2"
|
|
|
|
)
|
|
|
|
spotify_iface = dbus.Interface(spotify, "org.freedesktop.DBus.Properties")
|
|
|
|
props = spotify_iface.Get("org.mpris.MediaPlayer2.Player", "Metadata")
|
2020-06-28 21:57:01 +00:00
|
|
|
self.__song = self.__format.format(
|
2020-06-25 17:53:53 +00:00
|
|
|
album=str(props.get("xesam:album")),
|
|
|
|
title=str(props.get("xesam:title")),
|
2020-06-28 21:57:01 +00:00
|
|
|
artist=",".join(props.get("xesam:artist")),
|
|
|
|
trackNumber=str(props.get("xesam:trackNumber")),
|
|
|
|
)
|
2020-05-03 09:15:52 +00:00
|
|
|
|
2020-06-25 17:53:53 +00:00
|
|
|
def update(self):
|
|
|
|
try:
|
|
|
|
self.__get_song()
|
2020-06-28 21:57:01 +00:00
|
|
|
|
2020-09-06 12:00:32 +00:00
|
|
|
for widget in self.widgets():
|
|
|
|
if widget.name == "spotify.pause":
|
2020-06-28 21:57:01 +00:00
|
|
|
playback_status = str(
|
2020-06-30 05:20:54 +00:00
|
|
|
dbus.Interface(
|
2020-09-13 09:16:57 +00:00
|
|
|
self.__bus.get_object(
|
2020-06-30 05:20:54 +00:00
|
|
|
"org.mpris.MediaPlayer2.spotify",
|
|
|
|
"/org/mpris/MediaPlayer2",
|
|
|
|
),
|
|
|
|
"org.freedesktop.DBus.Properties",
|
|
|
|
).Get("org.mpris.MediaPlayer2.Player", "PlaybackStatus")
|
2020-06-28 21:57:01 +00:00
|
|
|
)
|
|
|
|
if playback_status == "Playing":
|
|
|
|
widget.set("state", "playing")
|
|
|
|
else:
|
|
|
|
widget.set("state", "paused")
|
2020-09-15 18:27:50 +00:00
|
|
|
elif widget.name == "spotify.song":
|
|
|
|
widget.set("state", "song")
|
|
|
|
widget.full_text(self.__song)
|
2020-06-25 17:23:45 +00:00
|
|
|
|
2020-09-06 12:00:32 +00:00
|
|
|
except Exception as e:
|
2020-09-06 12:27:17 +00:00
|
|
|
logging.exception(e)
|
2020-05-03 09:15:52 +00:00
|
|
|
self.__song = ""
|
2020-04-25 08:49:02 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def string_song(self):
|
|
|
|
if sys.version_info.major < 3:
|
2020-04-25 08:51:40 +00:00
|
|
|
return unicode(self.__song)
|
|
|
|
return str(self.__song)
|