mpdevil/bin/mpdevil

4138 lines
151 KiB
Plaintext
Raw Normal View History

2020-01-19 22:48:49 +03:00
#!/usr/bin/python3
2020-01-11 13:25:15 +03:00
# -*- coding: utf-8 -*-
#
# mpdevil - MPD Client.
2021-01-01 16:11:59 +03:00
# Copyright (C) 2020-2021 Martin Wagner <martin.wagner.dev@gmail.com>
2020-01-11 13:25:15 +03:00
#
2021-01-01 16:11:59 +03:00
# This program is free software: you can redistribute it and/or modify
2020-01-11 13:25:15 +03:00
# it under the terms of the GNU General Public License as published by
2021-01-01 16:11:59 +03:00
# the Free Software Foundation, either version 3 of the License, or
# (at your option) any later version.
2020-01-11 13:25:15 +03:00
#
2021-01-01 16:11:59 +03:00
# This program 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.
2020-01-11 13:25:15 +03:00
#
# You should have received a copy of the GNU General Public License
2021-01-01 16:11:59 +03:00
# along with this program. If not, see <https://www.gnu.org/licenses/>.
2020-01-11 13:25:15 +03:00
2020-03-22 19:05:51 +03:00
import gi
2020-09-24 22:17:10 +03:00
gi.require_version("Gtk", "3.0")
gi.require_version("Notify", "0.7")
2020-03-03 17:59:18 +03:00
from gi.repository import Gtk, Gio, Gdk, GdkPixbuf, Pango, GObject, GLib, Notify
from mpd import MPDClient, base as MPDBase
2020-09-24 17:50:19 +03:00
import requests
from bs4 import BeautifulSoup
import threading
2020-01-11 13:25:15 +03:00
import datetime
import os
import sys
2020-04-07 19:02:43 +03:00
import re
2021-07-03 16:05:56 +03:00
import locale
try:
locale.setlocale(locale.LC_ALL, "")
except locale.Error as e:
print(e)
from gettext import gettext as _, ngettext, textdomain, bindtextdomain
textdomain("mpdevil")
if os.path.isfile("/.flatpak-info"): # test for flatpak environment
bindtextdomain("mpdevil", "/app/share/locale")
else:
bindtextdomain("mpdevil", localedir=None) # replace "None" by a static path if needed (e.g when installing on a non-FHS distro)
2020-01-11 13:25:15 +03:00
2021-05-04 18:29:20 +03:00
VERSION="1.2.1" # sync with setup.py
2020-09-24 22:17:10 +03:00
COVER_REGEX=r"^\.?(album|cover|folder|front).*\.(gif|jpeg|jpg|png)$"
2021-06-02 22:34:01 +03:00
FALLBACK_COVER=Gtk.IconTheme.get_default().lookup_icon("media-optical", 128, Gtk.IconLookupFlags.FORCE_SVG).get_filename()
2020-03-22 19:05:51 +03:00
2020-07-04 14:16:17 +03:00
#########
# MPRIS #
#########
2020-01-11 13:25:15 +03:00
2020-12-24 15:21:08 +03:00
class MPRISInterface: # TODO emit Seeked if needed
"""
2020-12-24 15:21:08 +03:00
based on 'Lollypop' (master 22.12.2020) by Cedric Bellegarde <cedric.bellegarde@adishatz.org>
and 'mpDris2' (master 19.03.2020) by Jean-Philippe Braun <eon@patapon.info>, Mantas Mikulėnas <grawity@gmail.com>
"""
_MPRIS_IFACE="org.mpris.MediaPlayer2"
_MPRIS_PLAYER_IFACE="org.mpris.MediaPlayer2.Player"
_MPRIS_NAME="org.mpris.MediaPlayer2.mpdevil"
_MPRIS_PATH="/org/mpris/MediaPlayer2"
_INTERFACES_XML="""
<!DOCTYPE node PUBLIC
"-//freedesktop//DTD D-BUS Object Introspection 1.0//EN"
"http://www.freedesktop.org/standards/dbus/1.0/introspect.dtd">
<node>
<interface name="org.freedesktop.DBus.Introspectable">
<method name="Introspect">
<arg name="data" direction="out" type="s"/>
</method>
</interface>
<interface name="org.freedesktop.DBus.Properties">
<method name="Get">
<arg name="interface" direction="in" type="s"/>
<arg name="property" direction="in" type="s"/>
<arg name="value" direction="out" type="v"/>
</method>
<method name="Set">
<arg name="interface_name" direction="in" type="s"/>
<arg name="property_name" direction="in" type="s"/>
<arg name="value" direction="in" type="v"/>
</method>
<method name="GetAll">
<arg name="interface" direction="in" type="s"/>
<arg name="properties" direction="out" type="a{sv}"/>
</method>
</interface>
<interface name="org.mpris.MediaPlayer2">
<method name="Raise">
</method>
<method name="Quit">
</method>
<property name="CanQuit" type="b" access="read" />
<property name="CanRaise" type="b" access="read" />
<property name="HasTrackList" type="b" access="read"/>
<property name="Identity" type="s" access="read"/>
<property name="DesktopEntry" type="s" access="read"/>
<property name="SupportedUriSchemes" type="as" access="read"/>
<property name="SupportedMimeTypes" type="as" access="read"/>
</interface>
<interface name="org.mpris.MediaPlayer2.Player">
<method name="Next"/>
<method name="Previous"/>
<method name="Pause"/>
<method name="PlayPause"/>
<method name="Stop"/>
<method name="Play"/>
<method name="Seek">
<arg direction="in" name="Offset" type="x"/>
</method>
<method name="SetPosition">
<arg direction="in" name="TrackId" type="o"/>
<arg direction="in" name="Position" type="x"/>
</method>
<method name="OpenUri">
<arg direction="in" name="Uri" type="s"/>
</method>
<signal name="Seeked">
<arg name="Position" type="x"/>
</signal>
<property name="PlaybackStatus" type="s" access="read"/>
<property name="LoopStatus" type="s" access="readwrite"/>
<property name="Rate" type="d" access="readwrite"/>
<property name="Shuffle" type="b" access="readwrite"/>
<property name="Metadata" type="a{sv}" access="read"/>
<property name="Volume" type="d" access="readwrite"/>
<property name="Position" type="x" access="read"/>
<property name="MinimumRate" type="d" access="read"/>
<property name="MaximumRate" type="d" access="read"/>
<property name="CanGoNext" type="b" access="read"/>
<property name="CanGoPrevious" type="b" access="read"/>
<property name="CanPlay" type="b" access="read"/>
<property name="CanPause" type="b" access="read"/>
<property name="CanSeek" type="b" access="read"/>
<property name="CanControl" type="b" access="read"/>
</interface>
</node>
"""
2020-12-24 15:21:08 +03:00
def __init__(self, window, client, settings):
self._window=window
self._client=client
self._settings=settings
self._metadata={}
2020-12-24 15:21:08 +03:00
# MPRIS property mappings
self._prop_mapping={
self._MPRIS_IFACE:
{"CanQuit": (GLib.Variant("b", False), None),
"CanRaise": (GLib.Variant("b", True), None),
"HasTrackList": (GLib.Variant("b", False), None),
"Identity": (GLib.Variant("s", "mpdevil"), None),
"DesktopEntry": (GLib.Variant("s", "org.mpdevil.mpdevil"), None),
2020-12-24 15:21:08 +03:00
"SupportedUriSchemes": (GLib.Variant("s", "None"), None),
"SupportedMimeTypes": (GLib.Variant("s", "None"), None)},
self._MPRIS_PLAYER_IFACE:
{"PlaybackStatus": (self._get_playback_status, None),
"LoopStatus": (self._get_loop_status, self._set_loop_status),
"Rate": (GLib.Variant("d", 1.0), None),
"Shuffle": (self._get_shuffle, self._set_shuffle),
"Metadata": (self._get_metadata, None),
"Volume": (self._get_volume, self._set_volume),
"Position": (self._get_position, None),
"MinimumRate": (GLib.Variant("d", 1.0), None),
"MaximumRate": (GLib.Variant("d", 1.0), None),
"CanGoNext": (self._get_can_next_prev, None),
"CanGoPrevious": (self._get_can_next_prev, None),
"CanPlay": (self._get_can_play_pause_seek, None),
"CanPause": (self._get_can_play_pause_seek, None),
"CanSeek": (self._get_can_play_pause_seek, None),
"CanControl": (GLib.Variant("b", True), None)},
}
2020-03-30 18:08:59 +03:00
2020-09-25 18:13:39 +03:00
# start
2021-07-18 23:39:35 +03:00
self._bus=Gio.bus_get_sync(Gio.BusType.SESSION, None)
2020-12-24 15:21:08 +03:00
Gio.bus_own_name_on_connection(self._bus, self._MPRIS_NAME, Gio.BusNameOwnerFlags.NONE, None, None)
self._node_info=Gio.DBusNodeInfo.new_for_xml(self._INTERFACES_XML)
for interface in self._node_info.interfaces:
self._bus.register_object(self._MPRIS_PATH, interface, self._handle_method_call, None, None)
2020-09-25 18:13:39 +03:00
2020-07-04 14:16:17 +03:00
# connect
self._client.emitter.connect("state", self._on_state_changed)
self._client.emitter.connect("current_song_changed", self._on_song_changed)
self._client.emitter.connect("volume_changed", self._on_volume_changed)
self._client.emitter.connect("repeat", self._on_loop_changed)
self._client.emitter.connect("single", self._on_loop_changed)
self._client.emitter.connect("random", self._on_random_changed)
2020-09-25 18:13:39 +03:00
self._client.emitter.connect("connection_error", self._on_connection_error)
self._client.emitter.connect("reconnected", self._on_reconnected)
2020-12-24 15:21:08 +03:00
def _handle_method_call(self, connection, sender, object_path, interface_name, method_name, parameters, invocation):
args=list(parameters.unpack())
result=getattr(self, method_name)(*args)
out_args=self._node_info.lookup_interface(interface_name).lookup_method(method_name).out_args
if out_args == []:
invocation.return_value(None)
else:
signature="("+"".join([arg.signature for arg in out_args])+")"
variant=GLib.Variant(signature, (result,))
invocation.return_value(variant)
# setter and getter
def _get_playback_status(self):
2020-09-25 18:13:39 +03:00
if self._client.connected():
status=self._client.status()
2020-12-24 15:21:08 +03:00
return GLib.Variant("s", {"play": "Playing", "pause": "Paused", "stop": "Stopped"}[status["state"]])
return GLib.Variant("s", "Stopped")
2020-12-24 15:21:08 +03:00
def _set_loop_status(self, value):
2020-09-25 18:13:39 +03:00
if self._client.connected():
if value == "Playlist":
self._client.repeat(1)
self._client.single(0)
2020-09-25 18:13:39 +03:00
elif value == "Track":
self._client.repeat(1)
self._client.single(1)
2020-09-25 18:13:39 +03:00
elif value == "None":
self._client.repeat(0)
self._client.single(0)
2020-12-24 15:21:08 +03:00
def _get_loop_status(self):
2020-09-25 18:13:39 +03:00
if self._client.connected():
status=self._client.status()
2020-12-24 15:47:28 +03:00
if status["repeat"] == "1":
if status.get("single", "0") == "0":
2020-12-24 15:21:08 +03:00
return GLib.Variant("s", "Playlist")
2020-12-24 15:47:28 +03:00
else:
return GLib.Variant("s", "Track")
2020-07-04 14:16:17 +03:00
else:
2020-12-24 15:21:08 +03:00
return GLib.Variant("s", "None")
return GLib.Variant("s", "None")
2020-07-04 14:16:17 +03:00
2020-12-24 15:21:08 +03:00
def _set_shuffle(self, value):
2020-09-25 18:13:39 +03:00
if self._client.connected():
2020-12-24 15:21:08 +03:00
if value:
self._client.random("1")
2020-12-24 15:21:08 +03:00
else:
self._client.random("0")
2020-07-04 14:16:17 +03:00
2020-12-24 15:21:08 +03:00
def _get_shuffle(self):
2020-09-25 18:13:39 +03:00
if self._client.connected():
if self._client.status()["random"] == "1":
2020-12-24 15:21:08 +03:00
return GLib.Variant("b", True)
2020-09-25 18:13:39 +03:00
else:
2020-12-24 15:21:08 +03:00
return GLib.Variant("b", False)
return GLib.Variant("b", False)
2020-12-24 15:21:08 +03:00
def _get_metadata(self):
return GLib.Variant("a{sv}", self._metadata)
2020-03-30 12:54:04 +03:00
2020-12-24 15:21:08 +03:00
def _get_volume(self):
2020-09-25 18:13:39 +03:00
if self._client.connected():
return GLib.Variant("d", float(self._client.status().get("volume", 0))/100)
2020-12-24 15:21:08 +03:00
return GLib.Variant("d", 0)
2020-12-24 15:21:08 +03:00
def _set_volume(self, value):
2020-09-25 18:13:39 +03:00
if self._client.connected():
if value >= 0 and value <= 1:
self._client.setvol(int(value * 100))
2020-03-30 12:54:04 +03:00
2020-12-24 15:21:08 +03:00
def _get_position(self):
2020-09-25 18:13:39 +03:00
if self._client.connected():
status=self._client.status()
2020-12-24 15:21:08 +03:00
return GLib.Variant("x", float(status.get("elapsed", 0))*1000000)
return GLib.Variant("x", 0)
2020-12-24 15:21:08 +03:00
def _get_can_next_prev(self):
2020-09-25 18:13:39 +03:00
if self._client.connected():
status=self._client.status()
2020-09-25 18:13:39 +03:00
if status["state"] == "stop":
2020-12-24 15:21:08 +03:00
return GLib.Variant("b", False)
2020-09-25 18:13:39 +03:00
else:
2020-12-24 15:21:08 +03:00
return GLib.Variant("b", True)
return GLib.Variant("b", False)
2020-03-21 00:09:13 +03:00
2020-12-24 15:21:08 +03:00
def _get_can_play_pause_seek(self):
return GLib.Variant("b", self._client.connected())
# introspect methods
def Introspect(self):
return self._INTERFACES_XML
# property methods
def Get(self, interface_name, prop):
getter, setter=self._prop_mapping[interface_name][prop]
2020-07-04 14:16:17 +03:00
if callable(getter):
2020-12-24 15:21:08 +03:00
return getter()
2020-07-04 14:16:17 +03:00
return getter
2020-03-21 00:09:13 +03:00
2020-12-24 15:21:08 +03:00
def Set(self, interface_name, prop, value):
getter, setter=self._prop_mapping[interface_name][prop]
2020-07-04 14:16:17 +03:00
if setter is not None:
2020-12-24 15:21:08 +03:00
setter(value)
2020-03-21 00:09:13 +03:00
2020-12-24 15:21:08 +03:00
def GetAll(self, interface_name):
2020-07-04 14:16:17 +03:00
read_props={}
2020-12-24 15:21:08 +03:00
try:
props=self._prop_mapping[interface_name]
for key, (getter, setter) in props.items():
if callable(getter):
getter=getter()
read_props[key]=getter
except KeyError: # interface has no properties
pass
2020-07-04 14:16:17 +03:00
return read_props
2020-12-24 15:21:08 +03:00
def PropertiesChanged(self, interface_name, changed_properties, invalidated_properties):
self._bus.emit_signal(
None, self._MPRIS_PATH, "org.freedesktop.DBus.Properties", "PropertiesChanged",
GLib.Variant.new_tuple(
GLib.Variant("s", interface_name),
GLib.Variant("a{sv}", changed_properties),
GLib.Variant("as", invalidated_properties)
)
)
# root methods
2020-07-04 14:16:17 +03:00
def Raise(self):
self._window.present()
2020-07-04 14:16:17 +03:00
def Quit(self):
2020-12-24 15:21:08 +03:00
app_action_group=self._window.get_action_group("app")
quit_action=app_action_group.lookup_action("quit")
quit_action.activate()
2020-07-04 14:16:17 +03:00
2020-12-24 15:21:08 +03:00
# player methods
2020-07-04 14:16:17 +03:00
def Next(self):
self._client.next()
2020-07-04 14:16:17 +03:00
def Previous(self):
2021-06-12 23:37:37 +03:00
self._client.conditional_previous()
2020-07-04 14:16:17 +03:00
def Pause(self):
self._client.pause(1)
2020-07-04 14:16:17 +03:00
def PlayPause(self):
self._client.toggle_play()
2020-07-04 14:16:17 +03:00
def Stop(self):
self._client.stop()
2020-07-04 14:16:17 +03:00
def Play(self):
self._client.play()
2020-07-04 14:16:17 +03:00
2020-09-25 13:24:35 +03:00
def Seek(self, offset):
if offset > 0:
offset="+"+str(offset/1000000)
else:
offset=str(offset/1000000)
self._client.seekcur(offset)
2020-07-04 14:16:17 +03:00
def SetPosition(self, trackid, position):
song=self._client.currentsong()
2020-09-25 13:24:35 +03:00
if str(trackid).split("/")[-1] != song["id"]:
2020-07-04 14:16:17 +03:00
return
2020-09-25 13:24:35 +03:00
mpd_pos=position/1000000
if mpd_pos >= 0 and mpd_pos <= float(song["duration"]):
self._client.seekcur(str(mpd_pos))
2020-07-04 14:16:17 +03:00
2020-09-25 13:24:35 +03:00
def OpenUri(self, uri):
2020-12-24 15:21:08 +03:00
pass
2020-09-25 13:24:35 +03:00
2020-12-24 15:21:08 +03:00
def Seeked(self, position):
self._bus.emit_signal(
None, self._MPRIS_PATH, self._MPRIS_PLAYER_IFACE, "Seeked",
GLib.Variant.new_tuple(GLib.Variant("x", position))
)
# other methods
2020-09-25 13:24:35 +03:00
def _update_metadata(self):
"""
Translate metadata returned by MPD to the MPRIS v2 syntax.
http://www.freedesktop.org/wiki/Specifications/mpris-spec/metadata
"""
mpd_meta=self._client.currentsong() # raw values needed for cover
2020-10-17 12:30:00 +03:00
song=ClientHelper.song_to_list_dict(mpd_meta)
2020-09-26 14:29:31 +03:00
self._metadata={}
2020-12-24 15:21:08 +03:00
for tag, xesam_tag in (("album","album"),("title","title"),("date","contentCreated")):
2020-10-17 12:30:00 +03:00
if tag in song:
2020-12-24 15:21:08 +03:00
self._metadata["xesam:{}".format(xesam_tag)]=GLib.Variant("s", song[tag][0])
for tag, xesam_tag in (("track","trackNumber"),("disc","discNumber")):
if tag in song:
self._metadata["xesam:{}".format(xesam_tag)]=GLib.Variant("i", int(song[tag][0]))
2020-09-25 13:24:35 +03:00
for tag, xesam_tag in (("albumartist","albumArtist"),("artist","artist"),("composer","composer"),("genre","genre")):
2020-10-17 12:30:00 +03:00
if tag in song:
2020-12-24 15:21:08 +03:00
self._metadata["xesam:{}".format(xesam_tag)]=GLib.Variant("as", song[tag])
2020-10-17 12:30:00 +03:00
if "id" in song:
2020-12-24 15:21:08 +03:00
self._metadata["mpris:trackid"]=GLib.Variant("o", self._MPRIS_PATH+"/Track/{}".format(song["id"][0]))
if "duration" in song:
2020-12-24 15:21:08 +03:00
self._metadata["mpris:length"]=GLib.Variant("x", float(song["duration"][0])*1000000)
2020-10-17 12:30:00 +03:00
if "file" in song:
song_file=song["file"][0]
if "://" in song_file: # remote file
self._metadata["xesam:url"]=GLib.Variant("s", song_file)
else:
lib_path=self._settings.get_value("paths")[self._settings.get_int("active-profile")]
self._metadata["xesam:url"]=GLib.Variant("s", "file://{}".format(os.path.join(lib_path, song_file)))
cover_path=self._client.get_cover_path(mpd_meta)
if cover_path is not None:
self._metadata["mpris:artUrl"]=GLib.Variant("s", "file://{}".format(cover_path))
2020-12-24 15:21:08 +03:00
def _update_property(self, interface_name, prop):
getter, setter=self._prop_mapping[interface_name][prop]
2020-09-25 13:24:35 +03:00
if callable(getter):
2020-12-24 15:21:08 +03:00
value=getter()
2020-09-25 13:24:35 +03:00
else:
value=getter
2020-12-24 15:21:08 +03:00
self.PropertiesChanged(interface_name, {prop: value}, [])
2020-09-25 13:24:35 +03:00
return value
2020-08-21 19:03:36 +03:00
def _on_state_changed(self, *args):
2020-12-24 15:21:08 +03:00
self._update_property(self._MPRIS_PLAYER_IFACE, "PlaybackStatus")
self._update_property(self._MPRIS_PLAYER_IFACE, "CanGoNext")
self._update_property(self._MPRIS_PLAYER_IFACE, "CanGoPrevious")
2020-08-21 19:03:36 +03:00
def _on_song_changed(self, *args):
2020-09-25 13:24:35 +03:00
self._update_metadata()
2020-12-24 15:21:08 +03:00
self._update_property(self._MPRIS_PLAYER_IFACE, "Metadata")
2020-08-21 19:03:36 +03:00
def _on_volume_changed(self, *args):
2020-12-24 15:21:08 +03:00
self._update_property(self._MPRIS_PLAYER_IFACE, "Volume")
2020-08-21 19:03:36 +03:00
def _on_loop_changed(self, *args):
2020-12-24 15:21:08 +03:00
self._update_property(self._MPRIS_PLAYER_IFACE, "LoopStatus")
2020-08-21 19:03:36 +03:00
def _on_random_changed(self, *args):
2020-12-24 15:21:08 +03:00
self._update_property(self._MPRIS_PLAYER_IFACE, "Shuffle")
2020-08-21 19:03:36 +03:00
def _on_reconnected(self, *args):
2020-09-25 18:13:39 +03:00
properties=("CanPlay","CanPause","CanSeek")
for p in properties:
2020-12-24 15:21:08 +03:00
self._update_property(self._MPRIS_PLAYER_IFACE, p)
2020-09-25 18:13:39 +03:00
def _on_connection_error(self, *args):
self._metadata={}
2020-09-25 18:13:39 +03:00
properties=("PlaybackStatus","CanGoNext","CanGoPrevious","Metadata","Volume","LoopStatus","Shuffle","CanPlay","CanPause","CanSeek")
for p in properties:
2020-12-24 15:21:08 +03:00
self._update_property(self._MPRIS_PLAYER_IFACE, p)
2020-07-04 14:16:17 +03:00
######################
# MPD client wrapper #
######################
2020-03-21 00:09:13 +03:00
2020-07-04 14:16:17 +03:00
class ClientHelper():
2020-09-30 12:06:00 +03:00
def seconds_to_display_time(seconds):
delta=datetime.timedelta(seconds=seconds)
if delta.days > 0:
days=ngettext("{days} day", "{days} days", delta.days).format(days=delta.days)
time_string=days+", "+str(datetime.timedelta(seconds=delta.seconds))
else:
time_string=str(delta).lstrip("0").lstrip(":")
return time_string.replace(":", "") # use 'ratio' as delimiter
2020-09-30 12:06:00 +03:00
def convert_audio_format(audio_format):
# see: https://www.musicpd.org/doc/html/user.html#audio-output-format
samplerate, bits, channels=audio_format.split(":")
if bits == "f":
bits="32fp"
try:
int_chan=int(channels)
except:
int_chan=0
try:
freq=locale.str(int(samplerate)/1000)
except:
freq=samplerate
channels=ngettext("{channels} channel", "{channels} channels", int_chan).format(channels=channels)
return "{} kHz • {} bit • {}".format(freq, bits, channels)
2020-07-04 14:16:17 +03:00
def song_to_str_dict(song): # converts tags with multiple values to comma separated strings
2020-10-17 12:30:00 +03:00
return_song={}
for tag, value in song.items():
2021-07-22 20:42:49 +03:00
if isinstance(value, list):
2020-09-24 22:17:10 +03:00
return_song[tag]=(", ".join(value))
2020-10-17 12:30:00 +03:00
else:
return_song[tag]=value
2020-07-04 14:16:17 +03:00
return return_song
2020-03-21 00:09:13 +03:00
2020-07-04 14:16:17 +03:00
def song_to_first_str_dict(song): # extracts the first value of multiple value tags
2020-10-17 12:30:00 +03:00
return_song={}
for tag, value in song.items():
2021-07-22 20:42:49 +03:00
if isinstance(value, list):
2020-07-04 14:16:17 +03:00
return_song[tag]=value[0]
2020-10-17 12:30:00 +03:00
else:
return_song[tag]=value
return return_song
def song_to_list_dict(song): # converts all values to lists
return_song={}
for tag, value in song.items():
2021-07-22 20:42:49 +03:00
if isinstance(value, list):
2020-10-17 12:30:00 +03:00
return_song[tag]=value
2021-07-22 20:42:49 +03:00
else:
return_song[tag]=[value]
2020-07-04 14:16:17 +03:00
return return_song
2020-03-21 00:09:13 +03:00
2020-10-17 12:30:00 +03:00
def pepare_song_for_display(song):
2020-09-12 15:31:17 +03:00
base_song={
"title": _("Unknown Title"),
"track": "",
2020-09-12 15:31:17 +03:00
"disc": "",
"artist": "",
"album": "",
2020-09-12 15:31:17 +03:00
"duration": "0.0",
"date": "",
"genre": ""
}
2020-10-02 09:15:34 +03:00
if "range" in song: # translate .cue 'range' to 'duration' if needed
start, end=song["range"].split("-")
if start != "" and end != "":
base_song["duration"]=str((float(end)-float(start)))
2020-07-04 14:16:17 +03:00
base_song.update(song)
2020-09-30 12:06:00 +03:00
base_song["human_duration"]=ClientHelper.seconds_to_display_time(int(float(base_song["duration"])))
2020-10-17 12:30:00 +03:00
for tag in ("disc", "track"): # remove confusing multiple tags
if tag in song:
2021-07-22 20:42:49 +03:00
if isinstance(song[tag], list):
2020-10-17 12:30:00 +03:00
base_song[tag]=song[tag][0]
2020-07-04 14:16:17 +03:00
return base_song
2020-03-21 00:09:13 +03:00
2021-07-16 20:46:39 +03:00
def calc_display_duration(songs):
duration=sum([float(song.get("duration", 0.0)) for song in songs])
return ClientHelper.seconds_to_display_time(int(duration))
2020-03-21 00:09:13 +03:00
def binary_to_pixbuf(binary, size):
2021-07-19 23:41:32 +03:00
loader=GdkPixbuf.PixbufLoader()
2021-06-02 22:34:01 +03:00
try:
loader.write(binary)
loader.close()
raw_pixbuf=loader.get_pixbuf()
ratio=raw_pixbuf.get_width()/raw_pixbuf.get_height()
if ratio > 1:
pixbuf=raw_pixbuf.scale_simple(size,size/ratio,GdkPixbuf.InterpType.BILINEAR)
else:
pixbuf=raw_pixbuf.scale_simple(size*ratio,size,GdkPixbuf.InterpType.BILINEAR)
except gi.repository.GLib.Error: # load fallback if cover can't be loaded
pixbuf=GdkPixbuf.Pixbuf.new_from_file_at_size(FALLBACK_COVER, size, size)
return pixbuf
def file_to_pixbuf(file, size):
try:
pixbuf=GdkPixbuf.Pixbuf.new_from_file_at_size(file, size, size)
except gi.repository.GLib.Error: # load fallback if cover can't be loaded
pixbuf=GdkPixbuf.Pixbuf.new_from_file_at_size(FALLBACK_COVER, size, size)
return pixbuf
2021-03-26 18:39:21 +03:00
class EventEmitter(GObject.Object):
2020-07-04 14:16:17 +03:00
__gsignals__={
2020-09-24 22:17:10 +03:00
"update": (GObject.SignalFlags.RUN_FIRST, None, ()),
"disconnected": (GObject.SignalFlags.RUN_FIRST, None, ()),
"reconnected": (GObject.SignalFlags.RUN_FIRST, None, ()),
"connection_error": (GObject.SignalFlags.RUN_FIRST, None, ()),
"current_song_changed": (GObject.SignalFlags.RUN_FIRST, None, ()),
"state": (GObject.SignalFlags.RUN_FIRST, None, (str,)),
"elapsed_changed": (GObject.SignalFlags.RUN_FIRST, None, (float,float,)),
"volume_changed": (GObject.SignalFlags.RUN_FIRST, None, (float,)),
"playlist_changed": (GObject.SignalFlags.RUN_FIRST, None, (int,)),
"repeat": (GObject.SignalFlags.RUN_FIRST, None, (bool,)),
"random": (GObject.SignalFlags.RUN_FIRST, None, (bool,)),
2020-11-01 15:27:23 +03:00
"single": (GObject.SignalFlags.RUN_FIRST, None, (str,)),
2020-09-24 22:17:10 +03:00
"consume": (GObject.SignalFlags.RUN_FIRST, None, (bool,)),
"audio": (GObject.SignalFlags.RUN_FIRST, None, (str,)),
"bitrate": (GObject.SignalFlags.RUN_FIRST, None, (str,)),
2021-03-26 18:39:21 +03:00
"add_to_playlist": (GObject.SignalFlags.RUN_FIRST, None, (str,)),
"show_info": (GObject.SignalFlags.RUN_FIRST, None, ())
2020-03-21 00:09:13 +03:00
}
2020-07-04 14:16:17 +03:00
def __init__(self):
super().__init__()
2020-03-21 00:09:13 +03:00
2020-07-04 14:16:17 +03:00
class Client(MPDClient):
def __init__(self, settings):
2020-08-31 11:44:23 +03:00
super().__init__()
self._settings=settings
2021-03-26 18:39:21 +03:00
self.emitter=EventEmitter()
self._last_status={}
self._refresh_interval=self._settings.get_int("refresh-interval")
self._main_timeout_id=None
2020-03-21 00:09:13 +03:00
2020-10-13 18:12:54 +03:00
# connect
self._settings.connect("changed::active-profile", self._on_active_profile_changed)
2020-08-15 12:56:44 +03:00
2020-07-04 14:16:17 +03:00
def start(self):
self.emitter.emit("disconnected") # bring player in defined state
active=self._settings.get_int("active-profile")
try:
self.connect(self._settings.get_value("hosts")[active], self._settings.get_value("ports")[active])
if self._settings.get_value("passwords")[active] != "":
self.password(self._settings.get_value("passwords")[active])
except:
self.emitter.emit("connection_error")
return False
# connect successful
2020-12-22 14:33:21 +03:00
if "status" in self.commands():
self._main_timeout_id=GLib.timeout_add(self._refresh_interval, self._main_loop)
self.emitter.emit("reconnected")
return True
else:
self.disconnect()
self.emitter.emit("connection_error")
print("No read permission, check your mpd config.")
return False
def reconnect(self):
if self._main_timeout_id is not None:
GLib.source_remove(self._main_timeout_id)
self._main_timeout_id=None
self._last_status={}
self.disconnect()
self.start()
2020-03-21 00:09:13 +03:00
2020-07-04 14:16:17 +03:00
def connected(self):
try:
self.ping()
2020-07-04 14:16:17 +03:00
return True
except:
return False
2020-03-21 00:09:13 +03:00
2020-08-16 18:07:41 +03:00
def files_to_playlist(self, files, mode="default"): # modes: default, play, append, enqueue
def append(files):
2020-07-04 14:16:17 +03:00
for f in files:
self.add(f)
2020-08-16 18:07:41 +03:00
def play(files):
2020-08-21 13:03:42 +03:00
if files != []:
2020-08-16 18:07:41 +03:00
self.clear()
for f in files:
self.add(f)
self.play()
def enqueue(files):
status=self.status()
if status["state"] == "stop":
play(files)
2020-07-04 14:16:17 +03:00
else:
self.moveid(status["songid"], 0)
current_song_file=self.playlistinfo()[0]["file"]
try:
self.delete((1,)) # delete all songs, but the first. bad song index possible
except:
pass
for f in files:
2020-08-21 13:03:42 +03:00
if f == current_song_file:
2020-07-04 14:16:17 +03:00
self.move(0, (len(self.playlistinfo())-1))
2020-08-21 13:03:42 +03:00
else:
self.add(f)
2020-08-16 18:07:41 +03:00
if mode == "append":
append(files)
elif mode == "enqueue":
enqueue(files)
elif mode == "play":
play(files)
elif mode == "default":
if self._settings.get_boolean("force-mode"):
2020-08-16 18:07:41 +03:00
play(files)
else:
enqueue(files)
2020-03-21 00:09:13 +03:00
2021-04-06 21:59:25 +03:00
def album_to_playlist(self, album, artist, year, genre, mode="default"):
if genre is None:
genre_filter=()
else:
genre_filter=("genre", genre)
songs=self.find("album", album, "date", year, self._settings.get_artist_type(), artist, *genre_filter)
2020-09-24 22:17:10 +03:00
self.files_to_playlist([song["file"] for song in songs], mode)
2020-03-21 00:09:13 +03:00
2021-03-26 18:39:21 +03:00
def artist_to_playlist(self, artist, genre, mode):
def append():
if self._settings.get_boolean("sort-albums-by-year"):
sort_tag="date"
else:
2021-03-26 18:39:21 +03:00
sort_tag="album"
if artist is None: # treat 'None' as 'all artists'
2021-04-06 21:59:25 +03:00
if genre is None:
2021-03-26 18:39:21 +03:00
self.searchadd("any", "", "sort", sort_tag)
else:
self.findadd("genre", genre, "sort", sort_tag)
else:
2021-03-26 18:39:21 +03:00
artist_type=self._settings.get_artist_type()
2021-04-06 21:59:25 +03:00
if genre is None:
2021-03-26 18:39:21 +03:00
self.findadd(artist_type, artist, "sort", sort_tag)
else:
self.findadd(artist_type, artist, "genre", genre, "sort", sort_tag)
if mode == "append":
append()
elif mode == "play":
self.clear()
append()
self.play()
elif mode == "enqueue":
status=self.status()
if status["state"] == "stop":
self.clear()
append()
self.play()
else:
self.moveid(status["songid"], 0)
current_song_file=self.currentsong()["file"]
try:
self.delete((1,)) # delete all songs, but the first. bad song index possible
except:
pass
append()
duplicates=self.playlistfind("file", current_song_file)
if len(duplicates) > 1:
self.move(0, duplicates[1]["pos"])
self.delete(int(duplicates[1]["pos"])-1)
2020-07-04 14:16:17 +03:00
def comp_list(self, *args): # simulates listing behavior of python-mpd2 1.0
native_list=self.list(*args)
if len(native_list) > 0:
2021-07-22 20:42:49 +03:00
if isinstance(native_list[0], dict):
2020-07-04 14:16:17 +03:00
return ([l[args[0]] for l in native_list])
else:
return native_list
else:
return([])
def get_cover_path(self, raw_song):
path=None
song=ClientHelper.song_to_first_str_dict(raw_song)
song_file=song.get("file")
active_profile=self._settings.get_int("active-profile")
lib_path=self._settings.get_lib_path()
if lib_path is not None:
regex_str=self._settings.get_value("regex")[active_profile]
if regex_str == "":
regex=re.compile(COVER_REGEX, flags=re.IGNORECASE)
else:
regex_str=regex_str.replace("%AlbumArtist%", song.get("albumartist", ""))
regex_str=regex_str.replace("%Album%", song.get("album", ""))
try:
regex=re.compile(regex_str, flags=re.IGNORECASE)
except:
print("illegal regex:", regex_str)
return (None, None)
if song_file is not None:
song_dir=os.path.join(lib_path, os.path.dirname(song_file))
if song_dir.endswith(".cue"):
song_dir=os.path.dirname(song_dir) # get actual directory of .cue file
if os.path.exists(song_dir):
for f in os.listdir(song_dir):
if regex.match(f):
path=os.path.join(song_dir, f)
break
return path
def get_cover_binary(self, uri):
if uri is None:
binary=None
else:
try:
binary=self.albumart(uri)["binary"]
except:
try:
binary=self.readpicture(uri)["binary"]
except:
binary=None
return binary
def get_cover(self, song, size):
cover_path=self.get_cover_path(song)
if cover_path is None:
cover_binary=self.get_cover_binary(song.get("file"))
if cover_binary is None:
2021-06-02 22:34:01 +03:00
pixbuf=GdkPixbuf.Pixbuf.new_from_file_at_size(FALLBACK_COVER, size, size)
else:
pixbuf=ClientHelper.binary_to_pixbuf(cover_binary, size)
else:
2021-06-02 22:34:01 +03:00
pixbuf=ClientHelper.file_to_pixbuf(cover_path, size)
return pixbuf
def get_metadata(self, uri):
meta_base=self.lsinfo(uri)[0]
2020-10-02 09:15:34 +03:00
try: # .cue files produce an error here
meta_extra=self.readcomments(uri) # contains comment tag
meta_base.update(meta_extra)
except:
pass
return meta_base
def get_absolute_path(self, uri):
lib_path=self._settings.get_lib_path()
if lib_path is not None:
path=os.path.join(lib_path, uri)
if os.path.isfile(path):
return path
else:
return None
else:
return None
def get_albums(self, artist, genre):
albums=[]
artist_type=self._settings.get_artist_type()
if genre is None:
2021-04-06 21:59:25 +03:00
genre_filter=()
else:
2021-04-06 21:59:25 +03:00
genre_filter=("genre", genre)
album_candidates=self.comp_list("album", artist_type, artist, *genre_filter)
for album in album_candidates:
2021-07-22 20:25:55 +03:00
years=self.comp_list("date", "album", album, artist_type, artist, *genre_filter)
for year in years:
2021-04-06 21:59:25 +03:00
songs=self.find("album", album, "date", year, artist_type, artist, *genre_filter)
cover_path=self.get_cover_path(songs[0])
if cover_path is None:
cover_binary=self.get_cover_binary(songs[0].get("file"))
if cover_binary is None:
albums.append({"artist": artist, "album": album, "year": year, "songs": songs})
else:
albums.append({"artist":artist,"album":album,"year":year,"songs":songs,"cover_binary":cover_binary})
else:
albums.append({"artist": artist, "album": album, "year": year, "songs": songs, "cover_path": cover_path})
return albums
2020-09-29 13:39:21 +03:00
def toggle_play(self):
status=self.status()
if status["state"] == "play":
self.pause(1)
elif status["state"] == "pause":
self.pause(0)
else:
try:
self.play()
except:
pass
2020-09-29 14:02:51 +03:00
def toggle_option(self, option): # repeat, random, single, consume
2020-11-01 15:27:23 +03:00
state=self.status()[option]
if state != "1" and state != "0": # support single oneshot
state="1"
new_state=(int(state)+1)%2 # toggle 0,1
2020-09-29 13:39:21 +03:00
func=getattr(self, option)
func(new_state)
def conditional_previous(self):
if self._settings.get_boolean("rewind-mode"):
double_click_time=Gtk.Settings.get_default().get_property("gtk-double-click-time")
status=self.status()
if float(status.get("elapsed", 0))*1000 > double_click_time:
self.seekcur(0)
else:
self.previous()
else:
self.previous()
def _main_loop(self, *args):
2020-07-04 14:16:17 +03:00
try:
status=self.status()
diff=set(status.items())-set(self._last_status.items())
2020-07-12 16:20:29 +03:00
for key, val in diff:
if key == "elapsed":
if "duration" in status:
self.emitter.emit("elapsed_changed", float(val), float(status["duration"]))
else:
self.emitter.emit("elapsed_changed", 0.0, 0.0)
elif key == "bitrate":
if val == "0":
self.emitter.emit("bitrate", None)
else:
self.emitter.emit("bitrate", val)
2020-07-12 16:20:29 +03:00
elif key == "songid":
self.emitter.emit("current_song_changed")
elif key in ("state", "single", "audio"):
2020-11-01 15:27:23 +03:00
self.emitter.emit(key, val)
2020-07-12 16:20:29 +03:00
elif key == "volume":
self.emitter.emit("volume_changed", float(val))
2020-07-12 16:20:29 +03:00
elif key == "playlist":
self.emitter.emit("playlist_changed", int(val))
2021-03-27 14:50:45 +03:00
elif key in ("repeat", "random", "consume"):
if val == "1":
self.emitter.emit(key, True)
else:
self.emitter.emit(key, False)
diff=set(self._last_status)-set(status)
2021-07-16 20:39:48 +03:00
for key in diff:
if "songid" == key:
self.emitter.emit("current_song_changed")
elif "volume" == key:
self.emitter.emit("volume_changed", -1)
elif "updating_db" == key:
self.emitter.emit("update")
elif "bitrate" == key:
self.emitter.emit("bitrate", None)
2021-07-16 20:39:48 +03:00
elif "audio" == key:
self.emitter.emit("audio", None)
self._last_status=status
2020-08-15 12:56:44 +03:00
except (MPDBase.ConnectionError, ConnectionResetError) as e:
2020-08-14 23:08:56 +03:00
self.disconnect()
self._last_status={}
2020-07-04 14:16:17 +03:00
self.emitter.emit("disconnected")
self.emitter.emit("connection_error")
self._main_timeout_id=None
2020-07-04 14:16:17 +03:00
return False
return True
2020-03-21 00:09:13 +03:00
def _on_active_profile_changed(self, *args):
self.reconnect()
2020-08-21 19:03:36 +03:00
2020-07-04 14:16:17 +03:00
########################
# gio settings wrapper #
########################
2020-03-21 00:09:13 +03:00
2020-01-28 20:39:18 +03:00
class Settings(Gio.Settings):
BASE_KEY="org.mpdevil.mpdevil"
2020-09-15 19:45:30 +03:00
# temp settings
mini_player=GObject.Property(type=bool, default=False)
cursor_watch=GObject.Property(type=bool, default=False)
2020-01-28 20:39:18 +03:00
def __init__(self):
2020-01-28 21:59:14 +03:00
super().__init__(schema=self.BASE_KEY)
2020-08-04 20:37:54 +03:00
# fix profile settings
2020-02-07 22:13:38 +03:00
if len(self.get_value("profiles")) < (self.get_int("active-profile")+1):
self.set_int("active-profile", 0)
2020-09-12 15:31:17 +03:00
profile_keys=[
2020-09-24 22:17:10 +03:00
("as", "profiles", "new profile"),
("as", "hosts", "localhost"),
("ai", "ports", 6600),
("as", "passwords", ""),
("as", "paths", ""),
("as", "regex", "")
2020-09-12 15:31:17 +03:00
]
2020-08-04 20:37:54 +03:00
profile_arrays=[]
for vtype, key, default in profile_keys:
profile_arrays.append(self.get_value(key).unpack())
max_len=max(len(x) for x in profile_arrays)
for index, (vtype, key, default) in enumerate(profile_keys):
profile_arrays[index]=(profile_arrays[index]+max_len*[default])[:max_len]
self.set_value(key, GLib.Variant(vtype, profile_arrays[index]))
2020-01-28 20:39:18 +03:00
def array_append(self, vtype, key, value): # append to Gio.Settings (self._settings) array
2020-01-28 20:39:18 +03:00
array=self.get_value(key).unpack()
array.append(value)
self.set_value(key, GLib.Variant(vtype, array))
def array_delete(self, vtype, key, pos): # delete entry of Gio.Settings (self._settings) array
2020-01-28 20:39:18 +03:00
array=self.get_value(key).unpack()
array.pop(pos)
self.set_value(key, GLib.Variant(vtype, array))
def array_modify(self, vtype, key, pos, value): # modify entry of Gio.Settings (self._settings) array
2020-01-28 20:39:18 +03:00
array=self.get_value(key).unpack()
array[pos]=value
self.set_value(key, GLib.Variant(vtype, array))
2020-02-26 01:37:27 +03:00
def get_artist_type(self):
if self.get_boolean("use-album-artist"):
2020-02-26 01:37:27 +03:00
return ("albumartist")
else:
return ("artist")
2020-02-26 01:37:27 +03:00
def get_lib_path(self, profile=None):
if profile is None: # use current profile if none is given
profile=self.get_int("active-profile")
lib_path=self.get_value("paths")[profile]
if lib_path == "":
lib_path=GLib.get_user_special_dir(GLib.UserDirectory.DIRECTORY_MUSIC)
return lib_path
2020-09-16 16:08:56 +03:00
###################
# settings dialog #
###################
2020-05-26 16:04:16 +03:00
2020-09-16 16:08:56 +03:00
class GeneralSettings(Gtk.Box):
def __init__(self, settings):
super().__init__(orientation=Gtk.Orientation.VERTICAL, spacing=6, border_width=18)
self._settings=settings
2020-07-04 14:16:17 +03:00
2020-10-13 18:12:54 +03:00
# int settings
2020-09-16 16:08:56 +03:00
int_settings={}
int_settings_data=[
(_("Main cover size:"), (100, 1200, 10), "track-cover"),
(_("Album view cover size:"), (50, 600, 10), "album-cover"),
(_("Action bar icon size:"), (16, 64, 2), "icon-size"),
]
for label, (vmin, vmax, step), key in int_settings_data:
int_settings[key]=(Gtk.Label(label=label, xalign=0), Gtk.SpinButton.new_with_range(vmin, vmax, step))
int_settings[key][1].set_value(self._settings.get_int(key))
2021-07-17 14:58:19 +03:00
self._settings.bind(key, int_settings[key][1], "value", Gio.SettingsBindFlags.DEFAULT)
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
# check buttons
check_buttons={}
check_buttons_data=[
2021-02-13 13:56:16 +03:00
(_("Use Client-side decoration"), "use-csd"),
(_("Show stop button"), "show-stop"),
2021-07-16 21:07:52 +03:00
(_("Show audio format"), "show-audio-format"),
2021-02-13 13:56:16 +03:00
(_("Show lyrics button"), "show-lyrics-button"),
(_("Place playlist at the side"), "playlist-right"),
(_("Use “Album Artist” tag"), "use-album-artist"),
(_("Send notification on title change"), "send-notify"),
(_("Stop playback on quit"), "stop-on-quit"),
(_("Play selected albums and titles immediately"), "force-mode"),
(_("Sort albums by year"), "sort-albums-by-year"),
2021-04-20 18:48:03 +03:00
(_("Support “MPRIS”"), "mpris"),
(_("Rewind via previous button"), "rewind-mode"),
2020-09-16 16:08:56 +03:00
]
2021-02-13 13:56:16 +03:00
for label, key in check_buttons_data:
2021-04-23 16:30:08 +03:00
check_buttons[key]=Gtk.CheckButton(label=label, margin_start=12)
2020-09-16 16:08:56 +03:00
check_buttons[key].set_active(self._settings.get_boolean(key))
2021-07-17 14:58:19 +03:00
self._settings.bind(key, check_buttons[key], "active", Gio.SettingsBindFlags.DEFAULT)
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
# headings
view_heading=Gtk.Label(label=_("<b>View</b>"), use_markup=True, xalign=0)
behavior_heading=Gtk.Label(label=_("<b>Behavior</b>"), use_markup=True, xalign=0)
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
# view grid
2021-04-23 16:30:08 +03:00
view_grid=Gtk.Grid(row_spacing=6, column_spacing=12, margin_start=12)
2020-09-16 16:08:56 +03:00
view_grid.add(int_settings["track-cover"][0])
view_grid.attach_next_to(int_settings["album-cover"][0], int_settings["track-cover"][0], Gtk.PositionType.BOTTOM, 1, 1)
view_grid.attach_next_to(int_settings["icon-size"][0], int_settings["album-cover"][0], Gtk.PositionType.BOTTOM, 1, 1)
view_grid.attach_next_to(int_settings["track-cover"][1], int_settings["track-cover"][0], Gtk.PositionType.RIGHT, 1, 1)
view_grid.attach_next_to(int_settings["album-cover"][1], int_settings["album-cover"][0], Gtk.PositionType.RIGHT, 1, 1)
view_grid.attach_next_to(int_settings["icon-size"][1], int_settings["icon-size"][0], Gtk.PositionType.RIGHT, 1, 1)
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
# packing
2021-04-17 14:22:07 +03:00
csd_box=Gtk.Box(spacing=12)
csd_box.pack_start(check_buttons["use-csd"], False, False, 0)
csd_box.pack_start(Gtk.Label(label=_("(restart required)"), sensitive=False), False, False, 0)
2020-09-16 16:08:56 +03:00
self.pack_start(view_heading, False, False, 0)
2021-04-17 14:22:07 +03:00
self.pack_start(csd_box, False, False, 0)
2020-09-16 16:08:56 +03:00
self.pack_start(check_buttons["show-stop"], False, False, 0)
2021-07-16 21:07:52 +03:00
self.pack_start(check_buttons["show-audio-format"], False, False, 0)
2020-09-16 16:08:56 +03:00
self.pack_start(check_buttons["show-lyrics-button"], False, False, 0)
self.pack_start(check_buttons["playlist-right"], False, False, 0)
2020-09-16 16:08:56 +03:00
self.pack_start(view_grid, False, False, 0)
self.pack_start(behavior_heading, False, False, 0)
2021-04-18 18:23:12 +03:00
mpris_box=Gtk.Box(spacing=12)
mpris_box.pack_start(check_buttons["mpris"], False, False, 0)
mpris_box.pack_start(Gtk.Label(label=_("(restart required)"), sensitive=False), False, False, 0)
self.pack_start(mpris_box, False, False, 0)
2020-09-16 16:08:56 +03:00
self.pack_start(check_buttons["use-album-artist"], False, False, 0)
self.pack_start(check_buttons["sort-albums-by-year"], False, False, 0)
2020-09-16 16:08:56 +03:00
self.pack_start(check_buttons["send-notify"], False, False, 0)
self.pack_start(check_buttons["force-mode"], False, False, 0)
self.pack_start(check_buttons["rewind-mode"], False, False, 0)
self.pack_start(check_buttons["stop-on-quit"], False, False, 0)
2020-09-16 16:08:56 +03:00
class ProfileSettings(Gtk.Grid):
def __init__(self, parent, client, settings):
super().__init__(row_spacing=6, column_spacing=12, border_width=18)
self._client=client
self._settings=settings
2020-09-16 16:08:56 +03:00
# widgets
2021-04-23 00:40:51 +03:00
self._profiles_combo=ComboBoxEntry()
2021-01-01 14:44:04 +03:00
add_button=Gtk.Button(image=Gtk.Image.new_from_icon_name("list-add-symbolic", Gtk.IconSize.BUTTON))
delete_button=Gtk.Button(image=Gtk.Image.new_from_icon_name("list-remove-symbolic", Gtk.IconSize.BUTTON))
2020-09-16 16:08:56 +03:00
add_delete_buttons=Gtk.ButtonBox(layout_style=Gtk.ButtonBoxStyle.EXPAND)
add_delete_buttons.pack_start(add_button, True, True, 0)
add_delete_buttons.pack_start(delete_button, True, True, 0)
2021-01-01 14:44:04 +03:00
connect_button=Gtk.Button.new_with_mnemonic(_("_Connect"))
2020-09-16 16:08:56 +03:00
self._host_entry=Gtk.Entry(hexpand=True)
self._port_entry=Gtk.SpinButton.new_with_range(0, 65535, 1)
address_entry=Gtk.Box(spacing=6)
address_entry.pack_start(self._host_entry, True, True, 0)
address_entry.pack_start(self._port_entry, False, False, 0)
2020-12-29 19:44:20 +03:00
self._password_entry=PasswordEntry(hexpand=True)
self._path_entry=Gtk.Entry(hexpand=True, placeholder_text=GLib.get_user_special_dir(GLib.UserDirectory.DIRECTORY_MUSIC))
2021-04-23 00:40:51 +03:00
self._path_entry.set_icon_from_icon_name(Gtk.EntryIconPosition.SECONDARY, "folder-open-symbolic")
2020-09-16 16:08:56 +03:00
self._regex_entry=Gtk.Entry(hexpand=True, placeholder_text=COVER_REGEX)
self._regex_entry.set_tooltip_text(
_("The first image in the same directory as the song file "\
"matching this regex will be displayed. %AlbumArtist% and "\
"%Album% will be replaced by the corresponding tags of the song.")
)
profiles_label=Gtk.Label(label=_("Profile:"), xalign=1)
host_label=Gtk.Label(label=_("Host:"), xalign=1)
password_label=Gtk.Label(label=_("Password:"), xalign=1)
path_label=Gtk.Label(label=_("Music lib:"), xalign=1)
regex_label=Gtk.Label(label=_("Cover regex:"), xalign=1)
2020-01-18 00:13:58 +03:00
2020-07-04 13:35:39 +03:00
# connect
2020-09-16 16:08:56 +03:00
add_button.connect("clicked", self._on_add_button_clicked)
delete_button.connect("clicked", self._on_delete_button_clicked)
connect_button.connect("clicked", self._on_connect_button_clicked)
2020-12-29 19:44:20 +03:00
style_context=connect_button.get_style_context()
style_context.add_class("suggested-action")
2021-04-23 00:40:51 +03:00
self._path_entry.connect("icon-release", self._on_path_entry_icon_release, parent)
self._profiles_select=self._profiles_combo.connect("select", self._on_profiles_select)
2020-09-16 16:08:56 +03:00
self.entry_changed_handlers=[]
2021-04-23 00:40:51 +03:00
self.entry_changed_handlers.append((self._profiles_combo, self._profiles_combo.connect("text", self._on_profile_entry_changed)))
2020-09-16 16:08:56 +03:00
self.entry_changed_handlers.append((self._host_entry, self._host_entry.connect("changed", self._on_host_entry_changed)))
self.entry_changed_handlers.append((self._port_entry, self._port_entry.connect("value-changed", self._on_port_entry_changed)))
self.entry_changed_handlers.append((self._password_entry, self._password_entry.connect("changed", self._on_password_entry_changed)))
self.entry_changed_handlers.append((self._path_entry, self._path_entry.connect("changed", self._on_path_entry_changed)))
self.entry_changed_handlers.append((self._regex_entry, self._regex_entry.connect("changed", self._on_regex_entry_changed)))
2020-01-18 00:13:58 +03:00
2020-09-16 16:08:56 +03:00
# packing
self.add(profiles_label)
2021-04-23 00:40:51 +03:00
self.attach_next_to(host_label, profiles_label, Gtk.PositionType.BOTTOM, 1, 1)
2020-09-16 16:08:56 +03:00
self.attach_next_to(password_label, host_label, Gtk.PositionType.BOTTOM, 1, 1)
self.attach_next_to(path_label, password_label, Gtk.PositionType.BOTTOM, 1, 1)
self.attach_next_to(regex_label, path_label, Gtk.PositionType.BOTTOM, 1, 1)
self.attach_next_to(self._profiles_combo, profiles_label, Gtk.PositionType.RIGHT, 2, 1)
self.attach_next_to(add_delete_buttons, self._profiles_combo, Gtk.PositionType.RIGHT, 1, 1)
self.attach_next_to(address_entry, host_label, Gtk.PositionType.RIGHT, 2, 1)
self.attach_next_to(self._password_entry, password_label, Gtk.PositionType.RIGHT, 2, 1)
2021-04-23 00:40:51 +03:00
self.attach_next_to(self._path_entry, path_label, Gtk.PositionType.RIGHT, 2, 1)
2020-09-16 16:08:56 +03:00
self.attach_next_to(self._regex_entry, regex_label, Gtk.PositionType.RIGHT, 2, 1)
connect_button.set_margin_top(12)
self.attach_next_to(connect_button, self._regex_entry, Gtk.PositionType.BOTTOM, 2, 1)
2020-01-18 00:13:58 +03:00
2021-04-23 16:30:08 +03:00
self._profiles_combo_reload()
self._profiles_combo.set_active(self._settings.get_int("active-profile"))
2020-09-16 16:08:56 +03:00
def _block_entry_changed_handlers(self, *args):
for obj, handler in self.entry_changed_handlers:
obj.handler_block(handler)
2020-09-16 16:08:56 +03:00
def _unblock_entry_changed_handlers(self, *args):
for obj, handler in self.entry_changed_handlers:
obj.handler_unblock(handler)
2020-09-16 16:08:56 +03:00
def _profiles_combo_reload(self, *args):
2021-04-23 00:40:51 +03:00
self._profiles_combo.handler_block(self._profiles_select)
2020-09-16 16:08:56 +03:00
self._profiles_combo.remove_all()
for profile in self._settings.get_value("profiles"):
self._profiles_combo.append_text(profile)
2021-04-23 00:40:51 +03:00
self._profiles_combo.handler_unblock(self._profiles_select)
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
def _on_add_button_clicked(self, *args):
model=self._profiles_combo.get_model()
2020-09-24 22:17:10 +03:00
self._settings.array_append("as", "profiles", "new profile ({})".format(len(model)))
self._settings.array_append("as", "hosts", "localhost")
self._settings.array_append("ai", "ports", 6600)
self._settings.array_append("as", "passwords", "")
self._settings.array_append("as", "paths", "")
self._settings.array_append("as", "regex", "")
2020-09-16 16:08:56 +03:00
self._profiles_combo_reload()
new_pos=len(model)-1
self._profiles_combo.set_active(new_pos)
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
def _on_delete_button_clicked(self, *args):
2021-04-23 00:40:51 +03:00
pos=self._profiles_combo.get_selected()
2020-09-24 22:17:10 +03:00
self._settings.array_delete("as", "profiles", pos)
self._settings.array_delete("as", "hosts", pos)
self._settings.array_delete("ai", "ports", pos)
self._settings.array_delete("as", "passwords", pos)
self._settings.array_delete("as", "paths", pos)
self._settings.array_delete("as", "regex", pos)
2020-09-16 16:08:56 +03:00
if len(self._settings.get_value("profiles")) == 0:
self._on_add_button_clicked()
else:
self._profiles_combo_reload()
new_pos=max(pos-1,0)
self._profiles_combo.set_active(new_pos)
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
def _on_connect_button_clicked(self, *args):
2021-04-23 17:17:37 +03:00
selected=self._profiles_combo.get_selected()
if selected == self._settings.get_int("active-profile"):
self._client.reconnect()
else:
self._settings.set_int("active-profile", selected)
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
def _on_profile_entry_changed(self, *args):
2021-04-23 00:40:51 +03:00
self._settings.array_modify("as", "profiles", self._profiles_combo.get_selected(), self._profiles_combo.get_text())
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
def _on_host_entry_changed(self, *args):
2021-04-23 00:40:51 +03:00
self._settings.array_modify("as", "hosts", self._profiles_combo.get_selected(), self._host_entry.get_text())
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
def _on_port_entry_changed(self, *args):
2021-04-23 00:40:51 +03:00
self._settings.array_modify("ai", "ports", self._profiles_combo.get_selected(), int(self._port_entry.get_value()))
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
def _on_password_entry_changed(self, *args):
2021-04-23 00:40:51 +03:00
self._settings.array_modify("as", "passwords", self._profiles_combo.get_selected(), self._password_entry.get_text())
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
def _on_path_entry_changed(self, *args):
2021-04-23 00:40:51 +03:00
self._settings.array_modify("as", "paths", self._profiles_combo.get_selected(), self._path_entry.get_text())
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
def _on_regex_entry_changed(self, *args):
2021-04-23 00:40:51 +03:00
self._settings.array_modify("as", "regex", self._profiles_combo.get_selected(), self._regex_entry.get_text())
2020-08-16 18:07:41 +03:00
2021-04-23 00:40:51 +03:00
def _on_path_entry_icon_release(self, widget, icon_pos, event, parent):
dialog=Gtk.FileChooserNative(title=_("Choose directory"), transient_for=parent, action=Gtk.FileChooserAction.SELECT_FOLDER)
2021-04-23 00:40:51 +03:00
folder=self._settings.get_lib_path(self._profiles_combo.get_selected())
if folder is not None:
dialog.set_current_folder(folder)
2020-09-16 16:08:56 +03:00
response=dialog.run()
if response == Gtk.ResponseType.ACCEPT:
2021-04-23 00:40:51 +03:00
self._settings.array_modify("as", "paths", self._profiles_combo.get_selected(), dialog.get_filename())
2020-09-16 16:08:56 +03:00
self._path_entry.set_text(dialog.get_filename())
dialog.destroy()
2020-08-16 18:07:41 +03:00
2021-04-23 00:40:51 +03:00
def _on_profiles_select(self, *args):
2020-09-16 16:08:56 +03:00
active=self._profiles_combo.get_active()
2020-10-13 17:44:02 +03:00
if active >= 0:
self._block_entry_changed_handlers()
self._host_entry.set_text(self._settings.get_value("hosts")[active])
self._port_entry.set_value(self._settings.get_value("ports")[active])
self._password_entry.set_text(self._settings.get_value("passwords")[active])
self._path_entry.set_text(self._settings.get_value("paths")[active])
self._regex_entry.set_text(self._settings.get_value("regex")[active])
self._unblock_entry_changed_handlers()
2020-09-16 16:08:56 +03:00
class PlaylistSettings(Gtk.Box):
def __init__(self, settings):
super().__init__(orientation=Gtk.Orientation.VERTICAL, spacing=6, border_width=18)
self._settings=settings
2020-09-16 16:08:56 +03:00
# label
label=Gtk.Label(label=_("Choose the order of information to appear in the playlist:"), wrap=True, xalign=0)
2020-10-13 18:12:54 +03:00
# treeview
2020-09-16 16:08:56 +03:00
# (toggle, header, actual_index)
self._store=Gtk.ListStore(bool, str, int)
2020-10-13 18:12:54 +03:00
treeview=Gtk.TreeView(model=self._store, reorderable=True, headers_visible=False, search_column=-1)
2020-09-16 16:08:56 +03:00
self._selection=treeview.get_selection()
2020-08-16 18:07:41 +03:00
2020-10-13 18:12:54 +03:00
# columns
2020-09-16 16:08:56 +03:00
renderer_text=Gtk.CellRendererText()
renderer_toggle=Gtk.CellRendererToggle()
column_toggle=Gtk.TreeViewColumn("", renderer_toggle, active=0)
treeview.append_column(column_toggle)
column_text=Gtk.TreeViewColumn("", renderer_text, text=1)
treeview.append_column(column_text)
2020-09-16 16:08:56 +03:00
# fill store
self._headers=[_("No"), _("Disc"), _("Title"), _("Artist"), _("Album"), _("Length"), _("Year"), _("Genre")]
self._fill()
2020-09-16 16:08:56 +03:00
# scroll
scroll=Gtk.ScrolledWindow()
scroll.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
scroll.add(treeview)
2020-10-13 18:12:54 +03:00
# toolbar
2020-10-17 01:03:18 +03:00
toolbar=Gtk.Toolbar(icon_size=Gtk.IconSize.SMALL_TOOLBAR)
2020-09-16 16:08:56 +03:00
style_context=toolbar.get_style_context()
style_context.add_class("inline-toolbar")
2020-10-17 01:03:18 +03:00
self._up_button=Gtk.ToolButton(icon_name="go-up-symbolic")
2020-09-16 16:08:56 +03:00
self._up_button.set_sensitive(False)
2020-10-17 01:03:18 +03:00
self._down_button=Gtk.ToolButton(icon_name="go-down-symbolic")
2020-09-16 16:08:56 +03:00
self._down_button.set_sensitive(False)
toolbar.insert(self._up_button, 0)
toolbar.insert(self._down_button, 1)
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
# column chooser
2020-10-13 18:12:54 +03:00
frame=Gtk.Frame()
frame.add(scroll)
2020-09-16 16:08:56 +03:00
column_chooser=Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
column_chooser.pack_start(frame, True, True, 0)
column_chooser.pack_start(toolbar, False, False, 0)
2020-08-16 18:07:41 +03:00
# connect
2020-09-16 16:08:56 +03:00
self._row_deleted=self._store.connect("row-deleted", self._save_permutation)
renderer_toggle.connect("toggled", self._on_cell_toggled)
self._up_button.connect("clicked", self._on_up_button_clicked)
self._down_button.connect("clicked", self._on_down_button_clicked)
self._selection.connect("changed", self._set_button_sensitivity)
2020-07-04 13:35:39 +03:00
# packing
2020-09-16 16:08:56 +03:00
self.pack_start(label, False, False, 0)
self.pack_start(column_chooser, True, True, 0)
2020-08-16 18:07:41 +03:00
2020-09-16 16:08:56 +03:00
def _fill(self, *args):
visibilities=self._settings.get_value("column-visibilities").unpack()
for actual_index in self._settings.get_value("column-permutation"):
self._store.append([visibilities[actual_index], self._headers[actual_index], actual_index])
2020-03-30 23:20:14 +03:00
2020-09-16 16:08:56 +03:00
def _save_permutation(self, *args):
permutation=[]
for row in self._store:
permutation.append(row[2])
self._settings.set_value("column-permutation", GLib.Variant("ai", permutation))
2020-09-16 15:51:28 +03:00
2020-09-16 16:08:56 +03:00
def _set_button_sensitivity(self, *args):
treeiter=self._selection.get_selected()[1]
if treeiter is None:
self._up_button.set_sensitive(False)
self._down_button.set_sensitive(False)
else:
path=self._store.get_path(treeiter)
if self._store.iter_next(treeiter) is None:
self._up_button.set_sensitive(True)
self._down_button.set_sensitive(False)
elif not path.prev():
self._up_button.set_sensitive(False)
self._down_button.set_sensitive(True)
else:
self._up_button.set_sensitive(True)
self._down_button.set_sensitive(True)
2020-02-27 22:05:48 +03:00
2020-09-16 16:08:56 +03:00
def _on_cell_toggled(self, widget, path):
self._store[path][0]=not self._store[path][0]
2020-09-24 22:17:10 +03:00
self._settings.array_modify("ab", "column-visibilities", self._store[path][2], self._store[path][0])
2020-02-27 22:05:48 +03:00
2020-09-16 16:08:56 +03:00
def _on_up_button_clicked(self, *args):
treeiter=self._selection.get_selected()[1]
path=self._store.get_path(treeiter)
path.prev()
prev=self._store.get_iter(path)
self._store.move_before(treeiter, prev)
self._set_button_sensitivity()
self._save_permutation()
2020-02-29 01:23:53 +03:00
2020-09-16 16:08:56 +03:00
def _on_down_button_clicked(self, *args):
treeiter=self._selection.get_selected()[1]
path=self._store.get_path(treeiter)
next=self._store.iter_next(treeiter)
self._store.move_after(treeiter, next)
self._set_button_sensitivity()
self._save_permutation()
2020-09-16 16:08:56 +03:00
class SettingsDialog(Gtk.Dialog):
def __init__(self, parent, client, settings, tab="general"):
2020-09-16 16:08:56 +03:00
use_csd=settings.get_boolean("use-csd")
if use_csd:
super().__init__(title=_("Settings"), transient_for=parent, use_header_bar=True)
else:
super().__init__(title=_("Settings"), transient_for=parent)
self.add_button(Gtk.STOCK_OK, Gtk.ResponseType.OK)
self.set_default_size(500, 400)
2020-02-27 22:05:48 +03:00
2020-09-16 16:08:56 +03:00
# widgets
general=GeneralSettings(settings)
profiles=ProfileSettings(parent, client, settings)
playlist=PlaylistSettings(settings)
2020-08-09 23:58:48 +03:00
2020-09-16 16:08:56 +03:00
# packing
vbox=self.get_content_area()
if use_csd:
stack=Gtk.Stack()
stack.add_titled(general, "general", _("General"))
stack.add_titled(profiles, "profiles", _("Profiles"))
stack.add_titled(playlist, "playlist", _("Playlist"))
stack_switcher=Gtk.StackSwitcher(stack=stack)
vbox.pack_start(stack, True, True, 0)
header_bar=self.get_header_bar()
header_bar.set_custom_title(stack_switcher)
else:
tabs=Gtk.Notebook()
tabs.append_page(general, Gtk.Label(label=_("General")))
tabs.append_page(profiles, Gtk.Label(label=_("Profiles")))
tabs.append_page(playlist, Gtk.Label(label=_("Playlist")))
vbox.set_property("spacing", 6)
vbox.set_property("border-width", 6)
vbox.pack_start(tabs, True, True, 0)
2020-09-16 16:08:56 +03:00
self.show_all()
if use_csd:
stack.set_visible_child_name(tab)
else:
tabs.set_current_page({"general": 0, "profiles": 1, "playlist": 2}[tab])
2020-09-16 15:51:28 +03:00
2020-09-16 16:08:56 +03:00
#################
# other dialogs #
#################
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
class ServerStats(Gtk.Dialog):
def __init__(self, parent, client, settings):
use_csd=settings.get_boolean("use-csd")
2020-10-17 01:21:33 +03:00
super().__init__(title=_("Stats"), transient_for=parent, use_header_bar=use_csd)
2020-10-20 19:14:20 +03:00
if not use_csd:
self.add_button(Gtk.STOCK_OK, Gtk.ResponseType.OK)
2020-09-16 16:08:56 +03:00
self.set_resizable(False)
2020-01-11 13:25:15 +03:00
2020-10-20 19:14:20 +03:00
# grid
grid=Gtk.Grid(row_spacing=6, column_spacing=12, border_width=6)
2020-03-31 18:36:41 +03:00
2020-10-13 18:12:54 +03:00
# populate
2020-10-20 19:14:20 +03:00
display_str={
"protocol": _("<b>Protocol:</b>"),
"uptime": _("<b>Uptime:</b>"),
"playtime": _("<b>Playtime:</b>"),
"artists": _("<b>Artists:</b>"),
"albums": _("<b>Albums:</b>"),
"songs": _("<b>Songs:</b>"),
"db_playtime": _("<b>Total Playtime:</b>"),
"db_update": _("<b>Database Update:</b>")
}
stats=client.stats()
2020-10-20 19:14:20 +03:00
stats["protocol"]=str(client.mpd_version)
for key in ("uptime","playtime","db_playtime"):
stats[key]=ClientHelper.seconds_to_display_time(int(stats[key]))
stats["db_update"]=str(datetime.datetime.fromtimestamp(int(stats["db_update"]))).replace(":", "")
for i, key in enumerate(("protocol","uptime","playtime","db_update","db_playtime","artists","albums","songs")):
grid.attach(Gtk.Label(label=display_str[key], use_markup=True, xalign=1), 0, i, 1, 1)
grid.attach(Gtk.Label(label=stats[key], xalign=0), 1, i, 1, 1)
2020-10-13 18:12:54 +03:00
# packing
vbox=self.get_content_area()
2020-10-20 19:14:20 +03:00
vbox.set_property("border-width", 6)
vbox.pack_start(grid, True, True, 0)
2020-09-16 16:08:56 +03:00
self.show_all()
self.run()
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
class AboutDialog(Gtk.AboutDialog):
def __init__(self, window):
super().__init__(transient_for=window, modal=True)
self.set_program_name("mpdevil")
self.set_version(VERSION)
self.set_comments(_("A simple music browser for MPD"))
self.set_authors(["Martin Wagner"])
self.set_translator_credits("Martin de Reuver\nMartin Wagner")
2020-09-16 16:08:56 +03:00
self.set_website("https://github.com/SoongNoonien/mpdevil")
2021-01-01 15:19:05 +03:00
self.set_copyright("Copyright \xa9 2020-2021 Martin Wagner")
2021-01-01 16:11:59 +03:00
self.set_license_type(Gtk.License.GPL_3_0)
self.set_logo_icon_name("org.mpdevil.mpdevil")
2020-01-11 13:25:15 +03:00
2020-12-06 02:45:25 +03:00
###########################
# general purpose widgets #
###########################
2020-10-21 12:14:17 +03:00
class AutoSizedIcon(Gtk.Image):
def __init__(self, icon_name, settings_key, settings):
2021-06-26 00:21:06 +03:00
super().__init__(icon_name=icon_name)
settings.bind(settings_key, self, "pixel-size", Gio.SettingsBindFlags.GET)
2020-10-20 14:13:50 +03:00
2020-12-29 19:44:20 +03:00
class PasswordEntry(Gtk.Entry):
def __init__(self, **kwargs):
super().__init__(visibility=False, caps_lock_warning=False, input_purpose=Gtk.InputPurpose.PASSWORD, **kwargs)
self.set_icon_from_icon_name(Gtk.EntryIconPosition.SECONDARY, "view-conceal-symbolic")
# connect
self.connect("icon-release", self._on_icon_release)
def _on_icon_release(self, *args):
if self.get_icon_name(Gtk.EntryIconPosition.SECONDARY) == "view-conceal-symbolic":
self.set_visibility(True)
self.set_icon_from_icon_name(Gtk.EntryIconPosition.SECONDARY, "view-reveal-symbolic")
else:
self.set_visibility(False)
self.set_icon_from_icon_name(Gtk.EntryIconPosition.SECONDARY, "view-conceal-symbolic")
2021-04-23 00:40:51 +03:00
class ComboBoxEntry(Gtk.ComboBoxText):
__gsignals__={"text": (GObject.SignalFlags.RUN_FIRST, None, ()), "select": (GObject.SignalFlags.RUN_FIRST, None, ())}
def __init__(self):
super().__init__(entry_text_column=0, has_entry=True)
self._store=self.get_property("model")
self._entry=self.get_child()
self._selected=-1
# connect
self.connect("changed", self._on_changed)
def get_text(self):
return self._entry.get_text()
def get_selected(self):
return self._selected
def _on_changed(self, *args):
active=self.get_active()
if active >= 0:
self._selected=active
self.emit("select")
else:
try:
self._store[self._selected][0]=self._entry.get_text()
self.emit("text")
except:
pass
2020-09-16 16:08:56 +03:00
class FocusFrame(Gtk.Overlay):
def __init__(self):
super().__init__()
2020-10-21 22:09:07 +03:00
self._frame=Gtk.Frame(no_show_all=True)
2020-09-16 16:08:56 +03:00
# css
style_context=self._frame.get_style_context()
provider=Gtk.CssProvider()
2020-12-11 00:39:46 +03:00
css=b"""* {border-color: @theme_selected_bg_color; border-width: 0px; border-top-width: 3px;}"""
2020-09-16 16:08:56 +03:00
provider.load_from_data(css)
2020-12-09 21:12:12 +03:00
style_context.add_provider(provider, 600)
2020-09-16 16:08:56 +03:00
self.add_overlay(self._frame)
self.set_overlay_pass_through(self._frame, True)
2020-09-11 00:07:00 +03:00
def disable(self):
self._frame.hide()
def enable(self):
if self._widget.has_focus():
self._frame.show()
2020-09-16 16:08:56 +03:00
def set_widget(self, widget):
self._widget=widget
2020-10-21 22:09:07 +03:00
self._widget.connect("focus-in-event", lambda *args: self._frame.show())
self._widget.connect("focus-out-event", lambda *args: self._frame.hide())
2020-01-11 13:25:15 +03:00
2021-06-24 22:24:22 +03:00
class ScrolledFocusFrame(FocusFrame):
def __init__(self, widget):
super().__init__()
self.set_widget(widget)
# scroll
scroll=Gtk.ScrolledWindow()
scroll.add(widget)
self.add(scroll)
2020-09-16 16:08:56 +03:00
class SongPopover(Gtk.Popover):
def __init__(self, client, show_buttons=True):
2020-09-16 16:08:56 +03:00
super().__init__()
2021-03-24 19:35:05 +03:00
self._client=client
self._rect=Gdk.Rectangle()
self._uri=None
box=Gtk.Box(orientation=Gtk.Orientation.VERTICAL, border_width=6, spacing=6)
2020-12-31 19:28:33 +03:00
2021-03-24 19:35:05 +03:00
# open-with button
2021-01-01 17:37:17 +03:00
open_button=Gtk.Button(image=Gtk.Image.new_from_icon_name("document-open-symbolic",Gtk.IconSize.BUTTON),tooltip_text=_("Open with…"))
open_button.set_margin_bottom(6)
open_button.set_margin_end(6)
2021-01-01 17:37:17 +03:00
style_context=open_button.get_style_context()
style_context.add_class("circular")
2021-03-24 19:35:05 +03:00
# open button revealer
2021-03-24 19:35:05 +03:00
self._open_button_revealer=Gtk.Revealer()
self._open_button_revealer.set_halign(Gtk.Align.END)
self._open_button_revealer.set_valign(Gtk.Align.END)
self._open_button_revealer.add(open_button)
2021-01-01 17:37:17 +03:00
# buttons
if show_buttons:
button_box=Gtk.ButtonBox(layout_style=Gtk.ButtonBoxStyle.EXPAND)
data=((_("Append"), "list-add-symbolic", "append"),
(_("Play"), "media-playback-start-symbolic", "play"),
(_("Enqueue"), "insert-object-symbolic", "enqueue")
)
for label, icon, mode in data:
button=Gtk.Button(label=label, image=Gtk.Image.new_from_icon_name(icon, Gtk.IconSize.BUTTON))
button.connect("clicked", self._on_button_clicked, mode)
button_box.pack_start(button, True, True, 0)
box.pack_end(button_box, False, False, 0)
2020-10-13 18:12:54 +03:00
# treeview
2020-09-16 16:08:56 +03:00
# (tag, display-value, tooltip)
2021-03-24 19:35:05 +03:00
self._store=Gtk.ListStore(str, str, str)
self._treeview=Gtk.TreeView(model=self._store, headers_visible=False, search_column=-1, tooltip_column=2)
self._treeview.set_can_focus(False)
self._treeview.get_selection().set_mode(Gtk.SelectionMode.NONE)
2020-01-11 13:25:15 +03:00
2020-10-13 18:12:54 +03:00
# columns
2020-09-16 16:08:56 +03:00
renderer_text=Gtk.CellRendererText(width_chars=50, ellipsize=Pango.EllipsizeMode.MIDDLE, ellipsize_set=True)
renderer_text_ralign=Gtk.CellRendererText(xalign=1.0, weight=Pango.Weight.BOLD)
2020-09-16 16:08:56 +03:00
column_tag=Gtk.TreeViewColumn(_("MPD-Tag"), renderer_text_ralign, text=0)
column_tag.set_property("resizable", False)
2021-03-24 19:35:05 +03:00
self._treeview.append_column(column_tag)
2020-09-16 16:08:56 +03:00
column_value=Gtk.TreeViewColumn(_("Value"), renderer_text, text=1)
column_value.set_property("resizable", False)
2021-03-24 19:35:05 +03:00
self._treeview.append_column(column_value)
2021-03-24 19:35:05 +03:00
# scroll
self._scroll=Gtk.ScrolledWindow()
2021-03-24 19:35:05 +03:00
self._scroll.set_policy(Gtk.PolicyType.NEVER, Gtk.PolicyType.AUTOMATIC)
self._scroll.set_propagate_natural_height(True)
self._scroll.add(self._treeview)
# overlay
overlay=Gtk.Overlay()
overlay.add(self._scroll)
overlay.add_overlay(self._open_button_revealer)
2021-03-24 19:35:05 +03:00
# connect
open_button.connect("clicked", self._on_open_button_clicked)
# packing
frame=Gtk.Frame()
frame.add(overlay)
box.pack_start(frame, True, True, 0)
self.add(box)
box.show_all()
2021-03-24 19:35:05 +03:00
def open(self, uri, widget, x, y, offset=26):
self._uri=uri
2021-03-24 19:35:05 +03:00
self._rect.x=x
# Gtk places popovers in treeviews 26px above the given position for no obvious reasons, so I move them 26px
# This seems to be related to the width/height of the headers in treeviews
self._rect.y=y+offset
self.set_pointing_to(self._rect)
self.set_relative_to(widget)
window=self.get_toplevel()
self._scroll.set_max_content_height(window.get_size()[1]//2)
self._store.clear()
song=ClientHelper.song_to_str_dict(self._client.get_metadata(uri))
song.pop("time", None)
2020-09-16 16:08:56 +03:00
for tag, value in song.items():
if tag == "duration":
self._store.append([tag+":", ClientHelper.seconds_to_display_time(int(float(value))), locale.str(float(value))])
2020-09-16 16:08:56 +03:00
elif tag == "last-modified":
2020-09-24 22:17:10 +03:00
time=datetime.datetime.strptime(value, "%Y-%m-%dT%H:%M:%SZ")
self._store.append([tag+":", time.strftime("%a %d %B %Y, %H%M UTC"), value])
elif tag == "format":
self._store.append([tag+":", ClientHelper.convert_audio_format(value), value])
2020-09-16 16:08:56 +03:00
else:
self._store.append([tag+":", value, value.replace("&", "&amp;")])
2021-03-24 19:35:05 +03:00
abs_path=self._client.get_absolute_path(uri)
if abs_path is None: # show open with button when song is on the same computer
self._open_button_revealer.set_reveal_child(False)
else:
self._gfile=Gio.File.new_for_path(abs_path)
2021-03-24 19:35:05 +03:00
self._open_button_revealer.set_reveal_child(True)
self.popup()
self._treeview.columns_autosize()
2020-10-13 18:12:54 +03:00
def _on_open_button_clicked(self, *args):
self.popdown()
2020-12-31 16:54:13 +03:00
dialog=Gtk.AppChooserDialog(gfile=self._gfile, transient_for=self.get_toplevel())
app_chooser=dialog.get_widget()
response=dialog.run()
if response == Gtk.ResponseType.OK:
app=app_chooser.get_app_info()
app.launch([self._gfile], None)
dialog.destroy()
2020-08-21 19:03:36 +03:00
def _on_button_clicked(self, widget, mode):
self._client.files_to_playlist([self._uri], mode)
self.popdown()
2020-09-16 16:08:56 +03:00
class SongsView(Gtk.TreeView):
def __init__(self, client, store, file_column_id):
2021-03-26 18:39:21 +03:00
super().__init__(model=store, search_column=-1, activate_on_single_click=True)
2020-09-16 16:08:56 +03:00
self._client=client
self._store=store
self._file_column_id=file_column_id
# selection
self._selection=self.get_selection()
2021-03-24 19:35:05 +03:00
# song popover
self._song_popover=SongPopover(self._client)
2020-09-16 16:08:56 +03:00
# connect
self.connect("row-activated", self._on_row_activated)
self.connect("button-press-event", self._on_button_press_event)
2021-03-26 18:39:21 +03:00
self._client.emitter.connect("show-info", self._on_show_info)
self._client.emitter.connect("add-to-playlist", self._on_add_to_playlist)
2020-09-16 16:08:56 +03:00
def clear(self):
2021-07-04 19:37:14 +03:00
self._song_popover.popdown()
2020-09-16 16:08:56 +03:00
self._store.clear()
def get_files(self):
return_list=[]
for row in self._store:
return_list.append(row[self._file_column_id])
return return_list
def _on_row_activated(self, widget, path, view_column):
2021-03-26 18:39:21 +03:00
self._client.files_to_playlist([self._store[path][self._file_column_id]])
2020-09-16 16:08:56 +03:00
def _on_button_press_event(self, widget, event):
path_re=widget.get_path_at_pos(int(event.x), int(event.y))
if path_re is not None:
path=path_re[0]
if event.button == 1 and event.type == Gdk.EventType._2BUTTON_PRESS:
self._client.files_to_playlist([self._store[path][self._file_column_id]], "play")
elif event.button == 2 and event.type == Gdk.EventType.BUTTON_PRESS:
2021-03-26 21:19:34 +03:00
self._client.files_to_playlist([self._store[path][self._file_column_id]], "append")
elif event.button == 3 and event.type == Gdk.EventType.BUTTON_PRESS:
uri=self._store[path][self._file_column_id]
if self.get_property("headers-visible"):
self._song_popover.open(uri, widget, int(event.x), int(event.y))
else:
self._song_popover.open(uri, widget, int(event.x), int(event.y), offset=0)
2021-03-26 18:39:21 +03:00
def _on_show_info(self, *args):
if self.has_focus():
treeview, treeiter=self._selection.get_selected()
2021-03-27 15:07:32 +03:00
if treeiter is not None:
path=self._store.get_path(treeiter)
cell=self.get_cell_area(path, None)
self._song_popover.open(self._store[path][self._file_column_id], self, cell.x, cell.y)
2021-03-26 18:39:21 +03:00
def _on_add_to_playlist(self, emitter, mode):
if self.has_focus():
treeview, treeiter=self._selection.get_selected()
if treeiter is not None:
self._client.files_to_playlist([self._store.get_value(treeiter, self._file_column_id)], mode)
2020-03-30 23:20:14 +03:00
2020-09-16 16:08:56 +03:00
class SongsWindow(Gtk.Box):
2021-03-27 14:30:56 +03:00
__gsignals__={"button-clicked": (GObject.SignalFlags.RUN_FIRST, None, ())}
def __init__(self, client, store, file_column_id, popover_mode=False):
if popover_mode:
super().__init__(orientation=Gtk.Orientation.VERTICAL, border_width=6, spacing=6)
else:
super().__init__(orientation=Gtk.Orientation.VERTICAL)
2020-09-16 16:08:56 +03:00
self._client=client
2020-09-11 00:07:00 +03:00
2020-09-16 16:08:56 +03:00
# treeview
self._songs_view=SongsView(client, store, file_column_id)
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
# scroll
2020-12-06 02:45:25 +03:00
self._scroll=Gtk.ScrolledWindow()
self._scroll.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
self._scroll.add(self._songs_view)
2020-09-16 15:38:58 +03:00
2020-09-16 16:08:56 +03:00
# buttons
button_box=Gtk.ButtonBox(layout_style=Gtk.ButtonBoxStyle.EXPAND)
data=((_("_Append"), _("Add all titles to playlist"), "list-add-symbolic", "append"),
(_("_Play"), _("Directly play all titles"), "media-playback-start-symbolic", "play"),
(_("_Enqueue"), _("Append all titles after the currently playing track and clear the playlist from all other songs"),
"insert-object-symbolic", "enqueue")
)
for label, tooltip, icon, mode in data:
button=Gtk.Button.new_with_mnemonic(label)
button.set_image(Gtk.Image.new_from_icon_name(icon, Gtk.IconSize.BUTTON))
button.set_tooltip_text(tooltip)
button.connect("clicked", self._on_button_clicked, mode)
button_box.pack_start(button, True, True, 0)
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
# action bar
self._action_bar=Gtk.ActionBar()
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
# packing
if popover_mode:
self.pack_end(button_box, False, False, 0)
frame=Gtk.Frame()
else:
self._action_bar.pack_start(button_box)
self.pack_end(self._action_bar, False, False, 0)
2020-12-06 02:45:25 +03:00
frame=FocusFrame()
frame.set_widget(self._songs_view)
frame.add(self._scroll)
self.pack_start(frame, True, True, 0)
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
def get_treeview(self):
return self._songs_view
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
def get_action_bar(self):
return self._action_bar
2020-07-04 14:16:17 +03:00
2020-12-06 02:45:25 +03:00
def get_scroll(self):
return self._scroll
2021-03-27 13:55:49 +03:00
def _on_button_clicked(self, widget, mode):
self._client.files_to_playlist(self._songs_view.get_files(), mode)
2021-03-27 14:30:56 +03:00
self.emit("button-clicked")
2020-07-04 14:16:17 +03:00
2020-12-06 02:45:25 +03:00
class AlbumPopover(Gtk.Popover):
2021-03-24 19:35:05 +03:00
def __init__(self, client, settings):
2020-12-06 02:45:25 +03:00
super().__init__()
self._client=client
2021-03-24 19:35:05 +03:00
self._settings=settings
self._rect=Gdk.Rectangle()
2020-12-06 02:45:25 +03:00
# songs window
2021-03-26 18:39:21 +03:00
# (track, title (artist), duration, file, search text)
self._store=Gtk.ListStore(str, str, str, str, str)
songs_window=SongsWindow(self._client, self._store, 3, popover_mode=True)
2020-12-06 02:45:25 +03:00
# scroll
2021-03-24 19:35:05 +03:00
self._scroll=songs_window.get_scroll()
self._scroll.set_propagate_natural_height(True)
self._scroll.set_policy(Gtk.PolicyType.NEVER, Gtk.PolicyType.AUTOMATIC)
2020-12-06 02:45:25 +03:00
# songs view
2021-03-24 19:35:05 +03:00
self._songs_view=songs_window.get_treeview()
2021-03-26 18:39:21 +03:00
self._songs_view.set_property("headers-visible", False)
self._songs_view.set_property("search-column", 4)
2020-12-06 02:45:25 +03:00
# columns
renderer_text=Gtk.CellRendererText(width_chars=80, ellipsize=Pango.EllipsizeMode.END, ellipsize_set=True)
renderer_text_ralign=Gtk.CellRendererText(xalign=1.0)
column_track=Gtk.TreeViewColumn(_("No"), renderer_text_ralign, text=0)
column_track.set_property("resizable", False)
2021-03-24 19:35:05 +03:00
self._songs_view.append_column(column_track)
2020-12-06 02:45:25 +03:00
column_title=Gtk.TreeViewColumn(_("Title"), renderer_text, markup=1)
column_title.set_property("resizable", False)
2021-03-24 19:35:05 +03:00
self._songs_view.append_column(column_title)
2020-12-06 02:45:25 +03:00
column_time=Gtk.TreeViewColumn(_("Length"), renderer_text_ralign, text=2)
column_time.set_property("resizable", False)
2021-03-24 19:35:05 +03:00
self._songs_view.append_column(column_time)
2020-12-06 02:45:25 +03:00
2021-03-27 14:30:56 +03:00
# connect
songs_window.connect("button-clicked", lambda *args: self.popdown())
2020-12-06 02:45:25 +03:00
# packing
self.add(songs_window)
songs_window.show_all()
2021-04-06 21:59:25 +03:00
def open(self, album, album_artist, date, genre, widget, x, y):
2021-03-24 19:35:05 +03:00
self._rect.x=x
self._rect.y=y
self.set_pointing_to(self._rect)
self.set_relative_to(widget)
self._scroll.set_max_content_height(4*widget.get_allocated_height()//7)
2021-03-26 18:39:21 +03:00
self._songs_view.set_model(None) # clear old scroll position
2021-03-24 19:35:05 +03:00
self._store.clear()
2021-04-06 21:59:25 +03:00
if genre is None:
genre_filter=()
else:
genre_filter=("genre", genre)
songs=self._client.find("album", album, "date", date, self._settings.get_artist_type(), album_artist, *genre_filter)
2021-03-24 19:35:05 +03:00
for s in songs:
song=ClientHelper.song_to_list_dict(ClientHelper.pepare_song_for_display(s))
track=song["track"][0]
title=(", ".join(song["title"]))
# only show artists =/= albumartist
try:
song["artist"].remove(album_artist)
except:
pass
artist=(", ".join(song["artist"]))
if artist == album_artist or artist == "":
title_artist="<b>{}</b>".format(title)
else:
title_artist="<b>{}</b> • {}".format(title, artist)
2021-03-24 19:35:05 +03:00
title_artist=title_artist.replace("&", "&amp;")
2021-03-26 18:39:21 +03:00
self._store.append([track, title_artist, song["human_duration"][0], song["file"][0], title])
self._songs_view.set_model(self._store)
2021-03-24 19:35:05 +03:00
self.popup()
self._songs_view.columns_autosize()
2021-03-27 13:55:49 +03:00
class ArtistPopover(Gtk.Popover):
def __init__(self, client):
super().__init__()
self._client=client
self._rect=Gdk.Rectangle()
self._artist=None
self._genre=None
# buttons
2021-03-28 01:14:09 +03:00
vbox=Gtk.ButtonBox(orientation=Gtk.Orientation.VERTICAL, border_width=9)
2021-04-25 20:40:43 +03:00
data=((_("Append"), "list-add-symbolic", "append"),
(_("Play"), "media-playback-start-symbolic", "play"),
(_("Enqueue"), "insert-object-symbolic", "enqueue")
2021-03-28 01:14:09 +03:00
)
2021-04-25 20:40:43 +03:00
for label, icon, mode in data:
button=Gtk.ModelButton(label=label, image=Gtk.Image.new_from_icon_name(icon, Gtk.IconSize.BUTTON))
2021-03-28 01:14:09 +03:00
button.get_child().set_property("xalign", 0)
button.connect("clicked", self._on_button_clicked, mode)
vbox.pack_start(button, True, True, 0)
self.add(vbox)
vbox.show_all()
2021-03-27 13:55:49 +03:00
def open(self, artist, genre, widget, x, y):
self._rect.x=x
self._rect.y=y
self.set_pointing_to(self._rect)
self.set_relative_to(widget)
self._artist=artist
self._genre=genre
self.popup()
def _on_button_clicked(self, widget, mode):
self._client.artist_to_playlist(self._artist, self._genre, mode)
self.popdown()
2020-12-06 02:45:25 +03:00
###########
# browser #
###########
2020-09-16 16:08:56 +03:00
class SearchWindow(Gtk.Box):
def __init__(self, client):
super().__init__(orientation=Gtk.Orientation.VERTICAL)
self._client=client
2021-03-19 21:38:17 +03:00
self._stop_flag=False
self._done=True
self._pending=[]
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
# tag switcher
2020-10-19 00:45:36 +03:00
self._tag_combo_box=Gtk.ComboBoxText()
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
# search entry
self.search_entry=Gtk.SearchEntry()
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
# label
self._hits_label=Gtk.Label(xalign=1)
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
# store
# (track, title, artist, album, duration, file, sort track)
self._store=Gtk.ListStore(str, str, str, str, str, str, int)
self._store.set_default_sort_func(lambda *args: 0)
2020-09-15 19:45:30 +03:00
2020-09-16 16:08:56 +03:00
# songs window
self._songs_window=SongsWindow(self._client, self._store, 5)
2020-09-16 16:08:56 +03:00
# action bar
self._action_bar=self._songs_window.get_action_bar()
self._action_bar.set_sensitive(False)
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
# songs view
self._songs_view=self._songs_window.get_treeview()
2020-08-08 14:31:33 +03:00
2020-09-16 16:08:56 +03:00
# columns
renderer_text=Gtk.CellRendererText(ellipsize=Pango.EllipsizeMode.END, ellipsize_set=True)
renderer_text_ralign=Gtk.CellRendererText(xalign=1.0)
2020-08-08 14:31:33 +03:00
2020-09-16 16:08:56 +03:00
column_track=Gtk.TreeViewColumn(_("No"), renderer_text_ralign, text=0)
column_track.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
column_track.set_property("resizable", False)
self._songs_view.append_column(column_track)
2020-08-26 21:30:42 +03:00
2020-09-16 16:08:56 +03:00
column_title=Gtk.TreeViewColumn(_("Title"), renderer_text, text=1)
column_title.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
column_title.set_property("resizable", False)
column_title.set_property("expand", True)
self._songs_view.append_column(column_title)
2020-08-08 14:31:33 +03:00
2020-09-16 16:08:56 +03:00
column_artist=Gtk.TreeViewColumn(_("Artist"), renderer_text, text=2)
column_artist.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
column_artist.set_property("resizable", False)
column_artist.set_property("expand", True)
self._songs_view.append_column(column_artist)
2020-08-08 14:31:33 +03:00
2020-09-16 16:08:56 +03:00
column_album=Gtk.TreeViewColumn(_("Album"), renderer_text, text=3)
column_album.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
column_album.set_property("resizable", False)
column_album.set_property("expand", True)
self._songs_view.append_column(column_album)
2020-08-08 14:31:33 +03:00
2020-09-16 16:08:56 +03:00
column_time=Gtk.TreeViewColumn(_("Length"), renderer_text, text=4)
column_time.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
column_time.set_property("resizable", False)
self._songs_view.append_column(column_time)
column_track.set_sort_column_id(6)
2020-09-16 16:08:56 +03:00
column_title.set_sort_column_id(1)
column_artist.set_sort_column_id(2)
column_album.set_sort_column_id(3)
column_time.set_sort_column_id(4)
2020-08-08 14:31:33 +03:00
# connect
2021-07-04 20:38:05 +03:00
self.search_entry.connect("activate", self._search)
self._search_entry_changed=self.search_entry.connect("search-changed", self._search)
2021-04-23 23:06:13 +03:00
self.search_entry.connect("focus_in_event", self._on_search_entry_focus_event, True)
self.search_entry.connect("focus_out_event", self._on_search_entry_focus_event, False)
self._tag_combo_box_changed=self._tag_combo_box.connect("changed", self._search)
2020-09-16 16:08:56 +03:00
self._client.emitter.connect("reconnected", self._on_reconnected)
self._client.emitter.connect("disconnected", self._on_disconnected)
2021-07-04 20:38:05 +03:00
self._client.emitter.connect("update", self._search)
2020-08-08 14:31:33 +03:00
# packing
2020-09-16 16:08:56 +03:00
hbox=Gtk.Box(spacing=6, border_width=6)
hbox.pack_start(self.search_entry, True, True, 0)
2020-10-19 00:45:36 +03:00
hbox.pack_end(self._tag_combo_box, False, False, 0)
2020-09-16 16:08:56 +03:00
self._hits_label.set_margin_end(6)
self._action_bar.pack_end(self._hits_label)
self.pack_start(hbox, False, False, 0)
2021-07-19 23:41:32 +03:00
self.pack_start(Gtk.Separator(orientation=Gtk.Orientation.HORIZONTAL), False, False, 0)
2020-09-16 16:08:56 +03:00
self.pack_start(self._songs_window, True, True, 0)
2020-08-08 14:31:33 +03:00
2021-03-19 21:38:17 +03:00
def _clear(self, *args):
if self._done:
self.search_entry.handler_block(self._search_entry_changed)
self._tag_combo_box.handler_block(self._tag_combo_box_changed)
2021-03-19 21:38:17 +03:00
self._songs_view.clear()
self.search_entry.set_text("")
self._tag_combo_box.remove_all()
self.search_entry.handler_unblock(self._search_entry_changed)
self._tag_combo_box.handler_unblock(self._tag_combo_box_changed)
2021-03-19 21:38:17 +03:00
elif not self._clear in self._pending:
self._stop_flag=True
self._pending.append(self._clear)
2020-08-08 14:31:33 +03:00
2020-09-16 16:08:56 +03:00
def _on_disconnected(self, *args):
2020-10-19 00:45:36 +03:00
self._tag_combo_box.set_sensitive(False)
2020-09-16 16:08:56 +03:00
self.search_entry.set_sensitive(False)
2021-03-19 21:38:17 +03:00
self._clear()
2020-08-08 14:31:33 +03:00
2020-09-16 16:08:56 +03:00
def _on_reconnected(self, *args):
2021-03-19 21:38:17 +03:00
if self._done:
self._tag_combo_box.handler_block(self._tag_combo_box_changed)
2021-03-19 21:38:17 +03:00
self._tag_combo_box.append_text(_("all tags"))
for tag in self._client.tagtypes():
2021-03-19 21:38:17 +03:00
if not tag.startswith("MUSICBRAINZ"):
self._tag_combo_box.append_text(tag)
self._tag_combo_box.set_active(0)
self._tag_combo_box.set_sensitive(True)
self.search_entry.set_sensitive(True)
self._tag_combo_box.handler_unblock(self._tag_combo_box_changed)
2021-03-19 21:38:17 +03:00
elif not self._on_reconnected in self._pending:
self._stop_flag=True
self._pending.append(self._on_reconnected)
def _search(self, *args):
2021-03-19 21:38:17 +03:00
if self._done:
self._done=False
self._songs_view.clear()
self._hits_label.set_text("")
2020-09-16 16:08:56 +03:00
self._action_bar.set_sensitive(False)
2021-07-23 18:29:35 +03:00
hits=0
if self.search_entry.get_text():
if self._tag_combo_box.get_active() == 0:
search_tag="any"
else:
search_tag=self._tag_combo_box.get_active_text()
search_text=self.search_entry.get_text()
stripe_size=100
2021-07-21 19:36:26 +03:00
try: # client cloud meanwhile disconnect
2021-07-23 18:29:35 +03:00
songs=self._client.search(search_tag, search_text, "window", f"0:{stripe_size}")
2021-07-21 19:36:26 +03:00
except MPDBase.ConnectionError:
self._done_callback()
return
2021-07-23 18:29:35 +03:00
stripe_start=stripe_size
while songs:
hits+=len(songs)
for s in songs:
if self._stop_flag:
self._done_callback()
return
song=ClientHelper.song_to_str_dict(ClientHelper.pepare_song_for_display(s))
try:
int_track=int(song["track"])
except:
int_track=0
self._store.append([
song["track"], song["title"],
song["artist"], song["album"],
song["human_duration"], song["file"],
int_track
])
self.search_entry.progress_pulse()
self._hits_label.set_text(ngettext("{hits} hit", "{hits} hits", hits).format(hits=hits))
while Gtk.events_pending():
Gtk.main_iteration_do(True)
stripe_end=stripe_start+stripe_size
try: # client cloud meanwhile disconnect
songs=self._client.search(search_tag, search_text, "window", f"{stripe_start}:{stripe_end}")
except MPDBase.ConnectionError:
self._done_callback()
2021-03-19 21:38:17 +03:00
return
2021-07-23 18:29:35 +03:00
stripe_start+=stripe_size
if hits > 0:
2021-03-19 21:38:17 +03:00
self._action_bar.set_sensitive(True)
self._done_callback()
elif not self._search in self._pending:
2021-03-19 21:38:17 +03:00
self._stop_flag=True
self._pending.append(self._search)
2021-03-19 21:38:17 +03:00
def _done_callback(self, *args):
self.search_entry.set_progress_fraction(0.0)
self._stop_flag=False
self._done=True
pending=self._pending
self._pending=[]
for p in pending:
try:
p()
except:
pass
return False
2020-08-08 14:31:33 +03:00
2021-04-23 23:06:13 +03:00
def _on_search_entry_focus_event(self, widget, event, focus):
app=self.get_toplevel().get_application()
if focus:
app.set_accels_for_action("mpd.toggle-play", [])
else:
app.set_accels_for_action("mpd.toggle-play", ["space"])
2021-06-24 22:24:22 +03:00
class SelectionList(Gtk.TreeView):
__gsignals__={"item-selected": (GObject.SignalFlags.RUN_FIRST, None, ()), "clear": (GObject.SignalFlags.RUN_FIRST, None, ())}
def __init__(self, select_all_string):
super().__init__(activate_on_single_click=True, search_column=0, headers_visible=False, fixed_height_mode=True)
self.select_all_string=select_all_string
self._selected_path=None
# store
# (item, weight, initial-letter, weight-initials)
self._store=Gtk.ListStore(str, Pango.Weight, str, Pango.Weight)
self._store.append([self.select_all_string, Pango.Weight.BOOK, "", Pango.Weight.BOOK])
self.set_model(self._store)
self._selection=self.get_selection()
# columns
renderer_text_malign=Gtk.CellRendererText(xalign=0.5)
self._column_initial=Gtk.TreeViewColumn("", renderer_text_malign, text=2, weight=3)
self._column_initial.set_property("sizing", Gtk.TreeViewColumnSizing.FIXED)
self._column_initial.set_property("min-width", 30)
self.append_column(self._column_initial)
renderer_text=Gtk.CellRendererText(ellipsize=Pango.EllipsizeMode.END, ellipsize_set=True)
self._column_item=Gtk.TreeViewColumn("", renderer_text, text=0, weight=1)
self._column_item.set_property("sizing", Gtk.TreeViewColumnSizing.FIXED)
self._column_item.set_property("expand", True)
self.append_column(self._column_item)
# connect
self.connect("row-activated", self._on_row_activated)
def clear(self):
self._store.clear()
self._store.append([self.select_all_string, Pango.Weight.BOOK, "", Pango.Weight.BOOK])
self._selected_path=None
self.emit("clear")
def set_items(self, items):
self.clear()
current_char=""
2021-07-03 16:05:56 +03:00
items.sort(key=locale.strxfrm)
items.sort(key=lambda item: locale.strxfrm(item[:1]))
2021-06-24 22:24:22 +03:00
for item in items:
2021-07-03 16:05:56 +03:00
if current_char == item[:1].upper():
2021-06-24 22:24:22 +03:00
self._store.append([item, Pango.Weight.BOOK, "", Pango.Weight.BOOK])
2021-07-03 16:05:56 +03:00
else:
self._store.append([item, Pango.Weight.BOOK, item[:1].upper(), Pango.Weight.BOLD])
current_char=item[:1].upper()
2021-06-24 22:24:22 +03:00
def get_item(self, path):
if path == Gtk.TreePath(0):
return None
else:
return self._store[path][0]
def length(self):
return len(self._store)-1
def select_path(self, path):
self.set_cursor(path, None, False)
self.row_activated(path, self._column_item)
def select(self, item):
row_num=len(self._store)
for i in range(0, row_num):
path=Gtk.TreePath(i)
if self._store[path][0] == item:
self.select_path(path)
break
def select_all(self):
self.set_cursor(Gtk.TreePath(0), None, False)
self.row_activated(Gtk.TreePath(0), self._column_item)
def get_selected(self):
if self._selected_path is None:
raise ValueError("None selected")
elif self._selected_path == Gtk.TreePath(0):
return None
else:
return self._store[self._selected_path][0]
def highlight_selected(self):
self.set_cursor(self._selected_path, None, False)
def _on_row_activated(self, widget, path, view_column):
if path != self._selected_path:
if self._selected_path is not None:
self._store[self._selected_path][1]=Pango.Weight.BOOK
self._store[path][1]=Pango.Weight.BOLD
self._selected_path=path
self.emit("item-selected")
class GenreSelect(SelectionList):
2020-07-04 14:16:17 +03:00
def __init__(self, client):
2021-06-24 22:24:22 +03:00
super().__init__(_("all genres"))
self._client=client
2020-07-04 14:16:17 +03:00
# connect
2020-09-16 16:08:56 +03:00
self._client.emitter.connect("disconnected", self._on_disconnected)
self._client.emitter.connect("reconnected", self._on_reconnected)
self._client.emitter.connect("update", self._refresh)
2020-07-04 14:16:17 +03:00
2020-09-16 16:08:56 +03:00
def deactivate(self):
2021-06-24 22:24:22 +03:00
self.select_all()
2020-07-04 14:16:17 +03:00
def _refresh(self, *args):
2021-06-24 22:24:22 +03:00
self.set_items(self._client.comp_list("genre"))
self.select_all()
2020-09-16 16:08:56 +03:00
def _on_disconnected(self, *args):
self.set_sensitive(False)
2021-06-24 22:24:22 +03:00
self.clear()
2020-09-16 16:08:56 +03:00
def _on_reconnected(self, *args):
self._refresh()
self.set_sensitive(True)
2020-07-04 14:16:17 +03:00
2021-06-24 22:24:22 +03:00
class ArtistWindow(SelectionList):
2020-09-16 16:08:56 +03:00
def __init__(self, client, settings, genre_select):
2021-06-24 22:24:22 +03:00
super().__init__(_("all artists"))
self._client=client
self._settings=settings
2021-04-06 21:59:25 +03:00
self.genre_select=genre_select
2020-03-03 15:41:46 +03:00
2021-06-24 22:24:22 +03:00
# selection
self._selection=self.get_selection()
2021-03-27 13:55:49 +03:00
# artist popover
self._artist_popover=ArtistPopover(self._client)
2020-07-04 13:35:39 +03:00
# connect
2021-07-04 19:37:14 +03:00
self.connect("clear", lambda *args: self._artist_popover.popdown())
2021-06-24 22:24:22 +03:00
self.connect("button-press-event", self._on_button_press_event)
2020-09-16 16:08:56 +03:00
self._settings.connect("changed::use-album-artist", self._refresh)
self._client.emitter.connect("disconnected", self._on_disconnected)
self._client.emitter.connect("reconnected", self._on_reconnected)
2021-03-26 18:39:21 +03:00
self._client.emitter.connect("add-to-playlist", self._on_add_to_playlist)
2021-03-27 15:07:32 +03:00
self._client.emitter.connect("show-info", self._on_show_info)
2021-06-24 22:24:22 +03:00
self.genre_select.connect_after("item-selected", self._refresh)
2020-09-16 16:08:56 +03:00
def _refresh(self, *args):
2021-06-24 22:24:22 +03:00
genre=self.genre_select.get_selected()
2020-09-16 16:08:56 +03:00
if genre is None:
artists=self._client.comp_list(self._settings.get_artist_type())
2020-09-16 16:08:56 +03:00
else:
artists=self._client.comp_list(self._settings.get_artist_type(), "genre", genre)
2021-06-24 22:24:22 +03:00
self.set_items(artists)
if genre is not None:
2021-06-24 22:24:22 +03:00
self.select_all()
else:
song=ClientHelper.song_to_first_str_dict(self._client.currentsong())
if song != {}:
artist=song.get(self._settings.get_artist_type())
if artist is None:
artist=song.get("artist", "")
self.select(artist)
else:
2021-06-24 22:24:22 +03:00
if self.length() > 0:
self.select_path(Gtk.TreePath(1))
else:
2021-06-24 22:24:22 +03:00
self.select_path(Gtk.TreePath(0))
2020-09-16 16:08:56 +03:00
2021-03-26 21:19:34 +03:00
def _on_button_press_event(self, widget, event):
if ((event.button in (2,3) and event.type == Gdk.EventType.BUTTON_PRESS)
or (event.button == 1 and event.type == Gdk.EventType._2BUTTON_PRESS)):
2021-03-27 13:55:49 +03:00
path_re=widget.get_path_at_pos(int(event.x), int(event.y))
if path_re is not None:
path=path_re[0]
2021-06-24 22:24:22 +03:00
genre=self.genre_select.get_selected()
artist=self.get_item(path)
if event.button == 1:
self._client.artist_to_playlist(artist, genre, "play")
elif event.button == 2:
self._client.artist_to_playlist(artist, genre, "append")
elif event.button == 3:
2021-06-24 22:24:22 +03:00
self._artist_popover.open(artist, genre, self, event.x, event.y)
2020-09-16 16:08:56 +03:00
2021-03-26 18:39:21 +03:00
def _on_add_to_playlist(self, emitter, mode):
2021-06-24 22:24:22 +03:00
if self.has_focus():
selected_rows=self._selection.get_selected_rows()
if selected_rows is not None:
path=selected_rows[1][0]
genre=self.genre_select.get_selected()
artist=self.get_item(path)
self._client.artist_to_playlist(artist, genre, mode)
2021-03-26 18:39:21 +03:00
2021-03-27 15:07:32 +03:00
def _on_show_info(self, *args):
2021-06-24 22:24:22 +03:00
if self.has_focus():
selected_rows=self._selection.get_selected_rows()
if selected_rows is not None:
path=selected_rows[1][0]
genre=self.genre_select.get_selected()
artist=self.get_item(path)
cell=self.get_cell_area(path, None)
self._artist_popover.open(artist, genre, self, cell.x, cell.y)
2021-03-27 15:07:32 +03:00
2020-09-16 16:08:56 +03:00
def _on_disconnected(self, *args):
self.set_sensitive(False)
2021-06-24 22:24:22 +03:00
self.clear()
2020-09-16 16:08:56 +03:00
def _on_reconnected(self, *args):
self.set_sensitive(True)
class AlbumWindow(FocusFrame):
2020-12-06 02:45:25 +03:00
def __init__(self, client, settings, artist_window):
2020-09-16 16:08:56 +03:00
super().__init__()
self._settings=settings
self._client=client
self._artist_window=artist_window
self._stop_flag=False
2020-09-16 16:08:56 +03:00
self._done=True
self._pending=[]
# cover, display_label, display_label_artist, tooltip(titles), album, year, artist
self._store=Gtk.ListStore(GdkPixbuf.Pixbuf, str, str, str, str, str, str)
self._sort_settings()
# iconview
2021-03-26 18:39:21 +03:00
self._iconview=Gtk.IconView(
model=self._store, item_width=0, pixbuf_column=0, markup_column=1, tooltip_column=3, activate_on_single_click=True
)
2020-09-16 16:08:56 +03:00
# scroll
scroll=Gtk.ScrolledWindow()
scroll.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
scroll.add(self._iconview)
2020-12-06 02:45:25 +03:00
self._scroll_vadj=scroll.get_vadjustment()
self._scroll_hadj=scroll.get_hadjustment()
2020-09-16 16:08:56 +03:00
# progress bar
self._progress_bar=Gtk.ProgressBar(no_show_all=True)
2021-03-27 13:55:49 +03:00
# popover
2021-03-24 19:35:05 +03:00
self._album_popover=AlbumPopover(self._client, self._settings)
2021-03-27 13:55:49 +03:00
self._artist_popover=ArtistPopover(self._client)
2021-03-24 19:35:05 +03:00
2020-09-16 16:08:56 +03:00
# connect
self._iconview.connect("item-activated", self._on_item_activated)
self._iconview.connect("button-press-event", self._on_button_press_event)
self._client.emitter.connect("disconnected", self._on_disconnected)
self._client.emitter.connect("reconnected", self._on_reconnected)
2021-03-26 18:39:21 +03:00
self._client.emitter.connect("show-info", self._on_show_info)
self._client.emitter.connect("add-to-playlist", self._on_add_to_playlist)
2020-09-16 16:08:56 +03:00
self._settings.connect("changed::sort-albums-by-year", self._sort_settings)
self._settings.connect("changed::album-cover", self._on_cover_size_changed)
2021-06-24 22:24:22 +03:00
self._artist_window.connect("item-selected", self._refresh)
self._artist_window.connect("clear", self._clear)
2020-09-16 16:08:56 +03:00
# packing
2020-09-16 16:08:56 +03:00
self.set_widget(self._iconview)
box=Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
box.pack_start(scroll, True, True, 0)
box.pack_start(self._progress_bar, False, False, 0)
self.add(box)
2020-09-16 16:08:56 +03:00
def _workaround_clear(self):
self._store.clear()
# workaround (scrollbar still visible after clear)
self._iconview.set_model(None)
self._iconview.set_model(self._store)
def _clear(self, *args):
if self._done:
2021-07-04 19:37:14 +03:00
self._album_popover.popdown()
self._artist_popover.popdown()
2020-09-16 16:08:56 +03:00
self._workaround_clear()
elif not self._clear in self._pending:
self._stop_flag=True
2020-09-16 16:08:56 +03:00
self._pending.append(self._clear)
def scroll_to_current_album(self):
def callback():
song=ClientHelper.song_to_first_str_dict(self._client.currentsong())
2020-09-23 17:21:24 +03:00
album=song.get("album", "")
2020-09-16 16:08:56 +03:00
self._iconview.unselect_all()
row_num=len(self._store)
for i in range(0, row_num):
path=Gtk.TreePath(i)
treeiter=self._store.get_iter(path)
if self._store.get_value(treeiter, 4) == album:
self._iconview.set_cursor(path, None, False)
self._iconview.select_path(path)
self._iconview.scroll_to_path(path, True, 0, 0)
break
if self._done:
callback()
elif not self.scroll_to_current_album in self._pending:
self._pending.append(self.scroll_to_current_album)
def _sort_settings(self, *args):
if self._settings.get_boolean("sort-albums-by-year"):
self._store.set_sort_column_id(5, Gtk.SortType.ASCENDING)
else:
self._store.set_sort_column_id(1, Gtk.SortType.ASCENDING)
def _refresh(self, *args):
if self._done:
self._done=False
self._settings.set_property("cursor-watch", True)
self._progress_bar.show()
self._store.clear()
self._iconview.set_model(None)
2020-12-13 15:44:25 +03:00
try: # self._artist_window can still be empty (e.g. when client is not connected and cover size gets changed)
2021-06-24 22:24:22 +03:00
artist=self._artist_window.get_selected()
genre=self._artist_window.genre_select.get_selected()
2020-09-16 16:08:56 +03:00
except:
self._done_callback()
2020-12-13 15:44:25 +03:00
return
if artist is None:
self._iconview.set_markup_column(2) # show artist names
if genre is None:
artists=self._client.comp_list(self._settings.get_artist_type())
else:
artists=self._client.comp_list(self._settings.get_artist_type(), "genre", genre)
2020-09-16 16:08:56 +03:00
else:
self._iconview.set_markup_column(1) # hide artist names
artists=[artist]
2020-09-16 16:08:56 +03:00
# prepare albmus list (run all mpd related commands)
albums=[]
for i, artist in enumerate(artists):
2020-09-16 16:08:56 +03:00
try: # client cloud meanwhile disconnect
if self._stop_flag:
self._done_callback()
2020-09-16 16:08:56 +03:00
return
else:
if i > 0: # more than one artist to show (all artists)
self._progress_bar.pulse()
albums.extend(self._client.get_albums(artist, genre))
2020-09-16 16:08:56 +03:00
while Gtk.events_pending():
Gtk.main_iteration_do(True)
except MPDBase.ConnectionError:
self._done_callback()
2020-09-16 16:08:56 +03:00
return
def display_albums():
for i, album in enumerate(albums):
# tooltip
2021-07-16 20:46:39 +03:00
duration=ClientHelper.calc_display_duration(album["songs"])
2021-07-16 20:39:48 +03:00
length=len(album["songs"])
discs=album["songs"][-1].get("disc", 1)
2021-07-22 20:42:49 +03:00
if isinstance(discs, list):
discs=int(discs[0])
else:
discs=int(discs)
if discs > 1:
2021-07-16 21:28:02 +03:00
tooltip=_("{number} songs on {discs} discs ({duration})").format(
number=length, discs=discs, duration=duration)
2021-07-16 20:39:48 +03:00
else:
2021-07-16 21:28:02 +03:00
tooltip=ngettext("{number} song ({duration})", "{number} songs ({duration})", length).format(
number=length, duration=duration)
# album label
if album["year"] == "":
display_label="<b>{}</b>".format(album["album"])
else:
display_label="<b>{}</b> ({})".format(album["album"], album["year"])
display_label_artist=display_label+"\n"+album["artist"]
display_label=display_label.replace("&", "&amp;")
display_label_artist=display_label_artist.replace("&", "&amp;")
# add album
self._store.append(
[album["cover"], display_label, display_label_artist,
tooltip, album["album"], album["year"], album["artist"]]
)
self._iconview.set_model(self._store)
self._done_callback()
return False
def render_covers():
size=self._settings.get_int("album-cover")
total_albums=len(albums)
for i, album in enumerate(albums):
if self._stop_flag:
break
if "cover_path" in album:
2021-06-02 22:34:01 +03:00
album["cover"]=ClientHelper.file_to_pixbuf(album["cover_path"], size)
else:
if "cover_binary" in album:
album["cover"]=ClientHelper.binary_to_pixbuf(album["cover_binary"], size)
else:
2021-06-02 22:34:01 +03:00
album["cover"]=GdkPixbuf.Pixbuf.new_from_file_at_size(FALLBACK_COVER, size, size)
GLib.idle_add(self._progress_bar.set_fraction, (i+1)/total_albums)
if self._stop_flag:
GLib.idle_add(self._done_callback)
2020-09-16 16:08:56 +03:00
else:
GLib.idle_add(display_albums)
cover_thread=threading.Thread(target=render_covers, daemon=True)
cover_thread.start()
2020-09-16 16:08:56 +03:00
elif not self._refresh in self._pending:
self._stop_flag=True
2020-09-16 16:08:56 +03:00
self._pending.append(self._refresh)
def _path_to_playlist(self, path, mode="default"):
album=self._store[path][4]
year=self._store[path][5]
artist=self._store[path][6]
2021-06-24 22:24:22 +03:00
genre=self._artist_window.genre_select.get_selected()
2021-04-06 21:59:25 +03:00
self._client.album_to_playlist(album, artist, year, genre, mode)
2020-09-16 16:08:56 +03:00
def _done_callback(self, *args):
self._settings.set_property("cursor-watch", False)
self._progress_bar.hide()
self._progress_bar.set_fraction(0)
self._stop_flag=False
2020-09-16 16:08:56 +03:00
self._done=True
pending=self._pending
self._pending=[]
for p in pending:
try:
p()
except:
pass
return False
def _on_button_press_event(self, widget, event):
path=widget.get_path_at_pos(int(event.x), int(event.y))
if event.button == 1 and event.type == Gdk.EventType._2BUTTON_PRESS:
if path is not None:
self._path_to_playlist(path, "play")
elif event.button == 2 and event.type == Gdk.EventType.BUTTON_PRESS:
2021-03-27 13:55:49 +03:00
if path is not None:
2021-03-26 21:19:34 +03:00
self._path_to_playlist(path, "append")
2021-03-27 13:55:49 +03:00
elif event.button == 3 and event.type == Gdk.EventType.BUTTON_PRESS:
v=self._scroll_vadj.get_value()
h=self._scroll_hadj.get_value()
2021-06-24 22:24:22 +03:00
genre=self._artist_window.genre_select.get_selected()
2021-03-27 13:55:49 +03:00
if path is not None:
2021-03-26 21:19:34 +03:00
album=self._store[path][4]
year=self._store[path][5]
artist=self._store[path][6]
# when using "button-press-event" in iconview popovers only show up in combination with idle_add (bug in GTK?)
2021-04-06 21:59:25 +03:00
GLib.idle_add(self._album_popover.open, album, artist, year, genre, widget, event.x-h, event.y-v)
2021-03-27 13:55:49 +03:00
else:
2021-06-24 22:24:22 +03:00
artist=self._artist_window.get_selected()
GLib.idle_add(self._artist_popover.open, artist, genre, widget, event.x-h, event.y-v)
2020-03-03 15:41:46 +03:00
2020-09-16 16:08:56 +03:00
def _on_item_activated(self, widget, path):
treeiter=self._store.get_iter(path)
2021-04-06 21:59:25 +03:00
album=self._store.get_value(treeiter, 4)
year=self._store.get_value(treeiter, 5)
artist=self._store.get_value(treeiter, 6)
2021-06-24 22:24:22 +03:00
genre=self._artist_window.genre_select.get_selected()
2021-04-06 21:59:25 +03:00
self._client.album_to_playlist(album, artist, year, genre)
2020-03-03 15:41:46 +03:00
def _on_disconnected(self, *args):
2020-09-16 16:08:56 +03:00
self._iconview.set_sensitive(False)
def _on_reconnected(self, *args):
2020-09-16 16:08:56 +03:00
self._iconview.set_sensitive(True)
2021-03-26 18:39:21 +03:00
def _on_show_info(self, *args):
if self._iconview.has_focus():
paths=self._iconview.get_selected_items()
if len(paths) > 0:
rect=self._iconview.get_cell_rect(paths[0], None)[1]
x=rect.x+rect.width//2
y=rect.y+rect.height//2
2021-06-24 22:24:22 +03:00
genre=self._artist_window.genre_select.get_selected()
2021-03-26 18:39:21 +03:00
self._album_popover.open(
2021-04-06 21:59:25 +03:00
self._store[paths[0]][4], self._store[paths[0]][6], self._store[paths[0]][5], genre, self._iconview, x, y
2021-03-26 18:39:21 +03:00
)
def _on_add_to_playlist(self, emitter, mode):
if self._iconview.has_focus():
paths=self._iconview.get_selected_items()
if len(paths) != 0:
self._path_to_playlist(paths[0], mode)
2020-09-16 16:08:56 +03:00
def _on_cover_size_changed(self, *args):
def callback():
self._refresh()
return False
GLib.idle_add(callback)
2020-09-15 19:45:30 +03:00
2020-09-16 16:08:56 +03:00
class Browser(Gtk.Paned):
def __init__(self, client, settings):
2021-02-07 21:05:30 +03:00
super().__init__(orientation=Gtk.Orientation.HORIZONTAL)
self._client=client
self._settings=settings
2020-09-16 16:08:56 +03:00
self._use_csd=self._settings.get_boolean("use-csd")
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
# widgets
2020-10-20 14:13:50 +03:00
icons={}
icons_data=("go-previous-symbolic", "system-search-symbolic")
2021-07-16 20:39:48 +03:00
icon_size={True: Gtk.IconSize.BUTTON, False: Gtk.IconSize.LARGE_TOOLBAR}[self._use_csd]
for data in icons_data:
icons[data]=Gtk.Image.new_from_icon_name(data, icon_size)
self.back_to_current_album_button=Gtk.Button(image=icons["go-previous-symbolic"], tooltip_text=_("Back to current album"))
2020-09-29 14:02:51 +03:00
self.back_to_current_album_button.set_can_focus(False)
2020-10-20 14:13:50 +03:00
self.search_button=Gtk.ToggleButton(image=icons["system-search-symbolic"], tooltip_text=_("Search"))
2020-09-29 14:02:51 +03:00
self.search_button.set_can_focus(False)
2021-06-24 22:24:22 +03:00
label=Gtk.Label(ellipsize=Pango.EllipsizeMode.MIDDLE)
self._genres_button=Gtk.ToggleButton(image=label, tooltip_text=_("Filter by genre"))
self._genre_select=GenreSelect(self._client)
self._artist_window=ArtistWindow(self._client, self._settings, self._genre_select)
2020-09-16 16:08:56 +03:00
self._search_window=SearchWindow(self._client)
2020-12-06 02:45:25 +03:00
self._album_window=AlbumWindow(self._client, self._settings, self._artist_window)
2020-01-27 22:27:35 +03:00
2021-06-24 22:24:22 +03:00
# stacks
self._artists_genres_stack=Gtk.Stack(transition_type=Gtk.StackTransitionType.OVER_RIGHT_LEFT)
self._artists_genres_stack.add_named(ScrolledFocusFrame(self._artist_window), "artists")
self._artists_genres_stack.add_named(ScrolledFocusFrame(self._genre_select), "genres")
self._albums_search_stack=Gtk.Stack(transition_type=Gtk.StackTransitionType.CROSSFADE)
self._albums_search_stack.add_named(self._album_window, "albums")
self._albums_search_stack.add_named(self._search_window, "search")
2020-07-04 13:35:39 +03:00
# connect
self.back_to_current_album_button.connect("clicked", self._on_back_to_current_album_button_clicked)
self.back_to_current_album_button.connect("button-press-event", self._on_back_to_current_album_button_press_event)
2020-09-16 16:08:56 +03:00
self.search_button.connect("toggled", self._on_search_toggled)
2021-06-24 22:24:22 +03:00
self._genres_button.connect("toggled", self._on_genres_toggled)
self._genre_select.connect("item-selected", self._on_genre_chnaged)
self._artist_window.connect("item-selected", self._on_artists_changed)
2020-09-16 16:08:56 +03:00
self._settings.connect("notify::mini-player", self._on_mini_player)
self._client.emitter.connect("disconnected", self._on_disconnected)
self._client.emitter.connect("reconnected", self._on_reconnected)
2020-07-04 13:35:39 +03:00
# packing
hbox=Gtk.Box(spacing=6, border_width=6)
2021-06-24 22:24:22 +03:00
if not self._use_csd:
2020-09-16 16:08:56 +03:00
hbox.pack_start(self.back_to_current_album_button, False, False, 0)
2021-06-24 22:24:22 +03:00
hbox.pack_end(self.search_button, False, False, 0)
hbox.pack_start(self._genres_button, True, True, 0)
vbox=Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
vbox.pack_start(hbox, False, False, 0)
2021-07-19 23:41:32 +03:00
vbox.pack_start(Gtk.Separator(orientation=Gtk.Orientation.HORIZONTAL), False, False, 0)
2021-06-24 22:24:22 +03:00
vbox.pack_start(self._artists_genres_stack, True, True, 0)
self.pack1(vbox, False, False)
self.pack2(self._albums_search_stack, True, False)
2020-09-16 16:08:56 +03:00
def _back_to_current_album(self, force=False):
song=ClientHelper.song_to_first_str_dict(self._client.currentsong())
if song != {}:
2020-09-16 16:08:56 +03:00
self.search_button.set_active(False)
2021-06-24 22:24:22 +03:00
self._genres_button.set_active(False)
2020-09-16 16:08:56 +03:00
# get artist name
artist=song.get(self._settings.get_artist_type())
if artist is None:
2020-09-23 17:21:24 +03:00
artist=song.get("artist", "")
2020-09-16 16:08:56 +03:00
# deactivate genre filter to show all artists (if needed)
2021-06-24 22:24:22 +03:00
if song.get("genre", "") != self._genre_select.get_selected() or force:
self._genre_select.deactivate()
2020-09-16 16:08:56 +03:00
# select artist
2021-06-24 22:24:22 +03:00
if self._artist_window.get_selected() is None and not force: # all artists selected
2020-09-16 16:08:56 +03:00
self.search_button.set_active(False)
self._artist_window.highlight_selected()
else: # one artist selected
self._artist_window.select(artist)
2020-09-16 16:08:56 +03:00
self._album_window.scroll_to_current_album()
2020-08-21 19:47:17 +03:00
def _on_back_to_current_album_button_clicked(self, *args):
self._back_to_current_album()
def _on_back_to_current_album_button_press_event(self, widget, event):
if event.button == 1 and event.type == Gdk.EventType._2BUTTON_PRESS:
self._back_to_current_album(force=True)
2021-06-24 22:24:22 +03:00
def _on_genres_toggled(self, widget):
if widget.get_active():
self._artists_genres_stack.set_visible_child_name("genres")
else:
self._artists_genres_stack.set_visible_child_name("artists")
2020-09-16 16:08:56 +03:00
def _on_search_toggled(self, widget):
if widget.get_active():
2021-06-24 22:24:22 +03:00
self._albums_search_stack.set_visible_child_name("search")
2020-09-16 16:08:56 +03:00
self._search_window.search_entry.grab_focus()
else:
2021-06-24 22:24:22 +03:00
self._albums_search_stack.set_visible_child_name("albums")
2020-09-16 16:08:56 +03:00
def _on_reconnected(self, *args):
self.back_to_current_album_button.set_sensitive(True)
self.search_button.set_sensitive(True)
2021-06-24 22:24:22 +03:00
self._genres_button.set_sensitive(True)
2020-08-18 12:54:15 +03:00
def _on_disconnected(self, *args):
2020-09-16 16:08:56 +03:00
self.back_to_current_album_button.set_sensitive(False)
self.search_button.set_active(False)
self.search_button.set_sensitive(False)
2021-06-24 22:24:22 +03:00
self._genres_button.set_active(False)
self._genres_button.set_sensitive(False)
def _on_genre_chnaged(self, *args):
genre=self._genre_select.get_selected()
if genre is None:
self._genres_button.get_image().set_text(self._genre_select.select_all_string)
else:
self._genres_button.get_image().set_text(genre)
self._genres_button.set_active(False)
2020-08-09 23:58:48 +03:00
2020-09-16 16:08:56 +03:00
def _on_artists_changed(self, *args):
self.search_button.set_active(False)
2020-06-27 17:11:41 +03:00
2020-09-15 19:45:30 +03:00
def _on_mini_player(self, obj, typestring):
2021-02-08 19:49:45 +03:00
state=obj.get_property("mini-player")
self.set_property("no-show-all", state)
self.back_to_current_album_button.set_property("no-show-all", state)
self.search_button.set_property("no-show-all", state)
self.set_property("visible", not(state))
self.back_to_current_album_button.set_property("visible", not(state))
self.search_button.set_property("visible", not(state))
if not state:
self.show_all()
2020-09-15 19:45:30 +03:00
2020-09-16 16:08:56 +03:00
######################
# playlist and cover #
######################
2020-09-16 17:57:58 +03:00
class LyricsWindow(FocusFrame):
2020-09-16 16:08:56 +03:00
def __init__(self, client, settings):
2020-08-31 11:44:23 +03:00
super().__init__()
self._settings=settings
2020-09-16 16:08:56 +03:00
self._client=client
2020-09-16 17:57:58 +03:00
self._displayed_song_file=None
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
# text view
2020-09-16 17:57:58 +03:00
self._text_view=Gtk.TextView(
2020-09-16 16:08:56 +03:00
editable=False,
cursor_visible=False,
wrap_mode=Gtk.WrapMode.WORD,
justification=Gtk.Justification.CENTER,
opacity=0.9
2020-09-12 15:31:17 +03:00
)
2020-09-16 17:57:58 +03:00
self._text_view.set_left_margin(5)
2020-09-24 17:50:19 +03:00
self._text_view.set_right_margin(5)
2020-09-16 17:57:58 +03:00
self._text_view.set_bottom_margin(5)
self._text_view.set_top_margin(3)
self.set_widget(self._text_view)
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
# text buffer
2020-09-16 17:57:58 +03:00
self._text_buffer=self._text_view.get_buffer()
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
# scroll
scroll=Gtk.ScrolledWindow()
scroll.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
2020-09-16 17:57:58 +03:00
scroll.add(self._text_view)
2020-06-26 21:24:06 +03:00
2020-07-04 14:16:17 +03:00
# connect
2020-09-16 17:57:58 +03:00
self._client.emitter.connect("disconnected", self._on_disconnected)
2020-09-16 16:08:56 +03:00
self._song_changed=self._client.emitter.connect("current_song_changed", self._refresh)
2020-09-16 17:57:58 +03:00
self._client.emitter.handler_block(self._song_changed)
2020-06-26 21:24:06 +03:00
2020-09-16 16:08:56 +03:00
# packing
2020-09-16 17:57:58 +03:00
self.add(scroll)
2020-06-26 21:24:06 +03:00
2020-09-16 17:57:58 +03:00
def enable(self, *args):
current_song=self._client.currentsong()
2020-09-16 17:57:58 +03:00
if current_song == {}:
if self._displayed_song_file is not None:
self._refresh()
else:
if current_song["file"] != self._displayed_song_file:
self._refresh()
self._client.emitter.handler_unblock(self._song_changed)
GLib.idle_add(self._text_view.grab_focus) # focus textview
def disable(self, *args):
self._client.emitter.handler_block(self._song_changed)
2020-01-11 13:25:15 +03:00
2020-09-24 17:50:19 +03:00
def _get_lyrics(self, title, artist):
2020-09-24 22:17:10 +03:00
replaces=((" ", "+"),(".", "_"),("@", "_"),(",", "_"),(";", "_"),("&", "_"),("\\", "_"),("/", "_"),('"', "_"),("(", "_"),(")", "_"))
2020-09-24 17:50:19 +03:00
for char1, char2 in replaces:
title=title.replace(char1, char2)
artist=artist.replace(char1, char2)
2020-09-24 22:17:10 +03:00
req=requests.get("https://www.letras.mus.br/winamp.php?musica={0}&artista={1}".format(title,artist))
soup=BeautifulSoup(req.text, "html.parser")
2020-09-24 17:50:19 +03:00
soup=soup.find(id="letra-cnt")
if soup is None:
raise ValueError("Not found")
paragraphs=[i for i in soup.children][1] # remove unneded paragraphs (NavigableString)
lyrics=""
for paragraph in paragraphs:
for line in paragraph.stripped_strings:
2020-09-24 22:17:10 +03:00
lyrics+=line+"\n"
lyrics+="\n"
2020-09-24 17:50:19 +03:00
output=lyrics[:-2] # omit last two newlines
if output == "": # assume song is instrumental when lyrics are empty
return "Instrumental"
else:
return output
2020-09-16 16:08:56 +03:00
def _display_lyrics(self, current_song):
2021-07-20 21:17:56 +03:00
GLib.idle_add(self._text_buffer.set_text, _("searching…"), -1)
2020-09-24 17:50:19 +03:00
try:
text=self._get_lyrics(current_song.get("title", ""), current_song.get("artist", ""))
except requests.exceptions.ConnectionError:
self._displayed_song_file=None
text=_("connection error")
except ValueError:
2020-09-23 13:01:10 +03:00
text=_("lyrics not found")
2020-09-16 16:08:56 +03:00
GLib.idle_add(self._text_buffer.set_text, text, -1)
def _refresh(self, *args):
current_song=self._client.currentsong()
2020-09-16 17:57:58 +03:00
if current_song == {}:
self._displayed_song_file=None
self._text_buffer.set_text("", -1)
2020-09-16 17:57:58 +03:00
else:
self._displayed_song_file=current_song["file"]
update_thread=threading.Thread(
target=self._display_lyrics,
kwargs={"current_song": ClientHelper.song_to_first_str_dict(current_song)},
daemon=True
)
update_thread.start()
2020-01-11 13:25:15 +03:00
2020-09-16 17:57:58 +03:00
def _on_disconnected(self, *args):
self._displayed_song_file=None
self._text_buffer.set_text("", -1)
2020-01-11 13:25:15 +03:00
2020-09-16 21:56:43 +03:00
class CoverEventBox(Gtk.EventBox):
2020-12-06 02:45:25 +03:00
def __init__(self, client, settings):
2020-09-16 16:08:56 +03:00
super().__init__()
self._client=client
self._settings=settings
2021-03-24 19:35:05 +03:00
# album popover
self._album_popover=AlbumPopover(self._client, self._settings)
2020-01-11 13:25:15 +03:00
2020-07-04 13:35:39 +03:00
# connect
2020-09-16 21:56:43 +03:00
self._button_press_event=self.connect("button-press-event", self._on_button_press_event)
2021-03-19 22:55:41 +03:00
self._client.emitter.connect("disconnected", self._on_disconnected)
2020-06-26 21:24:06 +03:00
2020-09-16 16:08:56 +03:00
def _on_button_press_event(self, widget, event):
if self._settings.get_property("mini-player"):
if event.button == 1 and event.type == Gdk.EventType.BUTTON_PRESS:
window=self.get_toplevel()
window.begin_move_drag(1, event.x_root, event.y_root, Gdk.CURRENT_TIME)
2020-09-16 21:56:43 +03:00
else:
if self._client.connected():
song=ClientHelper.song_to_first_str_dict(self._client.currentsong())
if song != {}:
try:
artist=song[self._settings.get_artist_type()]
except:
artist=song.get("artist", "")
album=song.get("album", "")
2021-04-06 21:59:25 +03:00
year=song.get("date", "")
if event.button == 1 and event.type == Gdk.EventType.BUTTON_PRESS:
self._client.album_to_playlist(album, artist, year, None)
elif event.button == 1 and event.type == Gdk.EventType._2BUTTON_PRESS:
self._client.album_to_playlist(album, artist, year, None, "play")
elif event.button == 2 and event.type == Gdk.EventType.BUTTON_PRESS:
self._client.album_to_playlist(album, artist, year, None, "append")
elif event.button == 3 and event.type == Gdk.EventType.BUTTON_PRESS:
self._album_popover.open(album, artist, year, None, widget, event.x, event.y)
2021-03-19 22:55:41 +03:00
def _on_disconnected(self, *args):
2021-03-24 19:35:05 +03:00
self._album_popover.popdown()
2020-09-16 21:56:43 +03:00
2021-04-02 18:45:52 +03:00
class MainCover(Gtk.Image):
2020-09-16 21:56:43 +03:00
def __init__(self, client, settings):
super().__init__()
2021-03-27 14:50:45 +03:00
self._client=client
self._settings=settings
2020-09-16 21:56:43 +03:00
# set default size
2021-04-02 18:45:52 +03:00
size=self._settings.get_int("track-cover")
self.set_size_request(size, size)
2020-09-16 21:56:43 +03:00
# connect
self._client.emitter.connect("current_song_changed", self._refresh)
self._client.emitter.connect("disconnected", self._on_disconnected)
self._client.emitter.connect("reconnected", self._on_reconnected)
self._settings.connect("changed::track-cover", self._on_settings_changed)
def _refresh(self, *args):
current_song=self._client.currentsong()
self.set_from_pixbuf(self._client.get_cover(current_song, self._settings.get_int("track-cover")))
2020-09-16 21:56:43 +03:00
2020-09-16 16:08:56 +03:00
def _on_disconnected(self, *args):
size=self._settings.get_int("track-cover")
2021-06-02 22:34:01 +03:00
self.set_from_pixbuf(GdkPixbuf.Pixbuf.new_from_file_at_size(FALLBACK_COVER, size, size))
2021-04-02 18:45:52 +03:00
self.set_sensitive(False)
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
def _on_reconnected(self, *args):
2021-04-02 18:45:52 +03:00
self.set_sensitive(True)
2020-09-16 16:08:56 +03:00
def _on_settings_changed(self, *args):
size=self._settings.get_int("track-cover")
2021-04-02 18:45:52 +03:00
self.set_size_request(size, size)
2020-09-16 16:08:56 +03:00
self._refresh()
2020-03-24 18:14:01 +03:00
2021-07-16 20:39:48 +03:00
class PlaylistWindow(Gtk.Overlay):
selected_path=GObject.Property(type=Gtk.TreePath, default=None) # currently marked song (bold text)
2020-09-16 16:08:56 +03:00
def __init__(self, client, settings):
2021-07-16 20:39:48 +03:00
super().__init__()
self._client=client
self._settings=settings
2020-09-16 16:08:56 +03:00
self._playlist_version=None
self._inserted_path=None # needed for drag and drop
2020-03-04 18:39:59 +03:00
2021-07-16 20:39:48 +03:00
# back button
2020-09-16 16:08:56 +03:00
self._back_to_current_song_button=Gtk.Button(
2021-07-16 20:39:48 +03:00
image=Gtk.Image.new_from_icon_name("go-previous-symbolic", Gtk.IconSize.LARGE_TOOLBAR),
2020-09-16 16:08:56 +03:00
tooltip_text=_("Scroll to current song"),
can_focus=False
2020-09-12 15:31:17 +03:00
)
2021-07-16 20:39:48 +03:00
self._back_to_current_song_button.set_margin_bottom(12)
self._back_to_current_song_button.set_margin_start(12)
2020-09-16 16:08:56 +03:00
style_context=self._back_to_current_song_button.get_style_context()
2021-07-16 20:39:48 +03:00
style_context.add_class("osd")
self._back_button_revealer=Gtk.Revealer(transition_duration=0)
self._back_button_revealer.set_halign(Gtk.Align.START)
self._back_button_revealer.set_valign(Gtk.Align.END)
self._back_button_revealer.add(self._back_to_current_song_button)
2020-03-24 18:14:01 +03:00
2020-10-13 18:12:54 +03:00
# treeview
2021-06-27 20:08:32 +03:00
# (track, disc, title, artist, album, human duration, date, genre, file, weight, duration)
self._store=Gtk.ListStore(str, str, str, str, str, str, str, str, str, Pango.Weight, float)
self._treeview=Gtk.TreeView(model=self._store,activate_on_single_click=True,reorderable=True,search_column=2,fixed_height_mode=True)
2020-09-16 16:08:56 +03:00
self._selection=self._treeview.get_selection()
2020-03-24 18:14:01 +03:00
2020-10-13 18:12:54 +03:00
# columns
2020-09-16 16:08:56 +03:00
renderer_text=Gtk.CellRendererText(ellipsize=Pango.EllipsizeMode.END, ellipsize_set=True)
renderer_text_ralign=Gtk.CellRendererText(xalign=1.0)
2020-10-13 18:12:54 +03:00
self._columns=(
Gtk.TreeViewColumn(_("No"), renderer_text_ralign, text=0, weight=9),
Gtk.TreeViewColumn(_("Disc"), renderer_text_ralign, text=1, weight=9),
Gtk.TreeViewColumn(_("Title"), renderer_text, text=2, weight=9),
Gtk.TreeViewColumn(_("Artist"), renderer_text, text=3, weight=9),
Gtk.TreeViewColumn(_("Album"), renderer_text, text=4, weight=9),
Gtk.TreeViewColumn(_("Length"), renderer_text, text=5, weight=9),
Gtk.TreeViewColumn(_("Year"), renderer_text, text=6, weight=9),
Gtk.TreeViewColumn(_("Genre"), renderer_text, text=7, weight=9)
)
2021-02-07 21:05:30 +03:00
for i, column in enumerate(self._columns):
2020-09-16 16:08:56 +03:00
column.set_property("resizable", True)
column.set_property("sizing", Gtk.TreeViewColumnSizing.FIXED)
2020-09-16 16:08:56 +03:00
column.set_min_width(30)
column.connect("notify::fixed-width", self._on_column_width, i)
2020-09-16 16:08:56 +03:00
self._load_settings()
2020-08-21 19:03:36 +03:00
2020-09-16 16:08:56 +03:00
# scroll
scroll=Gtk.ScrolledWindow()
scroll.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
scroll.add(self._treeview)
2020-09-16 16:08:56 +03:00
# frame
self._frame=FocusFrame()
self._frame.set_widget(self._treeview)
self._frame.add(scroll)
2020-01-11 13:25:15 +03:00
2021-03-24 19:35:05 +03:00
# song popover
self._song_popover=SongPopover(self._client, show_buttons=False)
2021-03-24 19:35:05 +03:00
2020-09-16 16:08:56 +03:00
# connect
self._treeview.connect("row-activated", self._on_row_activated)
self._treeview.connect("button-press-event", self._on_button_press_event)
self._treeview.connect("key-release-event", self._on_key_release_event)
self._treeview.connect("drag-begin", lambda *args: self._frame.disable())
self._treeview.connect("drag-end", lambda *args: self._frame.enable())
self._row_deleted=self._store.connect("row-deleted", self._on_row_deleted)
self._row_inserted=self._store.connect("row-inserted", self._on_row_inserted)
2020-09-16 16:08:56 +03:00
self._back_to_current_song_button.connect("clicked", self._on_back_to_current_song_button_clicked)
2021-07-16 20:39:48 +03:00
scroll.get_vadjustment().connect("value-changed", self._on_show_hide_back_button)
self.connect("notify::selected-path", self._on_show_hide_back_button)
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
self._client.emitter.connect("playlist_changed", self._on_playlist_changed)
self._client.emitter.connect("current_song_changed", self._on_song_changed)
self._client.emitter.connect("disconnected", self._on_disconnected)
self._client.emitter.connect("reconnected", self._on_reconnected)
2021-03-26 18:39:21 +03:00
self._client.emitter.connect("show-info", self._on_show_info)
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
self._settings.connect("notify::mini-player", self._on_mini_player)
self._settings.connect("changed::column-visibilities", self._load_settings)
self._settings.connect("changed::column-permutation", self._load_settings)
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
# packing
2021-07-16 20:39:48 +03:00
self.add(self._frame)
self.add_overlay(self._back_button_revealer)
2020-08-04 20:37:54 +03:00
2021-02-07 21:05:30 +03:00
def _on_column_width(self, obj, typestring, pos):
self._settings.array_modify("ai", "column-sizes", pos, obj.get_property("fixed-width"))
2020-06-27 17:11:41 +03:00
2020-09-16 16:08:56 +03:00
def _load_settings(self, *args):
columns=self._treeview.get_columns()
for column in columns:
self._treeview.remove_column(column)
sizes=self._settings.get_value("column-sizes").unpack()
visibilities=self._settings.get_value("column-visibilities").unpack()
for i in self._settings.get_value("column-permutation"):
if sizes[i] > 0:
self._columns[i].set_fixed_width(sizes[i])
self._columns[i].set_visible(visibilities[i])
self._treeview.append_column(self._columns[i])
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
def _clear(self, *args):
2021-07-04 19:37:14 +03:00
self._song_popover.popdown()
2021-07-16 20:39:48 +03:00
self._set_playlist_info("")
2020-09-16 16:08:56 +03:00
self._playlist_version=None
2021-07-16 20:39:48 +03:00
self.set_property("selected-path", None)
2020-10-25 13:43:02 +03:00
self._store.handler_block(self._row_inserted)
self._store.handler_block(self._row_deleted)
self._store.clear()
self._store.handler_unblock(self._row_inserted)
self._store.handler_unblock(self._row_deleted)
2020-01-11 13:25:15 +03:00
2021-06-27 20:08:32 +03:00
def _select(self, path):
self._unselect()
try:
self._store[path][9]=Pango.Weight.BOLD
2021-07-16 20:39:48 +03:00
self.set_property("selected-path", path)
2021-06-27 20:08:32 +03:00
except IndexError: # invalid path
pass
def _unselect(self):
2021-07-16 20:39:48 +03:00
if self.get_property("selected-path") is not None:
2021-06-27 20:08:32 +03:00
try:
2021-07-16 20:39:48 +03:00
self._store[self.get_property("selected-path")][9]=Pango.Weight.BOOK
self.set_property("selected-path", None)
2021-06-27 20:08:32 +03:00
except IndexError: # invalid path
2021-07-16 20:39:48 +03:00
self.set_property("selected-path", None)
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
def _scroll_to_selected_title(self, *args):
treeview, treeiter=self._selection.get_selected()
if treeiter is not None:
path=treeview.get_path(treeiter)
self._treeview.scroll_to_cell(path, None, True, 0.25)
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
def _refresh_selection(self): # Gtk.TreePath(len(self._store) is used to generate an invalid TreePath (needed to unset cursor)
self._treeview.set_cursor(Gtk.TreePath(len(self._store)), None, False)
2021-06-27 20:08:32 +03:00
song=self._client.status().get("song")
if song is None:
self._selection.unselect_all()
self._unselect()
else:
2020-09-16 16:08:56 +03:00
path=Gtk.TreePath(int(song))
self._selection.select_path(path)
2021-06-27 20:08:32 +03:00
self._select(path)
2020-01-11 13:25:15 +03:00
2021-07-16 20:39:48 +03:00
def _set_playlist_info(self, text):
if text == "":
self._columns[2].set_title(_("Title"))
else:
self._columns[2].set_title(" • ".join([_("Title"), text]))
def _on_button_press_event(self, widget, event):
path_re=widget.get_path_at_pos(int(event.x), int(event.y))
if path_re is not None:
path=path_re[0]
2021-03-26 21:19:34 +03:00
if event.button == 2 and event.type == Gdk.EventType.BUTTON_PRESS:
self._store.remove(self._store.get_iter(path))
elif event.button == 3 and event.type == Gdk.EventType.BUTTON_PRESS:
self._song_popover.open(self._store[path][8], widget, int(event.x), int(event.y))
2021-07-16 20:39:48 +03:00
def _on_show_hide_back_button(self, *args):
visible_range=self._treeview.get_visible_range()
if visible_range is None or self.get_property("selected-path") is None:
self._back_button_revealer.set_reveal_child(False)
else:
current_song_visible=(visible_range[0] <= self.get_property("selected-path") <= visible_range[1])
self._back_button_revealer.set_reveal_child(not(current_song_visible))
def _on_key_release_event(self, widget, event):
2020-09-17 13:52:07 +03:00
if event.keyval == Gdk.keyval_from_name("Delete"):
2020-09-16 16:08:56 +03:00
treeview, treeiter=self._selection.get_selected()
if treeiter is not None:
try:
self._store.remove(treeiter)
2020-09-16 16:08:56 +03:00
except:
pass
2020-01-11 13:25:15 +03:00
def _on_row_deleted(self, model, path): # sync treeview to mpd
2020-12-22 14:33:21 +03:00
try:
if self._inserted_path is not None: # move
path=int(path.to_string())
if path > self._inserted_path:
path=path-1
if path < self._inserted_path:
self._inserted_path=self._inserted_path-1
self._client.move(path, self._inserted_path)
2020-12-22 14:33:21 +03:00
self._inserted_path=None
else: # delete
self._client.delete(path) # bad song index possible
self._playlist_version=int(self._client.status()["playlist"])
2020-12-22 14:33:21 +03:00
except MPDBase.CommandError as e:
self._playlist_version=None
self._client.emitter.emit("playlist_changed", int(self._client.status()["playlist"]))
2020-12-22 14:33:21 +03:00
raise e # propagate exception
def _on_row_inserted(self, model, path, treeiter):
self._inserted_path=int(path.to_string())
2020-09-16 16:08:56 +03:00
def _on_row_activated(self, widget, path, view_column):
self._client.play(path)
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
def _on_playlist_changed(self, emitter, version):
self._store.handler_block(self._row_inserted)
self._store.handler_block(self._row_deleted)
2021-07-04 19:37:14 +03:00
self._song_popover.popdown()
2021-06-27 20:08:32 +03:00
self._unselect()
2020-09-16 16:08:56 +03:00
songs=[]
if self._playlist_version is not None:
songs=self._client.plchanges(self._playlist_version)
2020-09-16 16:08:56 +03:00
else:
songs=self._client.playlistinfo()
2020-09-16 16:08:56 +03:00
if songs != []:
2021-06-27 20:08:32 +03:00
self._treeview.freeze_child_notify()
2021-07-16 20:39:48 +03:00
self._set_playlist_info("")
2020-09-16 16:08:56 +03:00
for s in songs:
2020-10-17 12:30:00 +03:00
song=ClientHelper.song_to_str_dict(ClientHelper.pepare_song_for_display(s))
2020-09-16 16:08:56 +03:00
try:
treeiter=self._store.get_iter(song["pos"])
self._store.set(treeiter,
0, song["track"],
1, song["disc"],
2, song["title"],
3, song["artist"],
4, song["album"],
5, song["human_duration"],
6, song["date"],
7, song["genre"],
8, song["file"],
2021-06-27 20:08:32 +03:00
9, Pango.Weight.BOOK,
10, float(song["duration"])
2020-09-16 16:08:56 +03:00
)
except:
self._store.append([
song["track"], song["disc"],
song["title"], song["artist"],
song["album"], song["human_duration"],
song["date"], song["genre"],
2021-06-27 20:08:32 +03:00
song["file"], Pango.Weight.BOOK,
float(song["duration"])
2020-09-16 16:08:56 +03:00
])
2021-06-27 20:08:32 +03:00
self._treeview.thaw_child_notify()
for i in reversed(range(int(self._client.status()["playlistlength"]), len(self._store))):
2020-09-16 16:08:56 +03:00
treeiter=self._store.get_iter(i)
self._store.remove(treeiter)
2021-06-27 20:08:32 +03:00
playlist_length=len(self._store)
if playlist_length == 0:
2021-07-16 20:39:48 +03:00
self._set_playlist_info("")
2021-06-27 20:08:32 +03:00
else:
2021-07-16 20:39:48 +03:00
duration_human_readable=ClientHelper.seconds_to_display_time(int(sum([row[10] for row in self._store])))
translated_string=ngettext("{number} song ({duration})", "{number} songs ({duration})", playlist_length)
self._set_playlist_info(translated_string.format(number=playlist_length, duration=duration_human_readable))
2021-06-27 20:08:32 +03:00
self._refresh_selection()
if self._playlist_version != version:
2020-09-16 16:08:56 +03:00
self._scroll_to_selected_title()
self._playlist_version=version
self._store.handler_unblock(self._row_inserted)
self._store.handler_unblock(self._row_deleted)
2020-06-27 17:11:41 +03:00
2020-09-16 16:08:56 +03:00
def _on_song_changed(self, *args):
self._refresh_selection()
if self._client.status()["state"] == "play":
2020-09-16 16:08:56 +03:00
self._scroll_to_selected_title()
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
def _on_back_to_current_song_button_clicked(self, *args):
self._treeview.set_cursor(Gtk.TreePath(len(self._store)), None, False) # set to invalid TreePath (needed to unset cursor)
2021-07-16 20:39:48 +03:00
if self.get_property("selected-path") is not None:
self._selection.select_path(self.get_property("selected-path"))
2020-09-16 16:08:56 +03:00
self._scroll_to_selected_title()
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
def _on_disconnected(self, *args):
self._treeview.set_sensitive(False)
self._back_to_current_song_button.set_sensitive(False)
2020-10-25 13:43:02 +03:00
self._clear()
2020-01-11 13:25:15 +03:00
2020-09-16 16:08:56 +03:00
def _on_reconnected(self, *args):
self._back_to_current_song_button.set_sensitive(True)
self._treeview.set_sensitive(True)
2021-03-26 18:39:21 +03:00
def _on_show_info(self, *args):
if self._treeview.has_focus():
treeview, treeiter=self._selection.get_selected()
if treeiter is not None:
path=self._store.get_path(treeiter)
cell=self._treeview.get_cell_area(path, None)
self._song_popover.open(self._store[path][8], self._treeview, int(cell.x), int(cell.y))
2020-09-16 16:08:56 +03:00
def _on_mini_player(self, obj, typestring):
2021-02-08 19:49:45 +03:00
if obj.get_property("mini-player"):
self.set_property("no-show-all", True)
self.set_property("visible", False)
else:
self.set_property("no-show-all", False)
self.show_all()
2020-03-28 15:23:56 +03:00
class CoverPlaylistWindow(Gtk.Paned):
def __init__(self, client, settings):
2020-09-16 16:08:56 +03:00
super().__init__()
self._client=client
self._settings=settings
2020-09-16 16:08:56 +03:00
# cover
2020-09-16 21:56:43 +03:00
main_cover=MainCover(self._client, self._settings)
2020-12-06 02:45:25 +03:00
self._cover_event_box=CoverEventBox(self._client, self._settings)
2020-03-28 15:23:56 +03:00
# playlist
self._playlist_window=PlaylistWindow(self._client, self._settings)
2020-09-16 16:08:56 +03:00
# lyrics button
self.lyrics_button=Gtk.ToggleButton(
2020-09-16 16:08:56 +03:00
image=Gtk.Image.new_from_icon_name("media-view-subtitles-symbolic", Gtk.IconSize.BUTTON),
2020-09-29 14:02:51 +03:00
tooltip_text=_("Show lyrics")
2020-09-16 16:08:56 +03:00
)
2020-09-29 14:02:51 +03:00
self.lyrics_button.set_can_focus(False)
self.lyrics_button.set_margin_top(6)
self.lyrics_button.set_margin_end(6)
style_context=self.lyrics_button.get_style_context()
2020-09-16 16:08:56 +03:00
style_context.add_class("circular")
2020-03-28 15:23:56 +03:00
2020-09-16 17:57:58 +03:00
# lyrics window
self._lyrics_window=LyricsWindow(self._client, self._settings)
2020-09-16 16:08:56 +03:00
# revealer
2020-09-16 17:57:58 +03:00
self._lyrics_button_revealer=Gtk.Revealer()
self._lyrics_button_revealer.set_halign(Gtk.Align.END)
self._lyrics_button_revealer.set_valign(Gtk.Align.START)
self._lyrics_button_revealer.add(self.lyrics_button)
2021-07-17 14:58:19 +03:00
self._settings.bind("show-lyrics-button", self._lyrics_button_revealer, "reveal-child", Gio.SettingsBindFlags.DEFAULT)
2020-01-11 13:25:15 +03:00
2020-09-16 21:56:43 +03:00
# stack
2021-03-12 21:18:50 +03:00
self._stack=Gtk.Stack(transition_type=Gtk.StackTransitionType.OVER_DOWN_UP)
2020-09-16 21:56:43 +03:00
self._stack.add_named(self._cover_event_box, "cover")
self._stack.add_named(self._lyrics_window, "lyrics")
self._stack.set_visible_child(self._cover_event_box)
2020-09-16 16:08:56 +03:00
# connect
self.lyrics_button.connect("toggled", self._on_lyrics_toggled)
2020-09-16 16:08:56 +03:00
self._client.emitter.connect("disconnected", self._on_disconnected)
self._client.emitter.connect("reconnected", self._on_reconnected)
2020-03-03 18:11:30 +03:00
# packing
overlay=Gtk.Overlay()
overlay.add(main_cover)
overlay.add_overlay(self._stack)
overlay.add_overlay(self._lyrics_button_revealer)
self.pack1(overlay, False, False)
self.pack2(self._playlist_window, True, False)
2020-09-16 16:08:56 +03:00
def _on_reconnected(self, *args):
self.lyrics_button.set_sensitive(True)
2020-09-16 16:08:56 +03:00
def _on_disconnected(self, *args):
self.lyrics_button.set_active(False)
self.lyrics_button.set_sensitive(False)
2020-02-16 14:20:38 +03:00
2020-09-16 17:57:58 +03:00
def _on_lyrics_toggled(self, widget):
if widget.get_active():
2020-09-16 21:56:43 +03:00
self._stack.set_visible_child(self._lyrics_window)
2020-09-16 17:57:58 +03:00
self._lyrics_window.enable()
else:
2020-09-16 21:56:43 +03:00
self._stack.set_visible_child(self._cover_event_box)
2020-09-16 17:57:58 +03:00
self._lyrics_window.disable()
2020-09-16 16:08:56 +03:00
2021-07-16 20:39:48 +03:00
######################
# action bar widgets #
######################
2020-07-04 14:16:17 +03:00
2020-08-21 13:40:48 +03:00
class PlaybackControl(Gtk.ButtonBox):
2020-03-30 12:54:04 +03:00
def __init__(self, client, settings):
2020-08-31 16:49:06 +03:00
super().__init__(layout_style=Gtk.ButtonBoxStyle.EXPAND)
self._client=client
self._settings=settings
2020-01-11 13:25:15 +03:00
2020-07-04 13:35:39 +03:00
# widgets
2021-07-17 15:07:04 +03:00
self._play_button_icon=AutoSizedIcon("media-playback-start-symbolic", "icon-size", self._settings)
self._play_button=Gtk.Button(image=self._play_button_icon)
self._play_button.set_action_name("mpd.toggle-play")
2020-09-29 14:02:51 +03:00
self._play_button.set_can_focus(False)
2020-10-20 14:13:50 +03:00
self._stop_button=Gtk.Button(image=AutoSizedIcon("media-playback-stop-symbolic", "icon-size", self._settings))
self._stop_button.set_property("no-show-all", not(self._settings.get_boolean("show-stop")))
self._stop_button.set_action_name("mpd.stop")
2020-09-29 14:02:51 +03:00
self._stop_button.set_can_focus(False)
2020-10-20 14:13:50 +03:00
self._prev_button=Gtk.Button(image=AutoSizedIcon("media-skip-backward-symbolic", "icon-size", self._settings))
self._prev_button.set_action_name("mpd.prev")
2020-09-29 14:02:51 +03:00
self._prev_button.set_can_focus(False)
2020-10-20 14:13:50 +03:00
self._next_button=Gtk.Button(image=AutoSizedIcon("media-skip-forward-symbolic", "icon-size", self._settings))
self._next_button.set_action_name("mpd.next")
2020-09-29 14:02:51 +03:00
self._next_button.set_can_focus(False)
2020-01-11 13:25:15 +03:00
2020-07-04 13:35:39 +03:00
# connect
2020-09-15 19:45:30 +03:00
self._settings.connect("notify::mini-player", self._on_mini_player)
self._settings.connect("changed::show-stop", self._on_show_stop_changed)
self._client.emitter.connect("state", self._on_state)
self._client.emitter.connect("playlist_changed", self._refresh_tooltips)
self._client.emitter.connect("current_song_changed", self._refresh_tooltips)
2020-09-29 13:39:21 +03:00
self._client.emitter.connect("disconnected", self._on_disconnected)
2020-01-11 13:25:15 +03:00
2020-07-04 13:35:39 +03:00
# packing
2021-04-24 12:42:35 +03:00
self.pack_start(self._prev_button, True, True, 0)
self.pack_start(self._play_button, True, True, 0)
self.pack_start(self._stop_button, True, True, 0)
self.pack_start(self._next_button, True, True, 0)
2020-01-11 13:25:15 +03:00
def _refresh_tooltips(self, *args):
2021-06-27 20:08:32 +03:00
status=self._client.status()
song=status.get("song")
length=status.get("playlistlength")
if song is None or length is None:
self._prev_button.set_tooltip_text("")
self._next_button.set_tooltip_text("")
2021-06-27 20:08:32 +03:00
else:
elapsed=int(song)
rest=int(length)-elapsed-1
2021-07-17 15:15:31 +03:00
elapsed_songs=ngettext("{number} song", "{number} songs", elapsed).format(number=elapsed)
rest_songs=ngettext("{number} song", "{number} songs", rest).format(number=rest)
self._prev_button.set_tooltip_text(elapsed_songs)
self._next_button.set_tooltip_text(rest_songs)
2020-09-29 13:39:21 +03:00
def _on_state(self, emitter, state):
if state == "play":
2021-07-17 15:07:04 +03:00
self._play_button_icon.set_property("icon-name", "media-playback-pause-symbolic")
2020-09-29 13:39:21 +03:00
else:
2021-07-17 15:07:04 +03:00
self._play_button_icon.set_property("icon-name", "media-playback-start-symbolic")
2020-09-29 13:39:21 +03:00
def _on_disconnected(self, *args):
self._prev_button.set_tooltip_text("")
self._next_button.set_tooltip_text("")
2020-09-29 13:39:21 +03:00
2020-09-15 19:45:30 +03:00
def _on_mini_player(self, obj, typestring):
self._on_show_stop_changed()
def _on_show_stop_changed(self, *args):
2020-09-15 19:45:30 +03:00
visibility=(self._settings.get_boolean("show-stop") and not self._settings.get_property("mini-player"))
2020-09-29 13:39:21 +03:00
self._stop_button.set_property("visible", visibility)
self._stop_button.set_property("no-show-all", not(visibility))
2020-01-11 13:25:15 +03:00
2020-07-04 14:16:17 +03:00
class SeekBar(Gtk.Box):
2020-01-11 13:25:15 +03:00
def __init__(self, client):
super().__init__(hexpand=True)
self._client=client
self._update=True
self._jumped=False
2020-01-11 13:25:15 +03:00
2020-07-04 14:16:17 +03:00
# labels
self._elapsed=Gtk.Label(width_chars=5)
self._rest=Gtk.Label(width_chars=6)
2020-01-11 13:25:15 +03:00
2020-09-18 23:28:51 +03:00
# event boxes
self._elapsed_event_box=Gtk.EventBox()
self._rest_event_box=Gtk.EventBox()
2020-07-04 14:16:17 +03:00
# progress bar
2020-09-29 14:02:51 +03:00
self._scale=Gtk.Scale(orientation=Gtk.Orientation.HORIZONTAL)
self._scale.set_can_focus(False)
2020-09-29 13:39:21 +03:00
self._scale.set_show_fill_level(True)
self._scale.set_restrict_to_fill_level(False)
self._scale.set_draw_value(False)
self._scale.set_increments(10, 60)
self._adjustment=self._scale.get_adjustment()
2020-01-11 13:25:15 +03:00
2020-07-04 14:16:17 +03:00
# css (scale)
2020-09-29 13:39:21 +03:00
style_context=self._scale.get_style_context()
2020-07-04 14:16:17 +03:00
provider=Gtk.CssProvider()
css=b"""scale fill { background-color: @theme_selected_bg_color; }"""
provider.load_from_data(css)
2020-12-09 21:12:12 +03:00
style_context.add_provider(provider, 600)
2020-01-11 13:25:15 +03:00
2020-07-04 14:16:17 +03:00
# connect
2020-09-18 23:28:51 +03:00
self._elapsed_event_box.connect("button-release-event", self._on_elapsed_button_release_event)
self._rest_event_box.connect("button-release-event", self._on_rest_button_release_event)
2020-09-29 13:39:21 +03:00
self._scale.connect("change-value", self._on_change_value)
self._scale.connect("scroll-event", lambda *args: True) # disable mouse wheel
self._scale.connect("button-press-event", self._on_scale_button_press_event)
self._scale.connect("button-release-event", self._on_scale_button_release_event)
2020-09-18 23:28:51 +03:00
self._client.emitter.connect("disconnected", self._disable)
self._client.emitter.connect("state", self._on_state)
2020-08-21 19:03:36 +03:00
self._client.emitter.connect("elapsed_changed", self._refresh)
2020-07-04 14:16:17 +03:00
# packing
self._elapsed_event_box.add(self._elapsed)
self._rest_event_box.add(self._rest)
self.pack_start(self._elapsed_event_box, False, False, 0)
2020-09-29 13:39:21 +03:00
self.pack_start(self._scale, True, True, 0)
self.pack_end(self._rest_event_box, False, False, 0)
2020-01-11 13:25:15 +03:00
2020-08-21 19:03:36 +03:00
def _refresh(self, emitter, elapsed, duration):
2020-09-18 23:28:51 +03:00
self.set_sensitive(True)
2020-08-21 19:03:36 +03:00
if elapsed > duration: # fix display error
elapsed=duration
self._adjustment.set_upper(duration)
2020-08-21 19:03:36 +03:00
if self._update:
2020-09-29 13:39:21 +03:00
self._scale.set_value(elapsed)
2020-09-30 12:06:00 +03:00
self._elapsed.set_text(ClientHelper.seconds_to_display_time(int(elapsed)))
self._rest.set_text("-"+ClientHelper.seconds_to_display_time(int(duration-elapsed)))
2020-09-29 13:39:21 +03:00
self._scale.set_fill_level(elapsed)
2020-08-21 19:03:36 +03:00
def _disable(self, *args):
self.set_sensitive(False)
2020-09-29 13:39:21 +03:00
self._scale.set_fill_level(0)
self._scale.set_range(0, 0)
2020-09-30 12:06:00 +03:00
self._elapsed.set_text("")
self._rest.set_text("")
2020-08-21 19:03:36 +03:00
def _on_scale_button_press_event(self, widget, event):
2020-07-04 14:16:17 +03:00
if event.button == 1 and event.type == Gdk.EventType.BUTTON_PRESS:
self._update=False
2020-09-29 13:39:21 +03:00
self._scale.set_has_origin(False)
2020-11-01 15:00:10 +03:00
elif event.button == 3 and event.type == Gdk.EventType.BUTTON_PRESS:
self._jumped=False
2020-06-05 18:43:34 +03:00
def _on_scale_button_release_event(self, widget, event):
2020-07-04 14:16:17 +03:00
if event.button == 1:
self._update=True
2020-09-29 13:39:21 +03:00
self._scale.set_has_origin(True)
if self._jumped: # actual seek
self._client.seekcur(self._scale.get_value())
self._jumped=False
2020-09-18 23:28:51 +03:00
else: # restore state
status=self._client.status()
2020-08-21 19:03:36 +03:00
self._refresh(None, float(status["elapsed"]), float(status["duration"]))
2020-01-11 13:25:15 +03:00
2020-09-18 23:28:51 +03:00
def _on_change_value(self, range, scroll, value): # value is inaccurate (can be above upper limit)
if (scroll == Gtk.ScrollType.STEP_BACKWARD or scroll == Gtk.ScrollType.STEP_FORWARD or
scroll == Gtk.ScrollType.PAGE_BACKWARD or scroll == Gtk.ScrollType.PAGE_FORWARD):
self._client.seekcur(value)
2020-07-04 14:16:17 +03:00
elif scroll == Gtk.ScrollType.JUMP:
2020-09-18 23:28:51 +03:00
duration=self._adjustment.get_upper()
if value > duration: # fix display error
elapsed=duration
else:
elapsed=value
2020-09-30 12:06:00 +03:00
self._elapsed.set_text(ClientHelper.seconds_to_display_time(int(elapsed)))
self._rest.set_text("-"+ClientHelper.seconds_to_display_time(int(duration-elapsed)))
self._jumped=True
2020-01-11 13:25:15 +03:00
2020-09-18 23:28:51 +03:00
def _on_elapsed_button_release_event(self, widget, event):
if event.button == 1:
self._client.seekcur("-"+str(self._adjustment.get_property("step-increment")))
2020-09-18 23:28:51 +03:00
elif event.button == 3:
self._client.seekcur("+"+str(self._adjustment.get_property("step-increment")))
2020-09-18 23:28:51 +03:00
def _on_rest_button_release_event(self, widget, event):
if event.button == 1:
self._client.seekcur("+"+str(self._adjustment.get_property("step-increment")))
2020-09-18 23:28:51 +03:00
elif event.button == 3:
self._client.seekcur("-"+str(self._adjustment.get_property("step-increment")))
def _on_state(self, emitter, state):
if state == "stop":
self._disable()
2020-08-11 20:10:34 +03:00
2021-07-16 21:07:52 +03:00
class AudioFormat(Gtk.Box):
2020-07-04 14:16:17 +03:00
def __init__(self, client, settings):
2021-07-16 20:39:48 +03:00
super().__init__(spacing=6)
self._client=client
2021-07-16 20:39:48 +03:00
self._settings=settings
self._file_type_label=Gtk.Label(xalign=1)
self._separator_label=Gtk.Label(xalign=1)
2021-07-16 20:39:48 +03:00
self._brate_label=Gtk.Label(xalign=1, width_chars=5)
self._format_label=Gtk.Label()
2021-07-16 21:29:37 +03:00
self.set_property("no-show-all", not(self._settings.get_boolean("show-audio-format")))
2020-01-11 13:25:15 +03:00
2020-07-04 14:16:17 +03:00
# connect
2021-07-16 20:39:48 +03:00
self._settings.connect("notify::mini-player", self._on_mini_player)
2021-07-16 21:07:52 +03:00
self._settings.connect("changed::show-audio-format", self._on_show_audio_format_changed)
2021-07-16 20:39:48 +03:00
self._client.emitter.connect("audio", self._on_audio)
self._client.emitter.connect("bitrate", self._on_bitrate)
self._client.emitter.connect("current_song_changed", self._on_song_changed)
self._client.emitter.connect("disconnected", self._on_disconnected)
self._client.emitter.connect("reconnected", self._on_reconnected)
2021-07-16 20:39:48 +03:00
# packing
hbox=Gtk.Box(halign=Gtk.Align.END)
hbox.pack_start(self._brate_label, False, False, 0)
hbox.pack_start(self._separator_label, False, False, 0)
hbox.pack_start(self._file_type_label, False, False, 0)
2021-07-16 20:39:48 +03:00
vbox=Gtk.Box(orientation=Gtk.Orientation.VERTICAL, valign=Gtk.Align.CENTER)
vbox.pack_start(hbox, False, False, 0)
vbox.pack_start(self._format_label, False, False, 0)
self.pack_start(Gtk.Separator(), False, False, 0)
self.pack_start(vbox, False, False, 0)
self.pack_start(Gtk.Separator(), False, False, 0)
2021-07-16 20:39:48 +03:00
def _on_audio(self, emitter, audio_format):
if audio_format is None:
self._format_label.set_markup("<small> </small>")
else:
2021-07-16 20:39:48 +03:00
self._format_label.set_markup("<small>"+ClientHelper.convert_audio_format(audio_format)+"</small>")
2021-07-16 20:39:48 +03:00
def _on_bitrate(self, emitter, brate):
# handle unknown bitrates: https://github.com/MusicPlayerDaemon/MPD/issues/428#issuecomment-442430365
if brate is None:
self._brate_label.set_text("")
else:
self._brate_label.set_text(brate)
2021-07-16 20:39:48 +03:00
def _on_song_changed(self, *args):
current_song=self._client.currentsong()
if current_song == {}:
self._file_type_label.set_text("")
self._separator_label.set_text(" kb/s")
2021-07-16 20:39:48 +03:00
self._format_label.set_markup("<small> </small>")
else:
file_type=current_song["file"].split(".")[-1].split("/")[0].upper()
self._separator_label.set_text(" kb/s • ")
self._file_type_label.set_text(file_type)
2021-07-16 20:39:48 +03:00
def _on_mini_player(self, obj, typestring):
2021-07-16 21:07:52 +03:00
self._on_show_audio_format_changed()
2021-07-16 20:39:48 +03:00
2021-07-16 21:07:52 +03:00
def _on_show_audio_format_changed(self, *args):
visibility=(self._settings.get_boolean("show-audio-format") and not self._settings.get_property("mini-player"))
2021-07-16 20:39:48 +03:00
self.set_property("no-show-all", not(visibility))
2021-07-16 21:37:44 +03:00
if visibility:
self.show_all()
else:
self.hide()
def _on_disconnected(self, *args):
self.set_sensitive(False)
self._brate_label.set_text("")
self._separator_label.set_text(" kb/s")
self._file_type_label.set_text("")
2021-07-16 20:39:48 +03:00
self._format_label.set_markup("<small> </small>")
def _on_reconnected(self, *args):
self.set_sensitive(True)
class PlaybackOptions(Gtk.ButtonBox):
def __init__(self, client, settings):
super().__init__(layout_style=Gtk.ButtonBoxStyle.EXPAND)
self._client=client
self._settings=settings
# buttons
self._buttons={}
data=(
("repeat", "media-playlist-repeat-symbolic", _("Repeat mode")),
2021-04-24 11:24:59 +03:00
("random", "media-playlist-shuffle-symbolic", _("Random mode")),
("single", "org.mpdevil.mpdevil-single-symbolic", _("Single mode")),
("consume", "org.mpdevil.mpdevil-consume-symbolic", _("Consume mode")),
)
for name, icon, tooltip in data:
2021-04-24 11:24:59 +03:00
button=Gtk.ToggleButton(image=AutoSizedIcon(icon, "icon-size", self._settings), tooltip_text=tooltip, can_focus=False)
handler=button.connect("toggled", self._set_option, name)
2021-04-24 12:42:35 +03:00
self.pack_start(button, True, True, 0)
2021-04-24 11:24:59 +03:00
self._buttons[name]=(button, handler)
# css
self._provider=Gtk.CssProvider()
self._provider.load_from_data(b"""image {color: @error_color;}""") # red icon
# connect
2021-04-24 11:24:59 +03:00
for name in ("repeat", "random", "consume"):
self._client.emitter.connect(name, self._button_refresh, name)
self._client.emitter.connect("single", self._single_refresh)
2021-04-24 11:24:59 +03:00
self._buttons["single"][0].connect("button-press-event", self._on_single_button_press_event)
self._client.emitter.connect("disconnected", self._on_disconnected)
self._client.emitter.connect("reconnected", self._on_reconnected)
self._settings.connect("notify::mini-player", self._on_mini_player)
2021-04-24 11:24:59 +03:00
def _set_option(self, widget, option):
func=getattr(self._client, option)
if widget.get_active():
func("1")
else:
func("0")
2021-04-24 11:24:59 +03:00
def _button_refresh(self, emitter, val, name):
self._buttons[name][0].handler_block(self._buttons[name][1])
self._buttons[name][0].set_active(val)
self._buttons[name][0].handler_unblock(self._buttons[name][1])
def _single_refresh(self, emitter, val):
2021-04-24 11:24:59 +03:00
self._buttons["single"][0].handler_block(self._buttons["single"][1])
self._buttons["single"][0].set_active((val in ("1", "oneshot")))
if val == "oneshot":
2021-04-25 12:54:33 +03:00
self._buttons["single"][0].get_image().get_style_context().add_provider(self._provider, 600)
2020-11-01 15:27:23 +03:00
else:
2021-04-25 12:54:33 +03:00
self._buttons["single"][0].get_image().get_style_context().remove_provider(self._provider)
2021-04-24 11:24:59 +03:00
self._buttons["single"][0].handler_unblock(self._buttons["single"][1])
2020-11-01 15:27:23 +03:00
def _on_single_button_press_event(self, widget, event):
if event.button == 3 and event.type == Gdk.EventType.BUTTON_PRESS:
state=self._client.status()["single"]
2020-11-01 15:27:23 +03:00
if state == "oneshot":
self._client.single("0")
2020-11-01 15:27:23 +03:00
else:
self._client.single("oneshot")
2020-11-01 15:27:23 +03:00
def _on_disconnected(self, *args):
2021-04-24 11:24:59 +03:00
self.set_sensitive(False)
for name in ("repeat", "random", "consume"):
self._button_refresh(None, False, name)
2020-11-01 15:27:23 +03:00
self._single_refresh(None, "0")
2021-04-24 11:24:59 +03:00
def _on_reconnected(self, *args):
self.set_sensitive(True)
2020-08-13 18:40:27 +03:00
2020-09-15 19:45:30 +03:00
def _on_mini_player(self, obj, typestring):
2021-02-08 19:49:45 +03:00
if obj.get_property("mini-player"):
self.set_property("no-show-all", True)
self.set_property("visible", False)
2021-02-08 19:49:45 +03:00
else:
self.set_property("no-show-all", False)
self.show_all()
2020-09-15 19:45:30 +03:00
2021-07-16 20:39:48 +03:00
class OutputPopover(Gtk.Popover):
def __init__(self, client, relative):
super().__init__()
self.set_relative_to(relative)
self._client=client
# widgets
box=Gtk.Box(orientation=Gtk.Orientation.VERTICAL, border_width=9)
for output in self._client.outputs():
button=Gtk.ModelButton(label="{} ({})".format(output["outputname"], output["plugin"]), role=Gtk.ButtonRole.CHECK)
button.get_child().set_property("xalign", 0)
if output["outputenabled"] == "1":
button.set_property("active", True)
button.connect("clicked", self._on_button_clicked, output["outputid"])
box.pack_start(button, False, False, 0)
#connect
self.connect("closed", lambda *args: self.destroy())
# packing
self.add(box)
box.show_all()
def _on_button_clicked(self, button, out_id):
if button.get_property("active"):
self._client.disableoutput(out_id)
button.set_property("active", False)
else:
self._client.enableoutput(out_id)
button.set_property("active", True)
class VolumeButton(Gtk.VolumeButton):
def __init__(self, client, settings):
2021-07-16 20:46:39 +03:00
super().__init__(use_symbolic=True, can_focus=False)
2021-07-16 20:39:48 +03:00
self._client=client
self._popover=None
self._adj=self.get_adjustment()
self._adj.set_step_increment(0.05)
self._adj.set_page_increment(0.1)
self._adj.set_upper(0) # do not allow volume change by user when MPD has not yet reported volume (no output enabled/avail)
settings.bind("icon-size", self.get_child(), "pixel-size", Gio.SettingsBindFlags.GET)
# connect
self._changed=self.connect("value-changed", self._set_volume)
self._client.emitter.connect("volume_changed", self._refresh)
self._client.emitter.connect("disconnected", self._on_disconnected)
self._client.emitter.connect("reconnected", self._on_reconnected)
self.connect("button-press-event", self._on_button_press_event)
def _set_volume(self, widget, value):
self._client.setvol(str(int(value*100)))
def _refresh(self, emitter, volume):
self.handler_block(self._changed)
if volume < 0:
self.set_value(0)
self._adj.set_upper(0)
else:
self._adj.set_upper(1)
self.set_value(volume/100)
self.handler_unblock(self._changed)
def _on_button_press_event(self, widget, event):
if event.button == 3 and event.type == Gdk.EventType.BUTTON_PRESS:
self._popover=OutputPopover(self._client, self)
self._popover.popup()
def _on_reconnected(self, *args):
self.set_sensitive(True)
def _on_disconnected(self, *args):
self.set_sensitive(False)
self._refresh(None, -1)
if self._popover is not None:
self._popover.popdown()
self._popover=None
2020-09-29 13:39:21 +03:00
###################
# MPD gio actions #
###################
class MPDActionGroup(Gio.SimpleActionGroup):
def __init__(self, client):
super().__init__()
self._client=client
# actions
self._simple_actions_disable_on_stop_data=("next","prev","seek-forward","seek-backward")
self._simple_actions_enable_on_reconnect_data=("toggle-play","stop","clear","update","repeat","random","single","consume")
self._simple_actions_data=self._simple_actions_disable_on_stop_data+self._simple_actions_enable_on_reconnect_data
2020-09-29 13:39:21 +03:00
for name in self._simple_actions_data:
action=Gio.SimpleAction.new(name, None)
action.connect("activate", getattr(self, ("_on_"+name.replace("-","_"))))
self.add_action(action)
# connect
2020-09-30 12:21:41 +03:00
self._client.emitter.connect("state", self._on_state)
2020-09-29 13:39:21 +03:00
self._client.emitter.connect("disconnected", self._on_disconnected)
self._client.emitter.connect("reconnected", self._on_reconnected)
def _on_toggle_play(self, action, param):
self._client.toggle_play()
2020-09-29 13:39:21 +03:00
def _on_stop(self, action, param):
self._client.stop()
2020-09-29 13:39:21 +03:00
def _on_next(self, action, param):
self._client.next()
2020-09-29 13:39:21 +03:00
def _on_prev(self, action, param):
self._client.conditional_previous()
2020-09-29 13:39:21 +03:00
def _on_seek_forward(self, action, param):
self._client.seekcur("+10")
2020-09-29 13:39:21 +03:00
def _on_seek_backward(self, action, param):
self._client.seekcur("-10")
2020-09-29 13:39:21 +03:00
def _on_clear(self, action, param):
self._client.clear()
2020-09-29 13:39:21 +03:00
def _on_update(self, action, param):
self._client.update()
2020-09-29 13:39:21 +03:00
def _on_repeat(self, action, param):
self._client.toggle_option("repeat")
2020-09-29 13:39:21 +03:00
def _on_random(self, action, param):
self._client.toggle_option("random")
2020-09-29 13:39:21 +03:00
def _on_single(self, action, param):
self._client.toggle_option("single")
2020-09-29 13:39:21 +03:00
def _on_consume(self, action, param):
self._client.toggle_option("consume")
2020-09-29 13:39:21 +03:00
2020-09-30 12:21:41 +03:00
def _on_state(self, emitter, state):
state_dict={"play": True, "pause": True, "stop": False}
for action in self._simple_actions_disable_on_stop_data:
2020-09-30 12:21:41 +03:00
self.lookup_action(action).set_enabled(state_dict[state])
2020-09-29 13:39:21 +03:00
def _on_disconnected(self, *args):
for action in self._simple_actions_data:
self.lookup_action(action).set_enabled(False)
def _on_reconnected(self, *args):
for action in self._simple_actions_enable_on_reconnect_data:
2020-09-29 13:39:21 +03:00
self.lookup_action(action).set_enabled(True)
2020-09-16 20:15:17 +03:00
####################
# shortcuts window #
####################
class ShortcutsWindow(Gtk.ShortcutsWindow):
def __init__(self):
2020-09-29 15:01:31 +03:00
super().__init__()
2020-09-16 20:15:17 +03:00
general_group=Gtk.ShortcutsGroup(title=_("General"), visible=True)
window_group=Gtk.ShortcutsGroup(title=_("Window"), visible=True)
playback_group=Gtk.ShortcutsGroup(title=_("Playback"), visible=True)
2021-03-27 15:21:46 +03:00
items_group=Gtk.ShortcutsGroup(title=_("Search, Album Dialog, Album List and Artist List"), visible=True)
2020-09-17 13:31:25 +03:00
playlist_group=Gtk.ShortcutsGroup(title=_("Playlist"), visible=True)
2020-09-16 20:15:17 +03:00
section=Gtk.ShortcutsSection(section_name="shortcuts", visible=True)
section.add(general_group)
section.add(window_group)
section.add(playback_group)
2020-09-17 13:31:25 +03:00
section.add(items_group)
section.add(playlist_group)
2020-09-16 20:15:17 +03:00
2020-09-29 13:39:21 +03:00
shortcut_data=(
("F1", _("Open online help"), None, general_group),
("<Control>question", _("Open shortcuts window"), None, general_group),
("F10", _("Open menu"), None, general_group),
("F5", _("Update database"), None, general_group),
("<Control>q", _("Quit"), None, general_group),
2020-09-30 14:00:24 +03:00
("<Control>p", _("Cycle through profiles"), None, window_group),
("<Shift><Control>p", _("Cycle through profiles in reversed order"), None, window_group),
2020-09-17 13:31:25 +03:00
("<Control>m", _("Toggle mini player"), None, window_group),
("<Control>l", _("Toggle lyrics"), None, window_group),
2020-09-17 13:38:54 +03:00
("<Control>f", _("Toggle search"), None, window_group),
2020-09-17 13:31:25 +03:00
("Escape", _("Back to current album"), None, window_group),
("space", _("Play/Pause"), None, playback_group),
2020-09-29 13:39:21 +03:00
("<Control>space", _("Stop"), None, playback_group),
2020-09-17 13:31:25 +03:00
("KP_Add", _("Next title"), None, playback_group),
("KP_Subtract", _("Previous title"), None, playback_group),
("KP_Multiply", _("Seek forward"), None, playback_group),
("KP_Divide", _("Seek backward"), None, playback_group),
2020-09-29 13:39:21 +03:00
("<Control>r", _("Toggle repeat mode"), None, playback_group),
("<Control>s", _("Toggle random mode"), None, playback_group),
("<Control>1", _("Toggle single mode"), None, playback_group),
("<Control>o", _("Toggle consume mode"), None, playback_group),
2021-03-26 18:39:21 +03:00
("<Control>e", _("Enqueue selected item"), None, items_group),
("<Control>plus", _("Append selected item"), _("Middle-click"), items_group),
("<Control>Return", _("Play selected item immediately"), _("Double-click"), items_group),
2021-03-27 15:21:46 +03:00
("<Control>i Menu", _("Show additional information"), _("Right-click"), items_group),
2020-09-17 13:31:25 +03:00
("Delete", _("Remove selected song"), _("Middle-click"), playlist_group),
2020-09-29 13:39:21 +03:00
("<Shift>Delete", _("Clear playlist"), None, playlist_group),
2021-03-27 15:21:46 +03:00
("<Control>i Menu", _("Show additional information"), _("Right-click"), playlist_group)
2020-09-29 13:39:21 +03:00
)
2020-09-17 13:31:25 +03:00
for accel, title, subtitle, group in shortcut_data:
shortcut=Gtk.ShortcutsShortcut(visible=True, accelerator=accel, title=title, subtitle=subtitle)
2020-09-16 20:15:17 +03:00
group.pack_start(shortcut, False, False, 0)
self.add(section)
2020-07-04 14:16:17 +03:00
###############
# main window #
###############
class ConnectionNotify(Gtk.Revealer):
def __init__(self, client, settings):
super().__init__(valign=Gtk.Align.START, halign=Gtk.Align.CENTER)
self._client=client
self._settings=settings
# widgets
self._label=Gtk.Label(wrap=True)
close_button=Gtk.Button(image=Gtk.Image.new_from_icon_name("window-close-symbolic", Gtk.IconSize.BUTTON))
close_button.set_relief(Gtk.ReliefStyle.NONE)
connect_button=Gtk.Button(label=_("Connect"))
settings_button=Gtk.Button(label=_("Settings"))
settings_button.set_action_name("win.profile-settings")
# connect
close_button.connect("clicked", self._on_close_button_clicked)
connect_button.connect("clicked", self._on_connect_button_clicked)
self._client.emitter.connect("connection_error", self._on_connection_error)
self._client.emitter.connect("reconnected", self._on_reconnected)
# packing
box=Gtk.Box(spacing=12)
box.get_style_context().add_class("app-notification")
box.pack_start(self._label, False, True, 6)
box.pack_end(close_button, False, True, 0)
box.pack_end(connect_button, False, True, 0)
box.pack_end(settings_button, False, True, 0)
self.add(box)
def _on_connection_error(self, *args):
active=self._settings.get_int("active-profile")
2020-09-30 12:06:00 +03:00
string=_("Connection to “{profile}” ({host}:{port}) failed").format(
2020-09-24 22:17:10 +03:00
profile=self._settings.get_value("profiles")[active],
host=self._settings.get_value("hosts")[active],
port=self._settings.get_value("ports")[active]
)
self._label.set_text(string)
self.set_reveal_child(True)
def _on_reconnected(self, *args):
self.set_reveal_child(False)
def _on_close_button_clicked(self, *args):
self.set_reveal_child(False)
def _on_connect_button_clicked(self, *args):
self._client.reconnect()
2020-01-11 13:25:15 +03:00
class MainWindow(Gtk.ApplicationWindow):
2021-07-19 23:33:54 +03:00
def __init__(self, client, settings, notify, **kwargs):
super().__init__(title=("mpdevil"), icon_name="org.mpdevil.mpdevil", **kwargs)
2020-12-27 14:09:25 +03:00
self.set_default_icon_name("org.mpdevil.mpdevil")
self.set_default_size(settings.get_int("width"), settings.get_int("height"))
2020-10-11 12:40:11 +03:00
if settings.get_boolean("maximize"):
self.maximize() # request maximize
self._client=client
self._settings=settings
2021-07-19 23:33:54 +03:00
self._notify=notify
self._use_csd=self._settings.get_boolean("use-csd")
self._size=None # needed for window size saving
2020-01-11 13:25:15 +03:00
2020-07-04 13:35:39 +03:00
# MPRIS
2021-04-17 14:22:07 +03:00
if self._settings.get_boolean("mpris"):
dbus_service=MPRISInterface(self, self._client, self._settings)
2020-03-21 00:09:13 +03:00
2020-07-04 13:35:39 +03:00
# actions
2020-09-30 11:20:55 +03:00
simple_actions_data=(
"settings","profile-settings","stats","help","menu",
2020-09-30 11:20:55 +03:00
"toggle-lyrics","back-to-current-album","toggle-search",
2021-03-26 18:39:21 +03:00
"profile-next","profile-prev","show-info","append","play","enqueue"
2020-09-30 11:20:55 +03:00
)
2020-09-16 15:38:58 +03:00
for name in simple_actions_data:
action=Gio.SimpleAction.new(name, None)
action.connect("activate", getattr(self, ("_on_"+name.replace("-","_"))))
self.add_action(action)
2020-09-15 19:45:30 +03:00
mini_player_action=Gio.PropertyAction.new("mini-player", self._settings, "mini-player")
self.add_action(mini_player_action)
2020-09-29 23:19:55 +03:00
self._profiles_action=Gio.SimpleAction.new_stateful("profiles", GLib.VariantType.new("i"), GLib.Variant("i", 0))
self._profiles_action.connect("change-state", self._on_profiles)
self.add_action(self._profiles_action)
2020-01-11 13:25:15 +03:00
2021-03-27 14:50:45 +03:00
# shortcuts
shortcuts_window=ShortcutsWindow()
self.set_help_overlay(shortcuts_window)
shortcuts_window.set_modal(False)
2020-07-04 13:35:39 +03:00
# widgets
2020-10-20 14:13:50 +03:00
if self._use_csd:
icons={"open-menu-symbolic": Gtk.Image.new_from_icon_name("open-menu-symbolic", Gtk.IconSize.BUTTON)}
else:
icons={"open-menu-symbolic": AutoSizedIcon("open-menu-symbolic", "icon-size", self._settings)}
2021-02-07 21:05:30 +03:00
self._paned=Gtk.Paned()
self._browser=Browser(self._client, self._settings)
self._cover_playlist_window=CoverPlaylistWindow(self._client, self._settings)
2020-09-29 13:39:21 +03:00
playback_control=PlaybackControl(self._client, self._settings)
seek_bar=SeekBar(self._client)
2021-07-16 21:07:52 +03:00
audio=AudioFormat(self._client, self._settings)
playback_options=PlaybackOptions(self._client, self._settings)
volume_button=VolumeButton(self._client, self._settings)
connection_notify=ConnectionNotify(self._client, self._settings)
2020-01-11 13:25:15 +03:00
2020-07-04 13:35:39 +03:00
# menu
2020-05-28 23:46:38 +03:00
subsection=Gio.Menu()
subsection.append(_("Settings"), "win.settings")
2020-09-16 20:15:17 +03:00
subsection.append(_("Keyboard shortcuts"), "win.show-help-overlay")
2020-05-28 23:46:38 +03:00
subsection.append(_("Help"), "win.help")
subsection.append(_("About"), "app.about")
subsection.append(_("Quit"), "app.quit")
2020-09-11 14:48:40 +03:00
mpd_subsection=Gio.Menu()
2020-09-29 13:39:21 +03:00
mpd_subsection.append(_("Update database"), "mpd.update")
2020-09-11 14:48:40 +03:00
mpd_subsection.append(_("Server stats"), "win.stats")
2020-09-29 23:19:55 +03:00
self._profiles_submenu=Gio.Menu()
self._refresh_profiles_menu()
2020-04-09 01:26:21 +03:00
menu=Gio.Menu()
2020-09-29 23:19:55 +03:00
menu.append_submenu(_("Profiles"), self._profiles_submenu)
2020-09-15 19:45:30 +03:00
menu.append(_("Mini player"), "win.mini-player")
2020-09-11 14:48:40 +03:00
menu.append_section(None, mpd_subsection)
2020-05-28 23:46:38 +03:00
menu.append_section(None, subsection)
2020-01-11 13:25:15 +03:00
2020-09-29 23:19:55 +03:00
# menu button / popover
2020-10-20 14:13:50 +03:00
self._menu_button=Gtk.MenuButton(image=icons["open-menu-symbolic"], tooltip_text=_("Menu"))
2020-09-29 14:02:51 +03:00
self._menu_button.set_can_focus(False)
2020-09-29 13:39:21 +03:00
menu_popover=Gtk.Popover.new_from_model(self._menu_button, menu)
self._menu_button.set_popover(menu_popover)
# action bar
action_bar=Gtk.ActionBar()
2020-09-29 13:39:21 +03:00
action_bar.pack_start(playback_control)
action_bar.pack_start(seek_bar)
2021-07-16 20:39:48 +03:00
action_bar.pack_start(audio)
action_bar.pack_start(playback_options)
action_bar.pack_start(volume_button)
2020-01-11 13:25:15 +03:00
2020-07-04 13:35:39 +03:00
# connect
2020-09-29 23:19:55 +03:00
self._settings.connect("changed::profiles", self._refresh_profiles_menu)
self._settings.connect("changed::active-profile", self._on_active_profile_changed)
2020-09-15 19:45:30 +03:00
self._settings.connect_after("notify::mini-player", self._on_mini_player)
self._settings.connect_after("notify::cursor-watch", self._on_cursor_watch)
self._settings.connect("changed::playlist-right", self._on_playlist_pos_changed)
self._client.emitter.connect("current_song_changed", self._on_song_changed)
self._client.emitter.connect("disconnected", self._on_disconnected)
self._client.emitter.connect("reconnected", self._on_reconnected)
2021-02-08 11:32:55 +03:00
# auto save window state and size
2021-02-07 21:05:30 +03:00
self.connect("size-allocate", self._on_size_allocate)
self._settings.bind("maximize", self, "is-maximized", Gio.SettingsBindFlags.SET)
2021-02-08 19:49:45 +03:00
# save and restore mini player
self._settings.bind("mini-player", self._settings, "mini-player", Gio.SettingsBindFlags.DEFAULT)
2020-01-11 13:25:15 +03:00
2020-07-04 13:35:39 +03:00
# packing
self._on_playlist_pos_changed() # set orientation
2021-02-07 21:05:30 +03:00
self._paned.pack1(self._browser, True, False)
self._paned.pack2(self._cover_playlist_window, False, False)
vbox=Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
2021-02-07 21:05:30 +03:00
vbox.pack_start(self._paned, True, True, 0)
vbox.pack_start(action_bar, False, False, 0)
overlay=Gtk.Overlay()
overlay.add(vbox)
overlay.add_overlay(connection_notify)
if self._use_csd:
self._header_bar=Gtk.HeaderBar()
self._header_bar.set_show_close_button(True)
self.set_titlebar(self._header_bar)
self._header_bar.pack_start(self._browser.back_to_current_album_button)
2020-09-29 13:39:21 +03:00
self._header_bar.pack_end(self._menu_button)
self._header_bar.pack_end(self._browser.search_button)
else:
2020-09-29 13:39:21 +03:00
action_bar.pack_start(self._menu_button)
self.add(overlay)
self._client.emitter.emit("disconnected") # bring player in defined state
# indicate connection process in window title
if self._use_csd:
self._header_bar.set_subtitle(_("connecting…"))
else:
self.set_title("mpdevil "+_("connecting…"))
2020-01-19 02:00:40 +03:00
self.show_all()
while Gtk.events_pending(): # ensure window is visible
Gtk.main_iteration_do(True)
2021-02-08 11:32:55 +03:00
# restore paned settings when window is visible (fixes a bug when window is maximized)
self._cover_playlist_window.set_position(self._settings.get_int("paned0"))
self._browser.set_position(self._settings.get_int("paned1"))
self._paned.set_position(self._settings.get_int("paned2"))
# auto save paned positions
self._cover_playlist_window.connect("notify::position", self._on_paned_position, "paned0")
self._browser.connect("notify::position", self._on_paned_position, "paned1")
self._paned.connect("notify::position", self._on_paned_position, "paned2")
# start client
def callback(*args):
self._client.start() # connect client
return False
GLib.idle_add(callback)
2020-01-11 13:25:15 +03:00
2020-09-29 13:39:21 +03:00
def _on_toggle_lyrics(self, action, param):
self._cover_playlist_window.lyrics_button.set_active(not(self._cover_playlist_window.lyrics_button.get_active()))
def _on_back_to_current_album(self, action, param):
self._browser.back_to_current_album_button.emit("clicked")
def _on_toggle_search(self, action, param):
self._browser.search_button.set_active(not(self._browser.search_button.get_active()))
def _on_settings(self, action, param):
settings=SettingsDialog(self, self._client, self._settings)
settings.run()
settings.destroy()
def _on_profile_settings(self, action, param):
settings=SettingsDialog(self, self._client, self._settings, "profiles")
settings.run()
settings.destroy()
2020-09-29 13:39:21 +03:00
def _on_stats(self, action, param):
stats=ServerStats(self, self._client, self._settings)
stats.destroy()
def _on_help(self, action, param):
Gtk.show_uri_on_window(self, "https://github.com/SoongNoonien/mpdevil/wiki/Usage", Gdk.CURRENT_TIME)
def _on_menu(self, action, param):
self._menu_button.emit("clicked")
2020-09-30 11:20:55 +03:00
def _on_profile_next(self, action, param):
total_profiles=len(self._settings.get_value("profiles"))
current_profile=self._settings.get_int("active-profile")
self._settings.set_int("active-profile", (current_profile+1)%total_profiles)
def _on_profile_prev(self, action, param):
total_profiles=len(self._settings.get_value("profiles"))
current_profile=self._settings.get_int("active-profile")
self._settings.set_int("active-profile", (current_profile-1)%total_profiles)
2021-03-26 18:39:21 +03:00
def _on_show_info(self, action, param):
self._client.emitter.emit("show-info")
def _on_append(self, action, param):
self._client.emitter.emit("add-to-playlist", "append")
def _on_play(self, action, param):
self._client.emitter.emit("add-to-playlist", "play")
def _on_enqueue(self, action, param):
self._client.emitter.emit("add-to-playlist", "enqueue")
2020-09-29 23:19:55 +03:00
def _on_profiles(self, action, param):
self._settings.set_int("active-profile", param.unpack())
action.set_state(param)
def _on_song_changed(self, *args):
song=self._client.currentsong()
2020-07-15 23:39:26 +03:00
if song == {}:
if self._use_csd:
2020-08-29 11:51:31 +03:00
self.set_title("mpdevil")
self._header_bar.set_subtitle("")
2020-07-15 23:39:26 +03:00
else:
self.set_title("mpdevil")
else:
2020-10-17 12:30:00 +03:00
song=ClientHelper.song_to_str_dict(ClientHelper.pepare_song_for_display(song))
2020-08-21 13:03:42 +03:00
if song["date"] == "":
date=""
2020-08-21 13:03:42 +03:00
else:
date=" ("+song["date"]+")"
if self._use_csd:
self.set_title(song["title"]+" • "+song["artist"])
self._header_bar.set_subtitle(song["album"]+date)
else:
self.set_title(song["title"]+" • "+song["artist"]+" • "+song["album"]+date)
if self._settings.get_boolean("send-notify"):
if not self.is_active() and self._client.status()["state"] == "play":
2021-07-18 22:29:05 +03:00
self._notify.close() # clear previous notifications
self._notify.update(song["title"], f"{song['artist']}\n{song['album']}{date}")
pixbuf=self._client.get_cover(song, 400)
2021-07-18 22:29:05 +03:00
self._notify.set_image_from_pixbuf(pixbuf)
self._notify.show()
def _on_reconnected(self, *args):
2021-02-07 21:05:30 +03:00
for action in ("stats","toggle-lyrics","back-to-current-album","toggle-search"):
2020-09-29 13:39:21 +03:00
self.lookup_action(action).set_enabled(True)
def _on_disconnected(self, *args):
2020-09-11 00:40:27 +03:00
self.set_title("mpdevil")
if self._use_csd:
2020-09-11 00:40:27 +03:00
self._header_bar.set_subtitle("")
2021-02-07 21:05:30 +03:00
for action in ("stats","toggle-lyrics","back-to-current-album","toggle-search"):
2020-09-29 13:39:21 +03:00
self.lookup_action(action).set_enabled(False)
2020-01-11 13:25:15 +03:00
2021-02-08 11:32:55 +03:00
def _on_size_allocate(self, widget, rect):
if not self.is_maximized() and not self._settings.get_property("mini-player"):
size=self.get_size()
if size != self._size: # prevent unneeded write operations
self._settings.set_int("width", size[0])
self._settings.set_int("height", size[1])
self._size=size
2021-02-08 11:32:55 +03:00
def _on_paned_position(self, obj, typestring, key):
self._settings.set_int(key, obj.get_position())
2020-09-15 19:45:30 +03:00
def _on_mini_player(self, obj, typestring):
if obj.get_property("mini-player"):
2021-02-07 21:14:38 +03:00
if self.is_maximized():
2020-09-15 20:28:40 +03:00
self.unmaximize()
2020-09-15 19:45:30 +03:00
self.resize(1,1)
2020-03-24 18:14:01 +03:00
else:
2021-02-07 21:14:38 +03:00
self.resize(self._settings.get_int("width"), self._settings.get_int("height"))
2020-02-07 22:13:38 +03:00
def _on_cursor_watch(self, obj, typestring):
if obj.get_property("cursor-watch"):
2021-07-18 23:39:35 +03:00
watch_cursor=Gdk.Cursor(Gdk.CursorType.WATCH)
self.get_window().set_cursor(watch_cursor)
else:
self.get_window().set_cursor(None)
def _on_playlist_pos_changed(self, *args):
if self._settings.get_boolean("playlist-right"):
self._cover_playlist_window.set_orientation(Gtk.Orientation.VERTICAL)
2021-02-07 21:05:30 +03:00
self._paned.set_orientation(Gtk.Orientation.HORIZONTAL)
else:
self._cover_playlist_window.set_orientation(Gtk.Orientation.HORIZONTAL)
2021-02-07 21:05:30 +03:00
self._paned.set_orientation(Gtk.Orientation.VERTICAL)
2020-09-29 23:19:55 +03:00
def _refresh_profiles_menu(self, *args):
self._profiles_submenu.remove_all()
for num, profile in enumerate(self._settings.get_value("profiles")):
item=Gio.MenuItem.new(profile, None)
item.set_action_and_target_value("win.profiles", GLib.Variant("i", num))
self._profiles_submenu.append_item(item)
self._profiles_action.set_state(GLib.Variant("i", self._settings.get_int("active-profile")))
def _on_active_profile_changed(self, *args):
self._profiles_action.set_state(GLib.Variant("i", self._settings.get_int("active-profile")))
2020-07-04 14:16:17 +03:00
###################
# Gtk application #
###################
2020-01-11 13:25:15 +03:00
class mpdevil(Gtk.Application):
2021-07-19 23:33:54 +03:00
def __init__(self):
super().__init__(application_id="org.mpdevil.mpdevil", flags=Gio.ApplicationFlags.HANDLES_COMMAND_LINE)
self.add_main_option("debug", ord("d"), GLib.OptionFlags.NONE, GLib.OptionArg.NONE, _("Debug mode"), None)
self._settings=Settings()
self._client=Client(self._settings)
2021-07-18 22:29:05 +03:00
Notify.init("mpdevil")
2021-07-19 23:41:32 +03:00
self._notify=Notify.Notification()
2021-07-19 23:33:54 +03:00
self._window=None
2021-07-18 22:29:05 +03:00
2020-01-11 13:25:15 +03:00
def do_activate(self):
if not self._window: # allow just one instance
2021-07-19 23:33:54 +03:00
self._window=MainWindow(self._client, self._settings, self._notify, application=self)
2021-06-27 20:43:18 +03:00
self._window.connect("delete-event", self._on_quit)
self._window.insert_action_group("mpd", MPDActionGroup(self._client))
2020-09-16 15:38:58 +03:00
# accelerators
2020-09-29 13:39:21 +03:00
action_accels=(
("app.quit", ["<Control>q"]),("win.mini-player", ["<Control>m"]),("win.help", ["F1"]),("win.menu", ["F10"]),
("win.show-help-overlay", ["<Control>question"]),("win.toggle-lyrics", ["<Control>l"]),
("win.back-to-current-album", ["Escape"]),("win.toggle-search", ["<control>f"]),
("mpd.update", ["F5"]),("mpd.clear", ["<Shift>Delete"]),("mpd.toggle-play", ["space"]),
("mpd.stop", ["<Control>space"]),("mpd.next", ["KP_Add"]),("mpd.prev", ["KP_Subtract"]),
("mpd.repeat", ["<Control>r"]),("mpd.random", ["<Control>s"]),("mpd.single", ["<Control>1"]),
2020-09-30 11:20:55 +03:00
("mpd.consume", ["<Control>o"]),("mpd.seek-forward", ["KP_Multiply"]),("mpd.seek-backward", ["KP_Divide"]),
2021-03-26 18:39:21 +03:00
("win.profile-next", ["<Control>p"]),("win.profile-prev", ["<Shift><Control>p"]),
2021-03-27 15:21:46 +03:00
("win.show-info", ["<Control>i","Menu"]),("win.append", ["<Control>plus"]),
2021-03-26 18:39:21 +03:00
("win.play", ["<Control>Return"]),("win.enqueue", ["<Control>e"])
2020-09-29 13:39:21 +03:00
)
for action, accels in action_accels:
self.set_accels_for_action(action, accels)
# disable item activation on space key pressed in treeviews
Gtk.binding_entry_remove(Gtk.binding_set_find('GtkTreeView'), Gdk.keyval_from_name("space"), Gdk.ModifierType.MOD2_MASK)
self._window.present()
2020-01-11 13:25:15 +03:00
def do_startup(self):
Gtk.Application.do_startup(self)
2020-04-09 01:26:21 +03:00
action=Gio.SimpleAction.new("about", None)
action.connect("activate", self._on_about)
2020-01-11 13:25:15 +03:00
self.add_action(action)
2020-04-09 01:26:21 +03:00
action=Gio.SimpleAction.new("quit", None)
action.connect("activate", self._on_quit)
2020-01-11 13:25:15 +03:00
self.add_action(action)
2021-06-27 20:43:18 +03:00
def do_command_line(self, command_line):
# convert GVariantDict -> GVariant -> dict
options=command_line.get_options_dict().end().unpack()
if "debug" in options:
import logging
logging.basicConfig(level=logging.DEBUG)
self.activate()
return 0
2020-01-11 13:25:15 +03:00
2021-06-27 20:43:18 +03:00
def _on_about(self, *args):
dialog=AboutDialog(self._window)
2020-01-11 13:25:15 +03:00
dialog.run()
dialog.destroy()
2021-06-27 20:43:18 +03:00
def _on_quit(self, *args):
if self._settings.get_boolean("stop-on-quit") and self._client.connected():
self._client.stop()
2021-07-19 23:33:54 +03:00
self._notify.close()
2021-07-19 19:19:48 +03:00
Notify.uninit()
2020-01-11 13:25:15 +03:00
self.quit()
2020-09-24 22:17:10 +03:00
if __name__ == "__main__":
2020-04-09 01:26:21 +03:00
app=mpdevil()
2020-01-11 13:25:15 +03:00
app.run(sys.argv)