* Fix pycodestyle errors W503 is condracting PEP-8 and was added to the default ignore list by pycodestyle developers * Use setup.cfg for pycodestyle ignore parameters Addresses #651 (kind of). Appending ignores to the default list prevents us from getting warnings that we don't care about. * W503 is not in the default ignore list in 2.4.0
242 lines
8.0 KiB
Python
242 lines
8.0 KiB
Python
from os.path import basename
|
|
|
|
import dbus
|
|
|
|
from i3pystatus import IntervalModule, formatp
|
|
from i3pystatus.core.util import TimeWrapper
|
|
|
|
|
|
class Dbus:
|
|
obj_dbus = "org.freedesktop.DBus"
|
|
path_dbus = "/org/freedesktop/DBus"
|
|
obj_player = "org.mpris.MediaPlayer2"
|
|
path_player = "/org/mpris/MediaPlayer2"
|
|
intf_props = obj_dbus + ".Properties"
|
|
intf_player = obj_player + ".Player"
|
|
|
|
|
|
class NoPlayerException(Exception):
|
|
pass
|
|
|
|
|
|
class NowPlaying(IntervalModule):
|
|
"""
|
|
Shows currently playing track information. Supports media players that \
|
|
conform to the Media Player Remote Interfacing Specification.
|
|
|
|
* Requires ``python-dbus`` from your distro package manager, or \
|
|
``dbus-python`` from PyPI.
|
|
|
|
Left click on the module to play/pause, and right click to go to the next \
|
|
track.
|
|
|
|
.. rubric:: Available formatters (uses :ref:`formatp`)
|
|
|
|
* `{title}` — (the title of the current song)
|
|
* `{album}` — (the album of the current song, can be an empty string \
|
|
(e.g. for online streams))
|
|
* `{artist}` — (can be empty, too)
|
|
* `{filename}` — (file name with out extension and path; empty unless \
|
|
title is empty)
|
|
* `{song_elapsed}` — (position in the currently playing song, uses \
|
|
:ref:`TimeWrapper`, default is `%m:%S`)
|
|
* `{song_length}` — (length of the current song, same as song_elapsed)
|
|
* `{status}` — (play, pause, stop mapped through the `status` dictionary)
|
|
* `{volume}` — (volume)
|
|
|
|
.. rubric:: Available callbacks
|
|
|
|
* ``playpause`` — Plays if paused or stopped, otherwise pauses.
|
|
* ``next_song`` — Goes to next track in the playlist.
|
|
* ``player_command`` — Invoke a command with the `MediaPlayer2.Player` \
|
|
interface. The method name and its arguments are appended as list elements.
|
|
* ``player_prop`` — Get or set a property of the `MediaPlayer2.Player` \
|
|
interface. Append the property name to get, or the name and a value to set.
|
|
|
|
`MediaPlayer2.Player` methods and properties are documented at \
|
|
https://specifications.freedesktop.org/mpris-spec/latest/Player_Interface.html
|
|
|
|
Your player may not support the full interface.
|
|
|
|
Example module registration with callbacks:
|
|
|
|
::
|
|
|
|
status.register("now_playing",
|
|
on_leftclick=["player_command", "PlayPause"],
|
|
on_rightclick=["player_command", "Stop"],
|
|
on_middleclick=["player_prop", "Shuffle", True],
|
|
on_upscroll=["player_command", "Seek", -10000000],
|
|
on_downscroll=["player_command", "Seek", +10000000])
|
|
|
|
"""
|
|
|
|
interval = 1
|
|
|
|
settings = (
|
|
("player", "Player name. If not set, compatible players will be \
|
|
detected automatically."),
|
|
("status", "Dictionary mapping pause, play and stop to output text"),
|
|
("format", "formatp string"),
|
|
("color", "Text color"),
|
|
("format_no_player", "Text to show if no player is detected"),
|
|
("color_no_player", "Text color when no player is detected"),
|
|
("hide_no_player", "Hide output if no player is detected"),
|
|
)
|
|
|
|
hide_no_player = True
|
|
format_no_player = "No Player"
|
|
color_no_player = "#ffffff"
|
|
|
|
format = "{title} {status}"
|
|
color = "#ffffff"
|
|
status = {
|
|
"pause": "▷",
|
|
"play": "▶",
|
|
"stop": "◾",
|
|
}
|
|
statusmap = {
|
|
"Playing": "play",
|
|
"Paused": "pause",
|
|
"Stopped": "stop",
|
|
}
|
|
|
|
on_leftclick = "playpause"
|
|
on_rightclick = "next_song"
|
|
|
|
player = None
|
|
old_player = None
|
|
|
|
def find_player(self):
|
|
obj = dbus.SessionBus().get_object(Dbus.obj_dbus, Dbus.path_dbus)
|
|
|
|
def get_players(methodname):
|
|
method = obj.get_dbus_method(methodname, Dbus.obj_dbus)
|
|
return [a for a in method() if a.startswith(Dbus.obj_player + ".")]
|
|
|
|
players = get_players('ListNames')
|
|
if not players:
|
|
players = get_players('ListActivatableNames')
|
|
if self.old_player in players:
|
|
return self.old_player
|
|
if not players:
|
|
raise NoPlayerException()
|
|
self.old_player = players[0]
|
|
return players[0]
|
|
|
|
def get_player(self):
|
|
if self.player:
|
|
player = Dbus.obj_player + "." + self.player
|
|
try:
|
|
return dbus.SessionBus().get_object(player, Dbus.path_player)
|
|
except dbus.exceptions.DBusException:
|
|
raise NoPlayerException()
|
|
else:
|
|
player = self.find_player()
|
|
return dbus.SessionBus().get_object(player, Dbus.path_player)
|
|
|
|
def run(self):
|
|
try:
|
|
player = self.get_player()
|
|
properties = dbus.Interface(player, Dbus.intf_props)
|
|
|
|
def get_prop(name, default=None):
|
|
try:
|
|
return properties.Get(Dbus.intf_player, name)
|
|
except dbus.exceptions.DBusException:
|
|
return default
|
|
|
|
currentsong = get_prop("Metadata")
|
|
|
|
fdict = {
|
|
"status": self.status[self.statusmap[
|
|
get_prop("PlaybackStatus")]],
|
|
# TODO: Use optional(!) TrackList interface for this to
|
|
# gain 100 % mpd<->now_playing compat
|
|
"len": 0,
|
|
"pos": 0,
|
|
"volume": int(get_prop("Volume", 0) * 100),
|
|
|
|
"title": currentsong.get("xesam:title", ""),
|
|
"album": currentsong.get("xesam:album", ""),
|
|
"artist": ", ".join(currentsong.get("xesam:artist", "")),
|
|
"song_length": TimeWrapper(
|
|
(currentsong.get("mpris:length") or 0) / 1000 ** 2),
|
|
"song_elapsed": TimeWrapper(
|
|
(get_prop("Position") or 0) / 1000 ** 2),
|
|
"filename": "",
|
|
}
|
|
|
|
if not fdict["title"]:
|
|
fdict["filename"] = '.'.join(
|
|
basename((currentsong.get("xesam:url") or "")).
|
|
split('.')[:-1])
|
|
|
|
self.data = fdict
|
|
self.output = {
|
|
"full_text": formatp(self.format, **fdict).strip(),
|
|
"color": self.color,
|
|
}
|
|
|
|
except NoPlayerException:
|
|
if self.hide_no_player:
|
|
self.output = None
|
|
else:
|
|
self.output = {
|
|
"full_text": self.format_no_player,
|
|
"color": self.color_no_player,
|
|
}
|
|
if hasattr(self, "data"):
|
|
del self.data
|
|
return
|
|
|
|
except dbus.exceptions.DBusException as e:
|
|
if self.hide_no_player:
|
|
self.output = None
|
|
else:
|
|
self.output = {
|
|
"full_text": "DBus error: " + e.get_dbus_message(),
|
|
"color": "#ff0000",
|
|
}
|
|
if hasattr(self, "data"):
|
|
del self.data
|
|
return
|
|
|
|
def playpause(self):
|
|
try:
|
|
dbus.Interface(self.get_player(), Dbus.intf_player).PlayPause()
|
|
except NoPlayerException:
|
|
return
|
|
except dbus.exceptions.DBusException:
|
|
return
|
|
|
|
def next_song(self):
|
|
try:
|
|
dbus.Interface(self.get_player(), Dbus.intf_player).Next()
|
|
except NoPlayerException:
|
|
return
|
|
except dbus.exceptions.DBusException:
|
|
return
|
|
|
|
def player_command(self, command, *args):
|
|
try:
|
|
interface = dbus.Interface(self.get_player(), Dbus.intf_player)
|
|
getattr(interface, command)(*args)
|
|
except NoPlayerException:
|
|
return
|
|
except dbus.exceptions.DBusException:
|
|
return
|
|
|
|
def player_prop(self, name, value=None):
|
|
try:
|
|
properties = dbus.Interface(self.get_player(), Dbus.intf_props)
|
|
# None/null/nil implies get because it's not a valid DBus datatype.
|
|
if value is None:
|
|
return properties.Get(Dbus.intf_player, name)
|
|
else:
|
|
properties.Set(Dbus.intf_player, name, value)
|
|
except NoPlayerException:
|
|
return
|
|
except dbus.exceptions.DBusException:
|
|
return
|