mirror of
https://github.com/SoongNoonien/mpdevil.git
synced 2023-08-10 21:12:44 +03:00
3135 lines
104 KiB
Python
3135 lines
104 KiB
Python
#!/usr/bin/python3
|
|
# -*- coding: utf-8 -*-
|
|
#
|
|
# mpdevil - MPD Client.
|
|
# Copyright 2020 Martin Wagner
|
|
#
|
|
# This program is free software; you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation; version 3 of the License.
|
|
#
|
|
# 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.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with this program; if not, write to the Free Software
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
|
|
# USA
|
|
|
|
# MPRIS interface based on 'mpDris2' (master 19.03.2020) by Jean-Philippe Braun <eon@patapon.info>, Mantas Mikulėnas <grawity@gmail.com>
|
|
|
|
import gi
|
|
gi.require_version('Gtk', '3.0')
|
|
gi.require_version('Notify', '0.7')
|
|
from gi.repository import Gtk, Gio, Gdk, GdkPixbuf, Pango, GObject, GLib, Notify
|
|
from mpd import MPDClient
|
|
import requests #dev-python/requests
|
|
from bs4 import BeautifulSoup, Comment
|
|
import threading
|
|
import locale
|
|
import gettext
|
|
import datetime
|
|
import os
|
|
import sys
|
|
|
|
#MPRIS modules
|
|
import dbus
|
|
import dbus.service
|
|
from dbus.mainloop.glib import DBusGMainLoop
|
|
import base64
|
|
import re
|
|
|
|
DATADIR = '@datadir@'
|
|
NAME = 'mpdevil'
|
|
VERSION = '@version@'
|
|
PACKAGE = NAME.lower()
|
|
|
|
try:
|
|
locale.setlocale(locale.LC_ALL, '')
|
|
locale.bindtextdomain(PACKAGE, '@datadir@/locale')
|
|
gettext.bindtextdomain(PACKAGE, '@datadir@/locale')
|
|
gettext.textdomain(PACKAGE)
|
|
gettext.install(PACKAGE, localedir='@datadir@/locale')
|
|
except locale.Error:
|
|
print(' cannot use system locale.')
|
|
locale.setlocale(locale.LC_ALL, 'C')
|
|
gettext.textdomain(PACKAGE)
|
|
gettext.install(PACKAGE, localedir='@datadir@/locale')
|
|
|
|
# MPRIS allowed metadata tags
|
|
allowed_tags = {
|
|
'mpris:trackid': dbus.ObjectPath,
|
|
'mpris:length': dbus.Int64,
|
|
'mpris:artUrl': str,
|
|
'xesam:album': str,
|
|
'xesam:albumArtist': list,
|
|
'xesam:artist': list,
|
|
'xesam:asText': str,
|
|
'xesam:audioBPM': int,
|
|
'xesam:comment': list,
|
|
'xesam:composer': list,
|
|
'xesam:contentCreated': str,
|
|
'xesam:discNumber': int,
|
|
'xesam:firstUsed': str,
|
|
'xesam:genre': list,
|
|
'xesam:lastUsed': str,
|
|
'xesam:lyricist': str,
|
|
'xesam:title': str,
|
|
'xesam:trackNumber': int,
|
|
'xesam:url': str,
|
|
'xesam:useCount': int,
|
|
'xesam:userRating': float,
|
|
}
|
|
|
|
# python dbus bindings don't include annotations and properties
|
|
MPRIS2_INTROSPECTION = """<node name="/org/mpris/MediaPlayer2">
|
|
<interface name="org.freedesktop.DBus.Introspectable">
|
|
<method name="Introspect">
|
|
<arg direction="out" name="xml_data" type="s"/>
|
|
</method>
|
|
</interface>
|
|
<interface name="org.freedesktop.DBus.Properties">
|
|
<method name="Get">
|
|
<arg direction="in" name="interface_name" type="s"/>
|
|
<arg direction="in" name="property_name" type="s"/>
|
|
<arg direction="out" name="value" type="v"/>
|
|
</method>
|
|
<method name="GetAll">
|
|
<arg direction="in" name="interface_name" type="s"/>
|
|
<arg direction="out" name="properties" type="a{sv}"/>
|
|
</method>
|
|
<method name="Set">
|
|
<arg direction="in" name="interface_name" type="s"/>
|
|
<arg direction="in" name="property_name" type="s"/>
|
|
<arg direction="in" name="value" type="v"/>
|
|
</method>
|
|
<signal name="PropertiesChanged">
|
|
<arg name="interface_name" type="s"/>
|
|
<arg name="changed_properties" type="a{sv}"/>
|
|
<arg name="invalidated_properties" type="as"/>
|
|
</signal>
|
|
</interface>
|
|
<interface name="org.mpris.MediaPlayer2">
|
|
<method name="Raise"/>
|
|
<method name="Quit"/>
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="false"/>
|
|
<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">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="true"/>
|
|
</property>
|
|
<property name="LoopStatus" type="s" access="readwrite">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="true"/>
|
|
</property>
|
|
<property name="Rate" type="d" access="readwrite">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="true"/>
|
|
</property>
|
|
<property name="Shuffle" type="b" access="readwrite">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="true"/>
|
|
</property>
|
|
<property name="Metadata" type="a{sv}" access="read">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="true"/>
|
|
</property>
|
|
<property name="Volume" type="d" access="readwrite">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="false"/>
|
|
</property>
|
|
<property name="Position" type="x" access="read">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="false"/>
|
|
</property>
|
|
<property name="MinimumRate" type="d" access="read">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="true"/>
|
|
</property>
|
|
<property name="MaximumRate" type="d" access="read">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="true"/>
|
|
</property>
|
|
<property name="CanGoNext" type="b" access="read">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="true"/>
|
|
</property>
|
|
<property name="CanGoPrevious" type="b" access="read">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="true"/>
|
|
</property>
|
|
<property name="CanPlay" type="b" access="read">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="true"/>
|
|
</property>
|
|
<property name="CanPause" type="b" access="read">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="true"/>
|
|
</property>
|
|
<property name="CanSeek" type="b" access="read">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="true"/>
|
|
</property>
|
|
<property name="CanControl" type="b" access="read">
|
|
<annotation name="org.freedesktop.DBus.Property.EmitsChangedSignal" value="false"/>
|
|
</property>
|
|
</interface>
|
|
</node>"""
|
|
|
|
class IntEntry(Gtk.SpinButton):
|
|
def __init__(self, default, lower, upper, step):
|
|
Gtk.SpinButton.__init__(self)
|
|
adj = Gtk.Adjustment(value=default, lower=lower, upper=upper, step_increment=step)
|
|
self.set_adjustment(adj)
|
|
|
|
def get_int(self):
|
|
return int(self.get_value())
|
|
|
|
def set_int(self, value):
|
|
self.set_value(value)
|
|
|
|
class FocusFrame(Gtk.Frame):
|
|
def __init__(self):
|
|
Gtk.Frame.__init__(self)
|
|
|
|
#css
|
|
self.style_context=self.get_style_context()
|
|
self.provider = Gtk.CssProvider()
|
|
css = b"""* {border-color: @theme_selected_bg_color;}"""
|
|
self.provider.load_from_data(css)
|
|
|
|
provider_start = Gtk.CssProvider()
|
|
css_start = b"""* {border-color: @theme_bg_color;}"""
|
|
provider_start.load_from_data(css_start)
|
|
|
|
self.style_context.add_provider(provider_start, 800)
|
|
|
|
def set_widget(self, widget):
|
|
widget.connect("focus-in-event", self.on_focus_in_event)
|
|
widget.connect("focus-out-event", self.on_focus_out_event)
|
|
|
|
def on_focus_in_event(self, *args):
|
|
self.style_context.add_provider(self.provider, 800)
|
|
|
|
def on_focus_out_event(self, *args):
|
|
self.style_context.remove_provider(self.provider)
|
|
|
|
class Cover(object):
|
|
def __init__(self, lib_path, song_file):
|
|
self.lib_path=lib_path or "/"
|
|
self.path=None
|
|
if not song_file == None:
|
|
head_tail=os.path.split(song_file)
|
|
if self.lib_path[-1] == "/":
|
|
path=(self.lib_path+head_tail[0]+"/")
|
|
else:
|
|
path=(self.lib_path+"/"+head_tail[0]+"/")
|
|
if os.path.exists(path):
|
|
filelist=[file for file in os.listdir(path) if file.endswith('.jpg') or file.endswith('.png') or file.endswith('.gif')]
|
|
if not filelist == []:
|
|
self.path=(path+filelist[0])
|
|
|
|
def get_pixbuf(self, size):
|
|
if self.path == None:
|
|
self.path = Gtk.IconTheme.get_default().lookup_icon("mpdevil", size, Gtk.IconLookupFlags.FORCE_SVG).get_filename() #fallback cover
|
|
return GdkPixbuf.Pixbuf.new_from_file_at_size(self.path, size, size)
|
|
|
|
class AutoSettingsClient(MPDClient):
|
|
def __init__(self, settings):
|
|
MPDClient.__init__(self)
|
|
self.settings = settings
|
|
self.settings.connect("changed::active-profile", self.on_settings_changed)
|
|
|
|
def try_connect_default(self):
|
|
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(None)
|
|
else:
|
|
self.password(self.settings.get_value("passwords")[active])
|
|
except:
|
|
pass
|
|
|
|
def connected(self):
|
|
try:
|
|
self.ping()
|
|
return True
|
|
except:
|
|
return False
|
|
|
|
def on_settings_changed(self, *args):
|
|
self.disconnect()
|
|
|
|
class MpdEventEmitter(GObject.Object):
|
|
__gsignals__ = {
|
|
'database': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'update': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'stored_playlist': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'playlist': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'player': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'mixer': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'output': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'options': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'sticker': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'subscription': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'message': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'disconnected': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'reconnected': (GObject.SignalFlags.RUN_FIRST, None, ()),
|
|
'playing_file_changed': (GObject.SignalFlags.RUN_FIRST, None, ())
|
|
}
|
|
|
|
def __init__(self, settings):
|
|
super().__init__()
|
|
self.client=AutoSettingsClient(settings)
|
|
GLib.timeout_add(100, self.watch)
|
|
self.connected=True
|
|
self.current_file=None
|
|
|
|
def watch(self, *args):
|
|
try:
|
|
a=self.client.noidle()
|
|
for i in a:
|
|
self.emit(i)
|
|
except:
|
|
pass
|
|
try:
|
|
self.client.send_idle()
|
|
except:
|
|
self.client.try_connect_default()
|
|
if self.client.connected():
|
|
self.emit("disconnected")
|
|
self.emit("reconnected")
|
|
elif self.connected:
|
|
self.emit("disconnected")
|
|
return True
|
|
|
|
#mpd signals
|
|
def do_database(self):
|
|
pass
|
|
|
|
def do_update(self):
|
|
pass
|
|
|
|
def do_stored_playlist(self):
|
|
pass
|
|
|
|
def do_playlist(self):
|
|
pass
|
|
|
|
def do_player(self):
|
|
current_song=self.client.currentsong()
|
|
if not current_song == {}:
|
|
if not current_song['file'] == self.current_file:
|
|
self.emit("playing_file_changed")
|
|
self.current_file=current_song['file']
|
|
else:
|
|
self.emit("playing_file_changed")
|
|
self.current_file=None
|
|
|
|
def do_mixer(self):
|
|
pass
|
|
|
|
def do_output(self):
|
|
pass
|
|
|
|
def do_options(self):
|
|
pass
|
|
|
|
def do_sticker(self):
|
|
pass
|
|
|
|
def do_subscription(self):
|
|
pass
|
|
|
|
def do_message(self):
|
|
pass
|
|
|
|
#custom signals
|
|
def do_disconnected(self):
|
|
self.connected=False
|
|
self.current_file=None
|
|
|
|
def do_reconnected(self):
|
|
self.connected=True
|
|
|
|
def do_playing_file_changed(self):
|
|
pass
|
|
|
|
class Client(AutoSettingsClient):
|
|
def __init__(self, settings):
|
|
AutoSettingsClient.__init__(self, settings)
|
|
|
|
#adding vars
|
|
self.settings=settings
|
|
self.emitter=MpdEventEmitter(self.settings)
|
|
|
|
#connect
|
|
self.emitter.connect("reconnected", self.on_reconnected)
|
|
|
|
def files_to_playlist(self, files, append, force=False):
|
|
if append:
|
|
for f in files:
|
|
self.add(f)
|
|
else:
|
|
if self.settings.get_boolean("force-mode") or force or self.status()["state"] == "stop":
|
|
if not files == []:
|
|
self.clear()
|
|
for f in files:
|
|
self.add(f)
|
|
self.play()
|
|
else:
|
|
status=self.status()
|
|
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:
|
|
if not f == current_song_file:
|
|
self.add(f)
|
|
else:
|
|
self.move(0, (len(self.playlist())-1))
|
|
|
|
def album_to_playlist(self, album, artist, year, append, force=False):
|
|
songs=self.find("album", album, "date", year, self.settings.get_artist_type(), artist)
|
|
self.files_to_playlist([song['file'] for song in songs], append, force)
|
|
|
|
def on_reconnected(self, *args):
|
|
self.try_connect_default()
|
|
self.emitter.emit("playlist")
|
|
self.emitter.emit("player")
|
|
self.emitter.emit("options")
|
|
self.emitter.emit("mixer")
|
|
self.emitter.emit("update")
|
|
|
|
class MPRISInterface(dbus.service.Object): #TODO emit Seeked if needed
|
|
__introspect_interface = "org.freedesktop.DBus.Introspectable"
|
|
__prop_interface = dbus.PROPERTIES_IFACE
|
|
|
|
def __init__(self, window, client, settings):
|
|
dbus.service.Object.__init__(self, dbus.SessionBus(), "/org/mpris/MediaPlayer2")
|
|
self._name = "org.mpris.MediaPlayer2.mpdevil"
|
|
|
|
self._bus = dbus.SessionBus()
|
|
self._uname = self._bus.get_unique_name()
|
|
self._dbus_obj = self._bus.get_object("org.freedesktop.DBus", "/org/freedesktop/DBus")
|
|
self._dbus_obj.connect_to_signal("NameOwnerChanged", self._name_owner_changed_callback, arg0=self._name)
|
|
|
|
self.window=window
|
|
self.client=client
|
|
self.settings=settings
|
|
self.metadata={}
|
|
|
|
#connect
|
|
self.client.emitter.connect("player", self.on_player_changed)
|
|
self.client.emitter.connect("playing_file_changed", self.on_file_changed)
|
|
self.client.emitter.connect("mixer", self.on_volume_changed)
|
|
self.client.emitter.connect("options", self.on_options_changed)
|
|
|
|
def on_player_changed(self, *args):
|
|
self.update_property('org.mpris.MediaPlayer2.Player', 'PlaybackStatus')
|
|
self.update_property('org.mpris.MediaPlayer2.Player', 'CanGoNext')
|
|
self.update_property('org.mpris.MediaPlayer2.Player', 'CanGoPrevious')
|
|
|
|
def on_file_changed(self, *args):
|
|
self.update_metadata()
|
|
self.update_property('org.mpris.MediaPlayer2.Player', 'Metadata')
|
|
|
|
def on_volume_changed(self, *args):
|
|
self.update_property('org.mpris.MediaPlayer2.Player', 'Volume')
|
|
|
|
def on_options_changed(self, *args):
|
|
self.update_property('org.mpris.MediaPlayer2.Player', 'LoopStatus')
|
|
self.update_property('org.mpris.MediaPlayer2.Player', 'Shuffle')
|
|
|
|
def update_metadata(self): #TODO
|
|
"""
|
|
Translate metadata returned by MPD to the MPRIS v2 syntax.
|
|
http://www.freedesktop.org/wiki/Specifications/mpris-spec/metadata
|
|
"""
|
|
|
|
mpd_meta = self.client.currentsong()
|
|
self.metadata = {}
|
|
|
|
for tag in ('album', 'title'):
|
|
if tag in mpd_meta:
|
|
self.metadata['xesam:%s' % tag] = mpd_meta[tag]
|
|
|
|
if 'id' in mpd_meta:
|
|
self.metadata['mpris:trackid'] = "/org/mpris/MediaPlayer2/Track/%s" % mpd_meta['id']
|
|
|
|
if 'time' in mpd_meta:
|
|
self.metadata['mpris:length'] = int(mpd_meta['time']) * 1000000
|
|
|
|
if 'date' in mpd_meta:
|
|
self.metadata['xesam:contentCreated'] = mpd_meta['date'][0:4]
|
|
|
|
if 'track' in mpd_meta:
|
|
# TODO: Is it even *possible* for mpd_meta['track'] to be a list?
|
|
if type(mpd_meta['track']) == list and len(mpd_meta['track']) > 0:
|
|
track = str(mpd_meta['track'][0])
|
|
else:
|
|
track = str(mpd_meta['track'])
|
|
|
|
m = re.match('^([0-9]+)', track)
|
|
if m:
|
|
self.metadata['xesam:trackNumber'] = int(m.group(1))
|
|
# Ensure the integer is signed 32bit
|
|
if self.metadata['xesam:trackNumber'] & 0x80000000:
|
|
self.metadata['xesam:trackNumber'] += -0x100000000
|
|
else:
|
|
self.metadata['xesam:trackNumber'] = 0
|
|
|
|
if 'disc' in mpd_meta:
|
|
# TODO: Same as above. When is it a list?
|
|
if type(mpd_meta['disc']) == list and len(mpd_meta['disc']) > 0:
|
|
disc = str(mpd_meta['disc'][0])
|
|
else:
|
|
disc = str(mpd_meta['disc'])
|
|
|
|
m = re.match('^([0-9]+)', disc)
|
|
if m:
|
|
self.metadata['xesam:discNumber'] = int(m.group(1))
|
|
|
|
if 'artist' in mpd_meta:
|
|
if type(mpd_meta['artist']) == list:
|
|
self.metadata['xesam:artist'] = mpd_meta['artist']
|
|
else:
|
|
self.metadata['xesam:artist'] = [mpd_meta['artist']]
|
|
|
|
if 'composer' in mpd_meta:
|
|
if type(mpd_meta['composer']) == list:
|
|
self.metadata['xesam:composer'] = mpd_meta['composer']
|
|
else:
|
|
self.metadata['xesam:composer'] = [mpd_meta['composer']]
|
|
|
|
# Stream: populate some missings tags with stream's name
|
|
if 'name' in mpd_meta:
|
|
if 'xesam:title' not in self.metadata:
|
|
self.metadata['xesam:title'] = mpd_meta['name']
|
|
elif 'xesam:album' not in self.metadata:
|
|
self.metadata['xesam:album'] = mpd_meta['name']
|
|
|
|
if 'file' in mpd_meta:
|
|
song_file = mpd_meta['file']
|
|
self.metadata['xesam:url'] = "file://"+os.path.join(self.settings.get_value("paths")[self.settings.get_int("active-profile")], song_file)
|
|
cover=Cover(lib_path=self.settings.get_value("paths")[self.settings.get_int("active-profile")], song_file=song_file)
|
|
if not cover.path == None:
|
|
self.metadata['mpris:artUrl'] = "file://"+cover.path
|
|
else:
|
|
self.metadata['mpris:artUrl'] = None
|
|
|
|
# Cast self.metadata to the correct type, or discard it
|
|
for key, value in self.metadata.items():
|
|
try:
|
|
self.metadata[key] = allowed_tags[key](value)
|
|
except ValueError:
|
|
del self.metadata[key]
|
|
|
|
def _name_owner_changed_callback(self, name, old_owner, new_owner):
|
|
if name == self._name and old_owner == self._uname and new_owner != "":
|
|
try:
|
|
pid = self._dbus_obj.GetConnectionUnixProcessID(new_owner)
|
|
except:
|
|
pid = None
|
|
loop.quit()
|
|
|
|
def acquire_name(self):
|
|
self._bus_name = dbus.service.BusName(self._name, bus=self._bus, allow_replacement=True, replace_existing=True)
|
|
|
|
def release_name(self):
|
|
if hasattr(self, "_bus_name"):
|
|
del self._bus_name
|
|
|
|
__root_interface = "org.mpris.MediaPlayer2"
|
|
__root_props = {
|
|
"CanQuit": (False, None),
|
|
"CanRaise": (True, None),
|
|
"DesktopEntry": ("mpdevil", None),
|
|
"HasTrackList": (False, None),
|
|
"Identity": ("mpdevil", None),
|
|
"SupportedUriSchemes": (dbus.Array(signature="s"), None),
|
|
"SupportedMimeTypes": (dbus.Array(signature="s"), None)
|
|
}
|
|
|
|
def __get_playback_status(self):
|
|
status = self.client.status()
|
|
return {'play': 'Playing', 'pause': 'Paused', 'stop': 'Stopped'}[status['state']]
|
|
|
|
def __set_loop_status(self, value):
|
|
if value == "Playlist":
|
|
self.client.repeat(1)
|
|
self.client.single(0)
|
|
elif value == "Track":
|
|
self.client.repeat(1)
|
|
self.client.single(1)
|
|
elif value == "None":
|
|
self.client.repeat(0)
|
|
self.client.single(0)
|
|
else:
|
|
raise dbus.exceptions.DBusException("Loop mode %r not supported" % value)
|
|
return
|
|
|
|
def __get_loop_status(self):
|
|
status = self.client.status()
|
|
if int(status['repeat']) == 1:
|
|
if int(status.get('single', 0)) == 1:
|
|
return "Track"
|
|
else:
|
|
return "Playlist"
|
|
else:
|
|
return "None"
|
|
|
|
def __set_shuffle(self, value):
|
|
self.client.random(value)
|
|
return
|
|
|
|
def __get_shuffle(self):
|
|
if int(self.client.status()['random']) == 1:
|
|
return True
|
|
else:
|
|
return False
|
|
|
|
def __get_metadata(self):
|
|
return dbus.Dictionary(self.metadata, signature='sv')
|
|
|
|
def __get_volume(self):
|
|
vol = float(self.client.status().get('volume', 0))
|
|
if vol > 0:
|
|
return vol / 100.0
|
|
else:
|
|
return 0.0
|
|
|
|
def __set_volume(self, value):
|
|
if value >= 0 and value <= 1:
|
|
self.client.setvol(int(value * 100))
|
|
return
|
|
|
|
def __get_position(self):
|
|
status = self.client.status()
|
|
if 'time' in status:
|
|
current, end = status['time'].split(':')
|
|
return dbus.Int64((int(current) * 1000000))
|
|
else:
|
|
return dbus.Int64(0)
|
|
|
|
def __get_can_next_prev(self):
|
|
status = self.client.status()
|
|
if status['state'] == "stop":
|
|
return False
|
|
else:
|
|
return True
|
|
|
|
__player_interface = "org.mpris.MediaPlayer2.Player"
|
|
__player_props = {
|
|
"PlaybackStatus": (__get_playback_status, None),
|
|
"LoopStatus": (__get_loop_status, __set_loop_status),
|
|
"Rate": (1.0, None),
|
|
"Shuffle": (__get_shuffle, __set_shuffle),
|
|
"Metadata": (__get_metadata, None),
|
|
"Volume": (__get_volume, __set_volume),
|
|
"Position": (__get_position, None),
|
|
"MinimumRate": (1.0, None),
|
|
"MaximumRate": (1.0, None),
|
|
"CanGoNext": (__get_can_next_prev, None),
|
|
"CanGoPrevious": (__get_can_next_prev, None),
|
|
"CanPlay": (True, None),
|
|
"CanPause": (True, None),
|
|
"CanSeek": (True, None),
|
|
"CanControl": (True, None),
|
|
}
|
|
|
|
__prop_mapping = {
|
|
__player_interface: __player_props,
|
|
__root_interface: __root_props,
|
|
}
|
|
|
|
@dbus.service.method(__introspect_interface)
|
|
def Introspect(self):
|
|
return MPRIS2_INTROSPECTION
|
|
|
|
@dbus.service.signal(__prop_interface, signature="sa{sv}as")
|
|
def PropertiesChanged(self, interface, changed_properties, invalidated_properties):
|
|
pass
|
|
|
|
@dbus.service.method(__prop_interface, in_signature="ss", out_signature="v")
|
|
def Get(self, interface, prop):
|
|
getter, setter = self.__prop_mapping[interface][prop]
|
|
if callable(getter):
|
|
return getter(self)
|
|
return getter
|
|
|
|
@dbus.service.method(__prop_interface, in_signature="ssv", out_signature="")
|
|
def Set(self, interface, prop, value):
|
|
getter, setter = self.__prop_mapping[interface][prop]
|
|
if setter is not None:
|
|
setter(self, value)
|
|
|
|
@dbus.service.method(__prop_interface, in_signature="s", out_signature="a{sv}")
|
|
def GetAll(self, interface):
|
|
read_props = {}
|
|
props = self.__prop_mapping[interface]
|
|
for key, (getter, setter) in props.items():
|
|
if callable(getter):
|
|
getter = getter(self)
|
|
read_props[key] = getter
|
|
return read_props
|
|
|
|
def update_property(self, interface, prop):
|
|
getter, setter = self.__prop_mapping[interface][prop]
|
|
if callable(getter):
|
|
value = getter(self)
|
|
else:
|
|
value = getter
|
|
self.PropertiesChanged(interface, {prop: value}, [])
|
|
return value
|
|
|
|
# Root methods
|
|
@dbus.service.method(__root_interface, in_signature='', out_signature='')
|
|
def Raise(self):
|
|
self.window.present()
|
|
return
|
|
|
|
@dbus.service.method(__root_interface, in_signature='', out_signature='')
|
|
def Quit(self):
|
|
return
|
|
|
|
# Player methods
|
|
@dbus.service.method(__player_interface, in_signature='', out_signature='')
|
|
def Next(self):
|
|
self.client.next()
|
|
return
|
|
|
|
@dbus.service.method(__player_interface, in_signature='', out_signature='')
|
|
def Previous(self):
|
|
self.client.previous()
|
|
return
|
|
|
|
@dbus.service.method(__player_interface, in_signature='', out_signature='')
|
|
def Pause(self):
|
|
self.client.pause(1)
|
|
return
|
|
|
|
@dbus.service.method(__player_interface, in_signature='', out_signature='')
|
|
def PlayPause(self):
|
|
status = self.client.status()
|
|
if status['state'] == 'play':
|
|
self.client.pause(1)
|
|
else:
|
|
self.client.play()
|
|
return
|
|
|
|
@dbus.service.method(__player_interface, in_signature='', out_signature='')
|
|
def Stop(self):
|
|
self.client.stop()
|
|
return
|
|
|
|
@dbus.service.method(__player_interface, in_signature='', out_signature='')
|
|
def Play(self):
|
|
self.client.play()
|
|
return
|
|
|
|
@dbus.service.method(__player_interface, in_signature='x', out_signature='')
|
|
def Seek(self, offset): #TODO
|
|
status = self.client.status()
|
|
current, end = status['time'].split(':')
|
|
current = int(current)
|
|
end = int(end)
|
|
offset = int(offset) / 1000000
|
|
if current + offset <= end:
|
|
position = current + offset
|
|
if position < 0:
|
|
position = 0
|
|
self.client.seekid(int(status['songid']), position)
|
|
self.Seeked(position * 1000000)
|
|
return
|
|
|
|
@dbus.service.method(__player_interface, in_signature='ox', out_signature='')
|
|
def SetPosition(self, trackid, position):
|
|
song = self.client.currentsong()
|
|
# FIXME: use real dbus objects
|
|
if str(trackid) != '/org/mpris/MediaPlayer2/Track/%s' % song['id']:
|
|
return
|
|
# Convert position to seconds
|
|
position = int(position) / 1000000
|
|
if position <= int(song['time']):
|
|
self.client.seekid(int(song['id']), position)
|
|
self.Seeked(position * 1000000)
|
|
return
|
|
|
|
@dbus.service.signal(__player_interface, signature='x')
|
|
def Seeked(self, position):
|
|
return float(position)
|
|
|
|
@dbus.service.method(__player_interface, in_signature='', out_signature='')
|
|
def OpenUri(self):
|
|
return
|
|
|
|
class Settings(Gio.Settings):
|
|
BASE_KEY = "org.mpdevil"
|
|
def __init__(self):
|
|
super().__init__(schema=self.BASE_KEY)
|
|
if len(self.get_value("profiles")) < (self.get_int("active-profile")+1):
|
|
self.set_int("active-profile", 0)
|
|
|
|
def array_append(self, vtype, key, value): #append to Gio.Settings (self.settings) array
|
|
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
|
|
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
|
|
array=self.get_value(key).unpack()
|
|
array[pos]=value
|
|
self.set_value(key, GLib.Variant(vtype, array))
|
|
|
|
def get_gtk_icon_size(self, key):
|
|
icon_size=self.get_int(key)
|
|
if icon_size == 16:
|
|
return Gtk.IconSize.BUTTON
|
|
elif icon_size == 24:
|
|
return Gtk.IconSize.LARGE_TOOLBAR
|
|
elif icon_size == 32:
|
|
return Gtk.IconSize.DND
|
|
elif icon_size == 48:
|
|
return Gtk.IconSize.DIALOG
|
|
else:
|
|
# return Gtk.IconSize.INVALID
|
|
raise ValueError
|
|
|
|
def get_artist_type(self):
|
|
if self.get_boolean("use-album-artist"):
|
|
return ("albumartist")
|
|
else:
|
|
return ("artist")
|
|
|
|
class SongsView(Gtk.ScrolledWindow):
|
|
def __init__(self, client, show_album=True, sort_enable=True):
|
|
Gtk.ScrolledWindow.__init__(self)
|
|
self.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
|
|
|
|
#add vars
|
|
self.client=client
|
|
|
|
#store
|
|
#(track, title, artist, album, duration, file)
|
|
self.store = Gtk.ListStore(str, str, str, str, str, str)
|
|
|
|
#TreeView
|
|
self.treeview = Gtk.TreeView(model=self.store)
|
|
self.treeview.set_search_column(-1)
|
|
self.treeview.columns_autosize()
|
|
|
|
#selection
|
|
self.selection = self.treeview.get_selection()
|
|
self.selection.set_mode(Gtk.SelectionMode.SINGLE)
|
|
|
|
#columns
|
|
renderer_text = Gtk.CellRendererText()
|
|
renderer_text_ralign = Gtk.CellRendererText(xalign=1.0)
|
|
|
|
self.column_track = Gtk.TreeViewColumn(_("No"), renderer_text_ralign, text=0)
|
|
self.column_track.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
|
|
self.column_track.set_property("resizable", False)
|
|
self.treeview.append_column(self.column_track)
|
|
|
|
self.column_title = Gtk.TreeViewColumn(_("Title"), renderer_text, text=1)
|
|
self.column_title.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
|
|
self.column_title.set_property("resizable", False)
|
|
self.treeview.append_column(self.column_title)
|
|
|
|
self.column_artist = Gtk.TreeViewColumn(_("Artist"), renderer_text, text=2)
|
|
self.column_artist.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
|
|
self.column_artist.set_property("resizable", False)
|
|
self.treeview.append_column(self.column_artist)
|
|
|
|
self.column_album = Gtk.TreeViewColumn(_("Album"), renderer_text, text=3)
|
|
self.column_album.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
|
|
self.column_album.set_property("resizable", False)
|
|
if show_album:
|
|
self.treeview.append_column(self.column_album)
|
|
|
|
self.column_time = Gtk.TreeViewColumn(_("Length"), renderer_text, text=4)
|
|
self.column_time.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
|
|
self.column_time.set_property("resizable", False)
|
|
self.treeview.append_column(self.column_time)
|
|
|
|
if sort_enable:
|
|
self.column_track.set_sort_column_id(0)
|
|
self.column_title.set_sort_column_id(1)
|
|
self.column_artist.set_sort_column_id(2)
|
|
self.column_album.set_sort_column_id(3)
|
|
self.column_time.set_sort_column_id(4)
|
|
|
|
#connect
|
|
self.treeview.connect("row-activated", self.on_row_activated)
|
|
self.treeview.connect("button-press-event", self.on_button_press_event)
|
|
self.key_press_event=self.treeview.connect("key-press-event", self.on_key_press_event)
|
|
|
|
self.add(self.treeview)
|
|
|
|
def on_row_activated(self, widget, path, view_column):
|
|
self.client.files_to_playlist([self.store[path][5]], False, True)
|
|
|
|
def on_button_press_event(self, widget, event):
|
|
if event.button == 1 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
try:
|
|
path = widget.get_path_at_pos(int(event.x), int(event.y))[0]
|
|
self.client.files_to_playlist([self.store[path][5]], False)
|
|
except:
|
|
pass
|
|
elif event.button == 2 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
try:
|
|
path = widget.get_path_at_pos(int(event.x), int(event.y))[0]
|
|
self.client.files_to_playlist([self.store[path][5]], True)
|
|
except:
|
|
pass
|
|
|
|
def on_key_press_event(self, widget, event):
|
|
self.treeview.handler_block(self.key_press_event)
|
|
if event.keyval == 112: #p
|
|
treeview, treeiter=self.selection.get_selected()
|
|
if not treeiter == None:
|
|
self.client.files_to_playlist([self.store.get_value(treeiter, 5)], False)
|
|
elif event.keyval == 97: #a
|
|
treeview, treeiter=self.selection.get_selected()
|
|
if not treeiter == None:
|
|
self.client.files_to_playlist([self.store.get_value(treeiter, 5)], True)
|
|
# elif event.keyval == 65383: #menu key
|
|
self.treeview.handler_unblock(self.key_press_event)
|
|
|
|
def populate(self, songs):
|
|
for song in songs:
|
|
try:
|
|
title=song["title"]
|
|
except:
|
|
title=_("Unknown Title")
|
|
try:
|
|
track=song["track"]
|
|
except:
|
|
track="0"
|
|
try:
|
|
artist=song["artist"]
|
|
except:
|
|
artist=_("Unknown Artist")
|
|
try:
|
|
album=song["album"]
|
|
except:
|
|
album=_("Unknown Album")
|
|
try:
|
|
dura=float(song["duration"])
|
|
except:
|
|
dura=0.0
|
|
duration=str(datetime.timedelta(seconds=int(dura)))
|
|
self.store.append([track, title, artist, album, duration, song["file"]])
|
|
|
|
def clear(self):
|
|
self.store.clear()
|
|
|
|
def count(self):
|
|
return len(self.store)
|
|
|
|
class AlbumDialog(Gtk.Dialog):
|
|
def __init__(self, parent, client, settings, album, artist, year):
|
|
Gtk.Dialog.__init__(self, transient_for=parent)
|
|
self.add_buttons(Gtk.STOCK_ADD, Gtk.ResponseType.ACCEPT, Gtk.STOCK_MEDIA_PLAY, Gtk.ResponseType.YES, Gtk.STOCK_OPEN, Gtk.ResponseType.OK, Gtk.STOCK_CLOSE, Gtk.ResponseType.CLOSE)
|
|
self.set_default_size(800, 600)
|
|
|
|
#title
|
|
if year == "":
|
|
self.set_title(artist+" - "+album)
|
|
else:
|
|
self.set_title(artist+" - "+album+" ("+year+")")
|
|
|
|
#adding vars
|
|
self.client=client
|
|
self.settings=settings
|
|
|
|
#metadata
|
|
self.album=album
|
|
self.artist=artist
|
|
self.year=year
|
|
|
|
#songs view
|
|
self.songs_view=SongsView(self.client, False, False)
|
|
self.songs_view.populate(self.client.find("album", self.album, "date", self.year, self.settings.get_artist_type(), self.artist))
|
|
|
|
#packing
|
|
self.vbox.pack_start(self.songs_view, True, True, 0) #vbox default widget of dialogs
|
|
self.vbox.set_spacing(6)
|
|
self.show_all()
|
|
|
|
def open(self):
|
|
response = self.run()
|
|
if response == Gtk.ResponseType.OK:
|
|
self.client.album_to_playlist(self.album, self.artist, self.year, False)
|
|
elif response == Gtk.ResponseType.ACCEPT:
|
|
self.client.album_to_playlist(self.album, self.artist, self.year, True)
|
|
elif response == Gtk.ResponseType.YES:
|
|
self.client.album_to_playlist(self.album, self.artist, self.year, False, True)
|
|
|
|
class GenreSelect(Gtk.ComboBoxText):
|
|
def __init__(self, client, settings):
|
|
Gtk.ComboBoxText.__init__(self)
|
|
|
|
#adding vars
|
|
self.client=client
|
|
self.settings=settings
|
|
|
|
#connect
|
|
self.changed=self.connect("changed", self.on_changed)
|
|
self.update_signal=self.client.emitter.connect("update", self.refresh)
|
|
|
|
def deactivate(self):
|
|
self.set_active(0)
|
|
|
|
def refresh(self, *args):
|
|
self.handler_block(self.changed)
|
|
self.remove_all()
|
|
self.append_text(_("all genres"))
|
|
for genre in self.client.list("genre"):
|
|
self.append_text(genre)
|
|
self.set_active(0)
|
|
self.handler_unblock(self.changed)
|
|
|
|
def clear(self, *args):
|
|
self.handler_block(self.changed)
|
|
self.remove_all()
|
|
self.handler_unblock(self.changed)
|
|
|
|
def get_value(self):
|
|
if self.get_active() == 0:
|
|
return None
|
|
else:
|
|
return self.get_active_text()
|
|
|
|
def on_changed(self, *args):
|
|
self.client.emitter.handler_block(self.update_signal)
|
|
self.client.emitter.emit("update")
|
|
self.client.emitter.handler_unblock(self.update_signal)
|
|
|
|
class ArtistView(FocusFrame):
|
|
def __init__(self, client, settings, genre_select):
|
|
FocusFrame.__init__(self)
|
|
|
|
#adding vars
|
|
self.client=client
|
|
self.settings=settings
|
|
self.genre_select=genre_select
|
|
self.last_artist_path=None
|
|
|
|
#artistStore
|
|
#(name, weight, initial-letter, weight-initials)
|
|
self.store = Gtk.ListStore(str, Pango.Weight, str, Pango.Weight)
|
|
|
|
#TreeView
|
|
self.treeview = Gtk.TreeView(model=self.store)
|
|
self.treeview.set_search_column(0)
|
|
self.treeview.columns_autosize()
|
|
self.treeview.set_property("activate-on-single-click", True)
|
|
|
|
#artistSelection
|
|
self.selection = self.treeview.get_selection()
|
|
self.selection.set_mode(Gtk.SelectionMode.SINGLE)
|
|
|
|
#Columns
|
|
renderer_text_malign = Gtk.CellRendererText(xalign=0.5)
|
|
self.column_initials = Gtk.TreeViewColumn("", renderer_text_malign, text=2, weight=3)
|
|
self.column_initials.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
|
|
self.column_initials.set_property("resizable", False)
|
|
self.column_initials.set_visible(self.settings.get_boolean("show-initials"))
|
|
self.treeview.append_column(self.column_initials)
|
|
|
|
renderer_text = Gtk.CellRendererText()
|
|
self.column_name = Gtk.TreeViewColumn("", renderer_text, text=0, weight=1)
|
|
self.column_name.set_sizing(Gtk.TreeViewColumnSizing.AUTOSIZE)
|
|
self.column_name.set_property("resizable", False)
|
|
self.treeview.append_column(self.column_name)
|
|
|
|
#scroll
|
|
scroll=Gtk.ScrolledWindow()
|
|
scroll.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
|
|
scroll.add(self.treeview)
|
|
|
|
#connect
|
|
self.treeview.connect("row-activated", self.on_row_activated)
|
|
self.settings.connect("changed::use-album-artist", self.refresh)
|
|
self.settings.connect("changed::show-initials", self.on_show_initials_settings_changed)
|
|
self.client.emitter.connect("update", self.refresh)
|
|
|
|
self.set_widget(self.treeview)
|
|
self.add(scroll)
|
|
|
|
@GObject.Signal
|
|
def artists_changed(self):
|
|
pass
|
|
|
|
def clear(self):
|
|
self.store.clear()
|
|
self.last_artist_iter=None
|
|
|
|
def refresh(self, *args):
|
|
self.selection.set_mode(Gtk.SelectionMode.NONE)
|
|
self.clear()
|
|
if self.settings.get_artist_type() == "albumartist":
|
|
self.column_name.set_title(_("Album Artist"))
|
|
else:
|
|
self.column_name.set_title(_("Artist"))
|
|
self.store.append([_("all artists"), Pango.Weight.BOOK, "", Pango.Weight.BOOK])
|
|
genre=self.genre_select.get_value()
|
|
if genre == None:
|
|
artists=self.client.list(self.settings.get_artist_type())
|
|
else:
|
|
artists=self.client.list(self.settings.get_artist_type(), "genre", genre)
|
|
current_char=""
|
|
for artist in artists:
|
|
try:
|
|
if current_char != artist[0]:
|
|
self.store.append([artist, Pango.Weight.BOOK, artist[0], Pango.Weight.BOLD])
|
|
current_char=artist[0]
|
|
else:
|
|
self.store.append([artist, Pango.Weight.BOOK, "", Pango.Weight.BOOK])
|
|
except:
|
|
self.store.append([artist, Pango.Weight.BOOK, "", Pango.Weight.BOOK])
|
|
self.selection.set_mode(Gtk.SelectionMode.SINGLE)
|
|
|
|
def get_selected_artists(self):
|
|
artists=[]
|
|
if self.store[Gtk.TreePath(0)][1] == Pango.Weight.BOLD:
|
|
for row in self.store:
|
|
artists.append(row[0])
|
|
return artists[1:]
|
|
else:
|
|
for row in self.store:
|
|
if row[1] == Pango.Weight.BOLD:
|
|
artists.append(row[0])
|
|
break
|
|
return artists
|
|
|
|
def on_row_activated(self, widget, path, view_column):
|
|
if self.last_artist_path != None:
|
|
try:
|
|
self.store[self.last_artist_path][1]=Pango.Weight.BOOK
|
|
except:
|
|
pass
|
|
self.last_artist_path=path
|
|
self.store[path][1]=Pango.Weight.BOLD
|
|
self.emit("artists_changed")
|
|
|
|
def on_show_initials_settings_changed(self, *args):
|
|
self.column_initials.set_visible(self.settings.get_boolean("show-initials"))
|
|
|
|
class AlbumIconView(Gtk.IconView):
|
|
def __init__(self, client, settings, genre_select, window):
|
|
Gtk.IconView.__init__(self)
|
|
|
|
#adding vars
|
|
self.settings=settings
|
|
self.client=client
|
|
self.genre_select=genre_select
|
|
self.window=window
|
|
self.stop_flag=True
|
|
|
|
#cover, display_label, tooltip(titles), album, year, artist
|
|
self.store = Gtk.ListStore(GdkPixbuf.Pixbuf, str, str, str, str, str)
|
|
self.sort_settings()
|
|
|
|
#iconview
|
|
self.set_model(self.store)
|
|
self.set_pixbuf_column(0)
|
|
self.set_text_column(1)
|
|
self.set_item_width(0)
|
|
self.tooltip_settings()
|
|
|
|
#connect
|
|
self.connect("item-activated", self.on_item_activated)
|
|
self.connect("button-press-event", self.on_button_press_event)
|
|
self.key_press_event=self.connect("key-press-event", self.on_key_press_event)
|
|
self.settings.connect("changed::show-album-view-tooltips", self.tooltip_settings)
|
|
self.settings.connect("changed::sort-albums-by-year", self.sort_settings)
|
|
|
|
@GObject.Signal
|
|
def done(self):
|
|
self.stop_flag=True
|
|
pass
|
|
|
|
def tooltip_settings(self, *args):
|
|
if self.settings.get_boolean("show-album-view-tooltips"):
|
|
self.set_tooltip_column(2)
|
|
else:
|
|
self.set_tooltip_column(-1)
|
|
|
|
def sort_settings(self, *args):
|
|
if self.settings.get_boolean("sort-albums-by-year"):
|
|
self.store.set_sort_column_id(4, Gtk.SortType.ASCENDING)
|
|
else:
|
|
self.store.set_sort_column_id(1, Gtk.SortType.ASCENDING)
|
|
return False
|
|
|
|
def add_row(self, row, cover, size):
|
|
row[0]=cover.get_pixbuf(size)
|
|
self.store.append(row)
|
|
return False
|
|
|
|
def populate(self, artists):
|
|
self.stop_flag=False
|
|
#prepare albmus list
|
|
self.store.clear()
|
|
albums=[]
|
|
genre=self.genre_select.get_value()
|
|
artist_type=self.settings.get_artist_type()
|
|
for artist in artists:
|
|
try: #client cloud meanwhile disconnect
|
|
if not self.stop_flag:
|
|
if genre == None:
|
|
album_candidates=self.client.list("album", artist_type, artist)
|
|
else:
|
|
album_candidates=self.client.list("album", artist_type, artist, "genre", genre)
|
|
for album in album_candidates:
|
|
years=self.client.list("date", "album", album, artist_type, artist)
|
|
for year in years:
|
|
songs=self.client.find("album", album, "date", year, artist_type, artist)
|
|
albums.append({"artist": artist, "album": album, "year": year, "songs": songs})
|
|
while Gtk.events_pending():
|
|
Gtk.main_iteration_do(True)
|
|
else:
|
|
GLib.idle_add(self.emit, "done")
|
|
return
|
|
except:
|
|
GLib.idle_add(self.emit, "done")
|
|
return
|
|
#display albums
|
|
if self.settings.get_boolean("sort-albums-by-year"):
|
|
albums = sorted(albums, key=lambda k: k['year'])
|
|
else:
|
|
albums = sorted(albums, key=lambda k: k['album'])
|
|
music_lib=self.settings.get_value("paths")[self.settings.get_int("active-profile")]
|
|
size=self.settings.get_int("album-cover")
|
|
z=0
|
|
for album in albums:
|
|
if not self.stop_flag:
|
|
cover=Cover(lib_path=music_lib, song_file=album["songs"][0]["file"])
|
|
#tooltip
|
|
length=float(0)
|
|
for song in album["songs"]:
|
|
try:
|
|
dura=float(song["duration"])
|
|
except:
|
|
dura=0.0
|
|
length=length+dura
|
|
length_human_readable=str(datetime.timedelta(seconds=int(length)))
|
|
tooltip=(_("%(total_tracks)i titles (%(total_length)s)") % {"total_tracks": len(album["songs"]), "total_length": length_human_readable})
|
|
if album["year"] == "":
|
|
GLib.idle_add(self.add_row, [None, album["album"], tooltip, album["album"], album["year"], album["artist"]], cover, size)
|
|
else:
|
|
GLib.idle_add(self.add_row, [None, album["album"]+" ("+album["year"]+")", tooltip, album["album"], album["year"], album["artist"]], cover, size)
|
|
if z%16 == 0:
|
|
while Gtk.events_pending():
|
|
Gtk.main_iteration_do(True)
|
|
z=z+1
|
|
else:
|
|
break
|
|
GLib.idle_add(self.emit, "done")
|
|
|
|
def scroll_to_selected_album(self):
|
|
songid=self.client.status()["songid"]
|
|
song=self.client.playlistid(songid)[0]
|
|
self.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, 3) == song["album"]:
|
|
self.set_cursor(path, None, False)
|
|
self.select_path(path)
|
|
self.scroll_to_path(path, True, 0, 0)
|
|
break
|
|
|
|
def path_to_playlist(self, path, add, force=False):
|
|
album=self.store[path][3]
|
|
year=self.store[path][4]
|
|
artist=self.store[path][5]
|
|
self.client.album_to_playlist(album, artist, year, add, force)
|
|
|
|
def open_album_dialog(self, path):
|
|
if self.client.connected():
|
|
album=self.store[path][3]
|
|
year=self.store[path][4]
|
|
artist=self.store[path][5]
|
|
album_dialog = AlbumDialog(self.window, self.client, self.settings, album, artist, year)
|
|
album_dialog.open()
|
|
album_dialog.destroy()
|
|
|
|
def on_button_press_event(self, widget, event):
|
|
path = widget.get_path_at_pos(int(event.x), int(event.y))
|
|
if not path == None:
|
|
if event.button == 1 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
self.path_to_playlist(path, False)
|
|
elif event.button == 2 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
self.path_to_playlist(path, True)
|
|
elif event.button == 3 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
self.open_album_dialog(path)
|
|
|
|
def on_key_press_event(self, widget, event):
|
|
self.handler_block(self.key_press_event)
|
|
if event.keyval == 112: #p
|
|
paths=self.get_selected_items()
|
|
if not len(paths) == 0:
|
|
self.path_to_playlist(paths[0], False)
|
|
elif event.keyval == 97: #a
|
|
paths=self.get_selected_items()
|
|
if not len(paths) == 0:
|
|
self.path_to_playlist(paths[0], True)
|
|
elif event.keyval == 65383: #menu key
|
|
paths=self.get_selected_items()
|
|
if not len(paths) == 0:
|
|
self.open_album_dialog(paths[0])
|
|
self.handler_unblock(self.key_press_event)
|
|
|
|
def on_item_activated(self, widget, path):
|
|
treeiter=self.store.get_iter(path)
|
|
selected_album=self.store.get_value(treeiter, 3)
|
|
selected_album_year=self.store.get_value(treeiter, 4)
|
|
selected_artist=self.store.get_value(treeiter, 5)
|
|
self.client.album_to_playlist(selected_album, selected_artist, selected_album_year, False, True)
|
|
|
|
class AlbumView(FocusFrame):
|
|
def __init__(self, client, settings, genre_select, window):
|
|
FocusFrame.__init__(self)
|
|
|
|
#adding vars
|
|
self.settings=settings
|
|
self.client=client
|
|
self.genre_select=genre_select
|
|
self.window=window
|
|
self.artists=[]
|
|
self.done=True
|
|
self.pending=[]
|
|
|
|
#iconview
|
|
self.iconview=AlbumIconView(self.client, self.settings, self.genre_select, self.window)
|
|
|
|
#scroll
|
|
scroll=Gtk.ScrolledWindow()
|
|
scroll.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
|
|
scroll.add(self.iconview)
|
|
|
|
#connect
|
|
self.settings.connect("changed::album-cover", self.on_settings_changed)
|
|
self.iconview.connect("done", self.on_done)
|
|
self.client.emitter.connect("update", self.clear)
|
|
self.settings.connect("changed::use-album-artist", self.clear)
|
|
|
|
self.set_widget(self.iconview)
|
|
self.add(scroll)
|
|
|
|
def clear(self, *args):
|
|
if self.done:
|
|
self.artists=[]
|
|
self.iconview.store.clear()
|
|
elif not self.clear in self.pending:
|
|
self.iconview.stop_flag=True
|
|
self.pending.append(self.clear)
|
|
|
|
def refresh(self, artists):
|
|
self.artists=artists
|
|
if self.done:
|
|
self.done=False
|
|
self.populate()
|
|
elif not self.populate in self.pending:
|
|
self.iconview.stop_flag=True
|
|
self.pending.append(self.populate)
|
|
|
|
def populate(self):
|
|
self.iconview.populate(self.artists)
|
|
|
|
def scroll_to_selected_album(self):
|
|
if self.done:
|
|
self.iconview.scroll_to_selected_album()
|
|
elif not self.scroll_to_selected_album in self.pending:
|
|
self.pending.append(self.scroll_to_selected_album)
|
|
|
|
def on_done(self, *args):
|
|
self.done=True
|
|
pending=self.pending
|
|
self.pending=[]
|
|
for p in pending:
|
|
try:
|
|
p()
|
|
except:
|
|
pass
|
|
|
|
def on_settings_changed(self, *args):
|
|
if self.done:
|
|
self.populate()
|
|
|
|
class MainCover(Gtk.Frame):
|
|
def __init__(self, client, settings, window):
|
|
Gtk.Frame.__init__(self)
|
|
#css
|
|
style_context=self.get_style_context()
|
|
provider = Gtk.CssProvider()
|
|
css = b"""* {background-color: @theme_base_color; border-radius: 6px;}"""
|
|
provider.load_from_data(css)
|
|
style_context.add_provider(provider, 800)
|
|
|
|
#adding vars
|
|
self.client=client
|
|
self.settings=settings
|
|
self.window=window
|
|
|
|
#event box
|
|
event_box=Gtk.EventBox()
|
|
event_box.set_property("border-width", 6)
|
|
|
|
#cover
|
|
self.cover=Gtk.Image.new()
|
|
self.cover.set_from_pixbuf(Cover(lib_path=self.settings.get_value("paths")[self.settings.get_int("active-profile")], song_file=None).get_pixbuf(self.settings.get_int("track-cover"))) #set to fallback cover
|
|
|
|
#connect
|
|
event_box.connect("button-press-event", self.on_button_press_event)
|
|
self.client.emitter.connect("playing_file_changed", self.refresh)
|
|
self.settings.connect("changed::track-cover", self.on_settings_changed)
|
|
|
|
event_box.add(self.cover)
|
|
self.add(event_box)
|
|
|
|
def refresh(self, *args):
|
|
try:
|
|
current_song=self.client.currentsong()
|
|
song_file=current_song['file']
|
|
except:
|
|
song_file=None
|
|
self.cover.set_from_pixbuf(Cover(lib_path=self.settings.get_value("paths")[self.settings.get_int("active-profile")], song_file=song_file).get_pixbuf(self.settings.get_int("track-cover")))
|
|
|
|
def clear(self, *args):
|
|
self.cover.set_from_pixbuf(Cover(lib_path=self.settings.get_value("paths")[self.settings.get_int("active-profile")], song_file=None).get_pixbuf(self.settings.get_int("track-cover")))
|
|
self.song_file=None
|
|
|
|
def on_button_press_event(self, widget, event):
|
|
if self.client.connected():
|
|
song=self.client.currentsong()
|
|
if not song == {}:
|
|
try:
|
|
artist=song[self.settings.get_artist_type()]
|
|
except:
|
|
try:
|
|
artist=song["artist"]
|
|
except:
|
|
artist=""
|
|
try:
|
|
album=song["album"]
|
|
except:
|
|
album=""
|
|
try:
|
|
album_year=song["date"]
|
|
except:
|
|
album_year=""
|
|
if event.button == 1 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
self.client.album_to_playlist(album, artist, album_year, False)
|
|
elif event.button == 2 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
self.client.album_to_playlist(album, artist, album_year, True)
|
|
elif event.button == 3 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
album_dialog = AlbumDialog(self.window, self.client, self.settings, album, artist, album_year)
|
|
album_dialog.open()
|
|
album_dialog.destroy()
|
|
|
|
def on_settings_changed(self, *args):
|
|
self.song_file=None
|
|
self.refresh()
|
|
|
|
class PlaylistView(Gtk.Box):
|
|
def __init__(self, client, settings):
|
|
Gtk.Box.__init__(self, orientation=Gtk.Orientation.VERTICAL)
|
|
|
|
#adding vars
|
|
self.client=client
|
|
self.settings=settings
|
|
self.playlist_version=None
|
|
self.last_song_path=None
|
|
|
|
#Store
|
|
#(track, disc, title, artist, album, duration, date, genre, file, weight)
|
|
self.store = Gtk.ListStore(str, str, str, str, str, str, str, str, str, Pango.Weight)
|
|
|
|
#TreeView
|
|
self.treeview = Gtk.TreeView(model=self.store)
|
|
self.treeview.set_search_column(2)
|
|
self.treeview.set_property("activate-on-single-click", True)
|
|
|
|
#selection
|
|
self.selection = self.treeview.get_selection()
|
|
self.selection.set_mode(Gtk.SelectionMode.SINGLE)
|
|
|
|
#Column
|
|
renderer_text = Gtk.CellRendererText()
|
|
renderer_text_ralign = Gtk.CellRendererText(xalign=1.0)
|
|
self.columns=[None, None, None, None, None, None, None, None]
|
|
|
|
self.columns[0] = Gtk.TreeViewColumn(_("No"), renderer_text_ralign, text=0, weight=9)
|
|
self.columns[0].set_property("resizable", True)
|
|
|
|
self.columns[1] = Gtk.TreeViewColumn(_("Disc"), renderer_text_ralign, text=1, weight=9)
|
|
self.columns[1].set_property("resizable", True)
|
|
|
|
self.columns[2] = Gtk.TreeViewColumn(_("Title"), renderer_text, text=2, weight=9)
|
|
self.columns[2].set_property("resizable", True)
|
|
|
|
self.columns[3] = Gtk.TreeViewColumn(_("Artist"), renderer_text, text=3, weight=9)
|
|
self.columns[3].set_property("resizable", True)
|
|
|
|
self.columns[4] = Gtk.TreeViewColumn(_("Album"), renderer_text, text=4, weight=9)
|
|
self.columns[4].set_property("resizable", True)
|
|
|
|
self.columns[5] = Gtk.TreeViewColumn(_("Length"), renderer_text, text=5, weight=9)
|
|
self.columns[5].set_property("resizable", True)
|
|
|
|
self.columns[6] = Gtk.TreeViewColumn(_("Year"), renderer_text, text=6, weight=9)
|
|
self.columns[6].set_property("resizable", True)
|
|
|
|
self.columns[7] = Gtk.TreeViewColumn(_("Genre"), renderer_text, text=7, weight=9)
|
|
self.columns[7].set_property("resizable", True)
|
|
|
|
self.load_settings()
|
|
|
|
#scroll
|
|
scroll=Gtk.ScrolledWindow()
|
|
scroll.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
|
|
scroll.add(self.treeview)
|
|
|
|
#frame
|
|
frame=FocusFrame()
|
|
frame.set_widget(self.treeview)
|
|
frame.add(scroll)
|
|
|
|
#audio infos
|
|
audio=AudioType(self.client)
|
|
|
|
#playlist info
|
|
self.playlist_info=Gtk.Label()
|
|
self.playlist_info.set_margin_start(5)
|
|
self.playlist_info.set_xalign(0)
|
|
self.playlist_info.set_ellipsize(Pango.EllipsizeMode.END)
|
|
|
|
#status bar
|
|
status_bar=Gtk.Box(orientation=Gtk.Orientation.HORIZONTAL, spacing=12)
|
|
status_bar.set_property("border-width", 1)
|
|
status_bar.pack_start(self.playlist_info, True, True, 0)
|
|
status_bar.pack_end(audio, False, False, 0)
|
|
|
|
#connect
|
|
self.treeview.connect("row-activated", self.on_row_activated)
|
|
self.key_press_event=self.treeview.connect("key-press-event", self.on_key_press_event)
|
|
self.treeview.connect("button-press-event", self.on_button_press_event)
|
|
|
|
self.client.emitter.connect("playlist", self.on_playlist_changed)
|
|
self.client.emitter.connect("playing_file_changed", self.on_file_changed)
|
|
self.client.emitter.connect("disconnected", self.on_disconnected)
|
|
|
|
self.settings.connect("changed::column-visibilities", self.load_settings)
|
|
self.settings.connect("changed::column-permutation", self.load_settings)
|
|
|
|
#packing
|
|
self.pack_start(frame, True, True, 0)
|
|
self.pack_start(Gtk.Separator.new(orientation=Gtk.Orientation.HORIZONTAL), False, False, 0)
|
|
self.pack_end(status_bar, False, False, 0)
|
|
|
|
def save_settings(self): #only saves the column sizes
|
|
columns=self.treeview.get_columns()
|
|
permutation=self.settings.get_value("column-permutation").unpack()
|
|
sizes=[0] * len(permutation)
|
|
for i in range(len(permutation)):
|
|
sizes[permutation[i]]=columns[i].get_width()
|
|
self.settings.set_value("column-sizes", GLib.Variant("ai", sizes))
|
|
|
|
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])
|
|
|
|
def scroll_to_selected_title(self):
|
|
treeview, treeiter=self.selection.get_selected()
|
|
if not treeiter == None:
|
|
path=treeview.get_path(treeiter)
|
|
self.treeview.scroll_to_cell(path, None, True, 0.25)
|
|
|
|
def refresh_playlist_info(self):
|
|
songs=self.client.playlistinfo()
|
|
if not songs == []:
|
|
whole_length=float(0)
|
|
for song in songs:
|
|
try:
|
|
dura=float(song["duration"])
|
|
except:
|
|
dura=0.0
|
|
whole_length=whole_length+dura
|
|
whole_length_human_readable=str(datetime.timedelta(seconds=int(whole_length)))
|
|
self.playlist_info.set_text(_("%(total_tracks)i titles (%(total_length)s)") % {"total_tracks": len(songs), "total_length": whole_length_human_readable})
|
|
else:
|
|
self.playlist_info.set_text("")
|
|
|
|
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)
|
|
try:
|
|
song=self.client.status()["song"]
|
|
path = Gtk.TreePath(int(song))
|
|
self.selection.select_path(path)
|
|
if self.last_song_path != None:
|
|
try:
|
|
self.store[self.last_song_path][9]=Pango.Weight.BOOK
|
|
except:
|
|
pass
|
|
self.store[path][9]=Pango.Weight.BOLD
|
|
self.last_song_path=path
|
|
self.scroll_to_selected_title()
|
|
except:
|
|
if self.last_song_path != None:
|
|
self.store[self.last_song_path][9]=Pango.Weight.BOOK
|
|
self.last_song_path=None
|
|
self.selection.unselect_all()
|
|
|
|
def clear(self, *args):
|
|
self.playlist_info.set_text("")
|
|
self.store.clear()
|
|
self.playlist_version=None
|
|
|
|
def remove_song(self, path):
|
|
self.client.delete(path) #bad song index possible
|
|
self.store.remove(self.store.get_iter(path))
|
|
self.playlist_version=self.client.status()["playlist"]
|
|
|
|
def on_key_press_event(self, widget, event):
|
|
self.treeview.handler_block(self.key_press_event)
|
|
if event.keyval == 65535: #entf
|
|
treeview, treeiter=self.selection.get_selected()
|
|
if not treeiter == None:
|
|
path=self.store.get_path(treeiter)
|
|
try:
|
|
self.remove_song(path)
|
|
except:
|
|
pass
|
|
self.treeview.handler_unblock(self.key_press_event)
|
|
|
|
def on_button_press_event(self, widget, event):
|
|
if event.button == 2 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
try:
|
|
path = widget.get_path_at_pos(int(event.x), int(event.y))[0]
|
|
self.remove_song(path)
|
|
except:
|
|
pass
|
|
|
|
def on_row_activated(self, widget, path, view_column):
|
|
self.client.play(path)
|
|
|
|
def on_playlist_changed(self, *args):
|
|
songs=[]
|
|
if not self.playlist_version == None:
|
|
songs=self.client.plchanges(self.playlist_version)
|
|
else:
|
|
songs=self.client.playlistinfo()
|
|
if not songs == []:
|
|
self.playlist_info.set_text("")
|
|
for song in songs:
|
|
try:
|
|
title=song["title"]
|
|
except:
|
|
title=_("Unknown Title")
|
|
try:
|
|
track=song["track"]
|
|
except:
|
|
track="0"
|
|
try:
|
|
disc=song["disc"]
|
|
except:
|
|
disc=""
|
|
try:
|
|
artist=song["artist"]
|
|
except:
|
|
artist=_("Unknown Artist")
|
|
try:
|
|
album=song["album"]
|
|
except:
|
|
album=_("Unknown Album")
|
|
try:
|
|
dura=float(song["duration"])
|
|
except:
|
|
dura=0.0
|
|
try:
|
|
year=song["date"]
|
|
except:
|
|
year=""
|
|
try:
|
|
genre=song["genre"]
|
|
except:
|
|
genre=""
|
|
duration=str(datetime.timedelta(seconds=int(dura )))
|
|
try:
|
|
treeiter=self.store.get_iter(song["pos"])
|
|
self.store.set(treeiter, 0, track, 1, disc, 2, title, 3, artist, 4, album, 5, duration, 6, year, 7, genre, 8, song["file"], 9, Pango.Weight.BOOK)
|
|
except:
|
|
self.store.append([track, disc, title, artist, album, duration, year, genre, song["file"], Pango.Weight.BOOK])
|
|
for i in reversed(range(int(self.client.status()["playlistlength"]), len(self.store))):
|
|
treeiter=self.store.get_iter(i)
|
|
self.store.remove(treeiter)
|
|
self.refresh_playlist_info()
|
|
if self.playlist_version == None or not songs == []:
|
|
self.refresh_selection()
|
|
self.playlist_version=self.client.status()["playlist"]
|
|
|
|
def on_file_changed(self, *args):
|
|
self.refresh_selection()
|
|
|
|
def on_disconnected(self, *args):
|
|
self.playlist_version=None
|
|
|
|
class Browser(Gtk.Box):
|
|
def __init__(self, client, settings, window):
|
|
Gtk.Box.__init__(self, orientation=Gtk.Orientation.VERTICAL)
|
|
|
|
#adding vars
|
|
self.client=client
|
|
self.settings=settings
|
|
self.window=window
|
|
self.icon_size=self.settings.get_gtk_icon_size("icon-size")
|
|
|
|
#widgets
|
|
self.back_to_album_button=Gtk.Button(image=Gtk.Image.new_from_icon_name("go-previous-symbolic", self.icon_size))
|
|
self.back_to_album_button.set_tooltip_text(_("Back to current album"))
|
|
self.search_button=Gtk.ToggleButton(image=Gtk.Image.new_from_icon_name("system-search-symbolic", self.icon_size))
|
|
self.search_button.set_tooltip_text(_("Search"))
|
|
self.genre_select=GenreSelect(self.client, self.settings)
|
|
self.artist_view=ArtistView(self.client, self.settings, self.genre_select)
|
|
self.album_view=AlbumView(self.client, self.settings, self.genre_select, self.window)
|
|
self.main_cover=MainCover(self.client, self.settings, self.window)
|
|
self.main_cover.set_property("border-width", 3)
|
|
self.playlist_view=PlaylistView(self.client, self.settings)
|
|
|
|
#connect
|
|
self.back_to_album_button.connect("clicked", self.back_to_album)
|
|
self.search_button.connect("toggled", self.on_search_toggled)
|
|
self.artist_view.connect("artists_changed", self.on_artists_changed)
|
|
self.settings.connect("changed::playlist-right", self.on_playlist_pos_settings_changed)
|
|
self.client.emitter.connect("disconnected", self.on_disconnected)
|
|
self.client.emitter.connect("reconnected", self.on_reconnected)
|
|
|
|
#packing
|
|
hbox=Gtk.Box(spacing=6)
|
|
hbox.set_property("border-width", 6)
|
|
hbox.pack_start(self.back_to_album_button, False, False, 0)
|
|
hbox.pack_start(self.search_button, False, False, 0)
|
|
hbox.pack_start(self.genre_select, True, True, 0)
|
|
|
|
self.box1=Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
|
|
self.box1.pack_start(hbox, False, False, 0)
|
|
self.box1.pack_start(Gtk.Separator.new(orientation=Gtk.Orientation.HORIZONTAL), False, False, 0)
|
|
self.box1.pack_start(self.artist_view, True, True, 0)
|
|
|
|
self.box2=Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
|
|
self.box2.pack_start(self.main_cover, False, False, 0)
|
|
self.box2.pack_start(self.playlist_view, True, True, 0)
|
|
|
|
self.paned1=Gtk.Paned.new(Gtk.Orientation.HORIZONTAL)
|
|
self.paned1.set_wide_handle(True)
|
|
|
|
self.paned2=Gtk.Paned.new(Gtk.Orientation.HORIZONTAL)
|
|
self.paned2.set_wide_handle(True)
|
|
|
|
self.paned1.pack1(self.box1, False, False)
|
|
self.paned1.pack2(self.album_view, True, False)
|
|
|
|
self.paned2.pack1(self.paned1, True, False)
|
|
self.paned2.pack2(self.box2, False, False)
|
|
|
|
self.load_settings()
|
|
self.pack_start(self.paned2, True, True, 0)
|
|
|
|
self.on_playlist_pos_settings_changed()
|
|
|
|
def save_settings(self):
|
|
self.settings.set_int("paned1", self.paned1.get_position())
|
|
self.settings.set_int("paned2", self.paned2.get_position())
|
|
self.playlist_view.save_settings()
|
|
|
|
def load_settings(self):
|
|
self.paned1.set_position(self.settings.get_int("paned1"))
|
|
self.paned2.set_position(self.settings.get_int("paned2"))
|
|
|
|
def clear(self, *args):
|
|
self.genre_select.clear()
|
|
self.artist_view.clear()
|
|
self.album_view.clear()
|
|
self.playlist_view.clear()
|
|
self.main_cover.clear()
|
|
|
|
def back_to_album(self, *args):
|
|
try: #since this can still be running when the connection is lost, various exceptions can occur
|
|
song=self.client.currentsong()
|
|
try:
|
|
artist=song[self.settings.get_artist_type()]
|
|
except:
|
|
try:
|
|
artist=song["artist"]
|
|
except:
|
|
artist=""
|
|
try:
|
|
if not song['genre'] == self.genre_select.get_value():
|
|
self.genre_select.deactivate() #deactivate genre filter to show all artists
|
|
except:
|
|
pass
|
|
if len(self.artist_view.get_selected_artists()) <= 1:
|
|
row_num=len(self.artist_view.store)
|
|
for i in range(0, row_num):
|
|
path=Gtk.TreePath(i)
|
|
if self.artist_view.store[path][0] == artist:
|
|
self.artist_view.treeview.set_cursor(path, None, False)
|
|
if not self.artist_view.get_selected_artists() == [artist]:
|
|
self.artist_view.treeview.row_activated(path, self.artist_view.column_name)
|
|
break
|
|
else:
|
|
self.artist_view.treeview.set_cursor(Gtk.TreePath(0), None, False) #set cursor to 'all artists'
|
|
self.album_view.scroll_to_selected_album()
|
|
except:
|
|
pass
|
|
|
|
def on_search_toggled(self, widget):
|
|
if widget.get_active():
|
|
if self.client.connected():
|
|
def set_active(*args):
|
|
self.search_button.set_active(False)
|
|
self.search_win = SearchWindow(self.client)
|
|
self.search_win.connect("destroy", set_active)
|
|
else:
|
|
self.search_win.destroy()
|
|
|
|
def on_reconnected(self, *args):
|
|
self.back_to_album_button.set_sensitive(True)
|
|
self.search_button.set_sensitive(True)
|
|
self.genre_select.set_sensitive(True)
|
|
|
|
def on_disconnected(self, *args):
|
|
self.clear()
|
|
self.back_to_album_button.set_sensitive(False)
|
|
self.search_button.set_active(False)
|
|
self.search_button.set_sensitive(False)
|
|
self.genre_select.set_sensitive(False)
|
|
|
|
def on_artists_changed(self, *args):
|
|
artists=self.artist_view.get_selected_artists()
|
|
self.album_view.refresh(artists)
|
|
|
|
def on_playlist_pos_settings_changed(self, *args):
|
|
if self.settings.get_boolean("playlist-right"):
|
|
self.box2.set_orientation(Gtk.Orientation.VERTICAL)
|
|
self.paned2.set_orientation(Gtk.Orientation.HORIZONTAL)
|
|
else:
|
|
self.box2.set_orientation(Gtk.Orientation.HORIZONTAL)
|
|
self.paned2.set_orientation(Gtk.Orientation.VERTICAL)
|
|
|
|
class ProfileSettings(Gtk.Grid):
|
|
def __init__(self, parent, settings):
|
|
Gtk.Grid.__init__(self)
|
|
self.set_row_spacing(6)
|
|
self.set_column_spacing(12)
|
|
self.set_property("border-width", 18)
|
|
|
|
#adding vars
|
|
self.settings = settings
|
|
|
|
#widgets
|
|
self.profiles_combo=Gtk.ComboBoxText()
|
|
self.profiles_combo.set_entry_text_column(0)
|
|
|
|
add_button=Gtk.Button(label=None, image=Gtk.Image(stock=Gtk.STOCK_ADD))
|
|
delete_button=Gtk.Button(label=None, image=Gtk.Image(stock=Gtk.STOCK_DELETE))
|
|
add_delete_buttons=Gtk.ButtonBox()
|
|
add_delete_buttons.set_property("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)
|
|
|
|
self.profile_entry=Gtk.Entry()
|
|
self.host_entry=Gtk.Entry()
|
|
self.port_entry=IntEntry(0, 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)
|
|
self.password_entry=Gtk.Entry()
|
|
self.password_entry.set_visibility(False)
|
|
self.path_entry=Gtk.Entry()
|
|
self.path_select_button=Gtk.Button(image=Gtk.Image(stock=Gtk.STOCK_OPEN))
|
|
path_box=Gtk.Box(spacing=6)
|
|
path_box.pack_start(self.path_entry, True, True, 0)
|
|
path_box.pack_start(self.path_select_button, False, False, 0)
|
|
|
|
profiles_label=Gtk.Label(label=_("Profile:"))
|
|
profiles_label.set_xalign(1)
|
|
profile_label=Gtk.Label(label=_("Name:"))
|
|
profile_label.set_xalign(1)
|
|
host_label=Gtk.Label(label=_("Host:"))
|
|
host_label.set_xalign(1)
|
|
password_label=Gtk.Label(label=_("Password:"))
|
|
password_label.set_xalign(1)
|
|
path_label=Gtk.Label(label=_("Music lib:"))
|
|
path_label.set_xalign(1)
|
|
|
|
#connect
|
|
self.profile_entry_changed=self.profile_entry.connect("changed", self.on_profile_entry_changed)
|
|
self.host_entry_changed=self.host_entry.connect("changed", self.on_host_entry_changed)
|
|
self.port_entry_changed=self.port_entry.connect("value-changed", self.on_port_entry_changed)
|
|
self.password_entry_changed=self.password_entry.connect("changed", self.on_password_entry_changed)
|
|
self.path_entry_changed=self.path_entry.connect("changed", self.on_path_entry_changed)
|
|
self.path_select_button.connect("clicked", self.on_path_select_button_clicked, parent)
|
|
add_button.connect("clicked", self.on_add_button_clicked)
|
|
delete_button.connect("clicked", self.on_delete_button_clicked)
|
|
self.profiles_combo_changed=self.profiles_combo.connect("changed", self.on_profiles_changed)
|
|
|
|
self.profiles_combo_reload()
|
|
self.profiles_combo.set_active(0)
|
|
|
|
#packing
|
|
self.add(profiles_label)
|
|
self.attach_next_to(profile_label, profiles_label, Gtk.PositionType.BOTTOM, 1, 1)
|
|
self.attach_next_to(host_label, profile_label, Gtk.PositionType.BOTTOM, 1, 1)
|
|
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(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(self.profile_entry, profile_label, Gtk.PositionType.RIGHT, 2, 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)
|
|
self.attach_next_to(path_box, path_label, Gtk.PositionType.RIGHT, 2, 1)
|
|
|
|
def profiles_combo_reload(self, *args):
|
|
self.profiles_combo.handler_block(self.profiles_combo_changed)
|
|
self.profile_entry.handler_block(self.profile_entry_changed)
|
|
self.host_entry.handler_block(self.host_entry_changed)
|
|
self.port_entry.handler_block(self.port_entry_changed)
|
|
|
|
self.profiles_combo.remove_all()
|
|
for profile in self.settings.get_value("profiles"):
|
|
self.profiles_combo.append_text(profile)
|
|
|
|
self.profiles_combo.handler_unblock(self.profiles_combo_changed)
|
|
self.profile_entry.handler_unblock(self.profile_entry_changed)
|
|
self.host_entry.handler_unblock(self.host_entry_changed)
|
|
self.port_entry.handler_unblock(self.port_entry_changed)
|
|
|
|
def on_add_button_clicked(self, *args):
|
|
pos=self.profiles_combo.get_active()
|
|
self.settings.array_append('as', "profiles", "new profile")
|
|
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.profiles_combo_reload()
|
|
self.profiles_combo.set_active(pos)
|
|
|
|
def on_delete_button_clicked(self, *args):
|
|
pos=self.profiles_combo.get_active()
|
|
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)
|
|
if len(self.settings.get_value("profiles")) == 0:
|
|
self.on_add_button_clicked()
|
|
else:
|
|
self.profiles_combo_reload()
|
|
self.profiles_combo.set_active(0)
|
|
|
|
def on_profile_entry_changed(self, *args):
|
|
pos=self.profiles_combo.get_active()
|
|
self.settings.array_modify('as', "profiles", pos, self.profile_entry.get_text())
|
|
self.profiles_combo_reload()
|
|
self.profiles_combo.set_active(pos)
|
|
|
|
def on_host_entry_changed(self, *args):
|
|
self.settings.array_modify('as', "hosts", self.profiles_combo.get_active(), self.host_entry.get_text())
|
|
|
|
def on_port_entry_changed(self, *args):
|
|
self.settings.array_modify('ai', "ports", self.profiles_combo.get_active(), self.port_entry.get_int())
|
|
|
|
def on_password_entry_changed(self, *args):
|
|
self.settings.array_modify('as', "passwords", self.profiles_combo.get_active(), self.password_entry.get_text())
|
|
|
|
def on_path_entry_changed(self, *args):
|
|
self.settings.array_modify('as', "paths", self.profiles_combo.get_active(), self.path_entry.get_text())
|
|
|
|
def on_path_select_button_clicked(self, widget, parent):
|
|
dialog = Gtk.FileChooserDialog(title=_("Choose directory"), transient_for=parent, action=Gtk.FileChooserAction.SELECT_FOLDER)
|
|
dialog.add_buttons(Gtk.STOCK_CANCEL, Gtk.ResponseType.CANCEL)
|
|
dialog.add_buttons(Gtk.STOCK_OK, Gtk.ResponseType.OK)
|
|
dialog.set_default_size(800, 400)
|
|
dialog.set_current_folder(self.settings.get_value("paths")[self.profiles_combo.get_active()])
|
|
response = dialog.run()
|
|
if response == Gtk.ResponseType.OK:
|
|
self.settings.array_modify('as', "paths", self.profiles_combo.get_active(), dialog.get_filename())
|
|
self.path_entry.set_text(dialog.get_filename())
|
|
dialog.destroy()
|
|
|
|
def on_profiles_changed(self, *args):
|
|
active=self.profiles_combo.get_active()
|
|
self.profile_entry.handler_block(self.profile_entry_changed)
|
|
self.host_entry.handler_block(self.host_entry_changed)
|
|
self.port_entry.handler_block(self.port_entry_changed)
|
|
self.password_entry.handler_block(self.password_entry_changed)
|
|
|
|
self.profile_entry.set_text(self.settings.get_value("profiles")[active])
|
|
self.host_entry.set_text(self.settings.get_value("hosts")[active])
|
|
self.port_entry.set_int(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.profile_entry.handler_unblock(self.profile_entry_changed)
|
|
self.host_entry.handler_unblock(self.host_entry_changed)
|
|
self.port_entry.handler_unblock(self.port_entry_changed)
|
|
self.password_entry.handler_unblock(self.password_entry_changed)
|
|
|
|
class GeneralSettings(Gtk.Box):
|
|
def __init__(self, settings):
|
|
Gtk.Box.__init__(self, orientation=Gtk.Orientation.VERTICAL, spacing=6)
|
|
self.set_property("border-width", 18)
|
|
|
|
#adding vars
|
|
self.settings = settings
|
|
|
|
#widgets
|
|
track_cover_label=Gtk.Label(label=_("Main cover size:"))
|
|
track_cover_label.set_xalign(0)
|
|
track_cover_size=IntEntry(self.settings.get_int("track-cover"), 100, 1200, 10)
|
|
|
|
album_cover_label=Gtk.Label(label=_("Album view cover size:"))
|
|
album_cover_label.set_xalign(0)
|
|
album_cover_size=IntEntry(self.settings.get_int("album-cover"), 50, 600, 10)
|
|
|
|
icon_size_label1=Gtk.Label(label=_("Button icon size:"))
|
|
icon_size_label1.set_xalign(0)
|
|
icon_size_label2=Gtk.Label(label=_("(restart required)"))
|
|
icon_size_label2.set_xalign(0)
|
|
icon_size_label2.set_sensitive(False)
|
|
icon_size_combo=Gtk.ComboBoxText()
|
|
icon_size_combo.set_entry_text_column(0)
|
|
sizes=[16, 24, 32, 48]
|
|
for i in sizes:
|
|
icon_size_combo.append_text(str(i))
|
|
icon_size_combo.set_active(sizes.index(self.settings.get_int("icon-size")))
|
|
|
|
combo_settings={}
|
|
combo_settings_data=[(_("Sort albums by:"), _("name"), _("year"), "sort-albums-by-year"), \
|
|
(_("Position of playlist:"), _("bottom"), _("right"), "playlist-right")]
|
|
for data in combo_settings_data:
|
|
combo_settings[data[3]]=(Gtk.Label(), Gtk.ComboBoxText())
|
|
combo_settings[data[3]][0].set_label(data[0])
|
|
combo_settings[data[3]][0].set_xalign(0)
|
|
combo_settings[data[3]][1].set_entry_text_column(0)
|
|
combo_settings[data[3]][1].append_text(data[1])
|
|
combo_settings[data[3]][1].append_text(data[2])
|
|
if self.settings.get_boolean(data[3]):
|
|
combo_settings[data[3]][1].set_active(1)
|
|
else:
|
|
combo_settings[data[3]][1].set_active(0)
|
|
combo_settings[data[3]][1].connect("changed", self.on_combo_changed, data[3])
|
|
|
|
#headings
|
|
view_heading=Gtk.Label()
|
|
view_heading.set_markup(_("<b>View</b>"))
|
|
view_heading.set_xalign(0)
|
|
behavior_heading=Gtk.Label()
|
|
behavior_heading.set_markup(_("<b>Behavior</b>"))
|
|
behavior_heading.set_xalign(0)
|
|
|
|
#check buttons
|
|
check_buttons={}
|
|
settings_list=[(_("Show stop button"), "show-stop"), \
|
|
(_("Show initials in artist view"), "show-initials"), \
|
|
(_("Show tooltips in album view"), "show-album-view-tooltips"), \
|
|
(_("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")]
|
|
|
|
for data in settings_list:
|
|
check_buttons[data[1]]=Gtk.CheckButton(label=data[0])
|
|
check_buttons[data[1]].set_active(self.settings.get_boolean(data[1]))
|
|
check_buttons[data[1]].connect("toggled", self.on_toggled, data[1])
|
|
check_buttons[data[1]].set_margin_start(12)
|
|
|
|
#view grid
|
|
view_grid=Gtk.Grid()
|
|
view_grid.set_row_spacing(6)
|
|
view_grid.set_column_spacing(12)
|
|
view_grid.set_margin_start(12)
|
|
view_grid.add(track_cover_label)
|
|
view_grid.attach_next_to(album_cover_label, track_cover_label, Gtk.PositionType.BOTTOM, 1, 1)
|
|
view_grid.attach_next_to(icon_size_label1, album_cover_label, Gtk.PositionType.BOTTOM, 1, 1)
|
|
view_grid.attach_next_to(combo_settings["playlist-right"][0], icon_size_label1, Gtk.PositionType.BOTTOM, 1, 1)
|
|
view_grid.attach_next_to(track_cover_size, track_cover_label, Gtk.PositionType.RIGHT, 1, 1)
|
|
view_grid.attach_next_to(album_cover_size, album_cover_label, Gtk.PositionType.RIGHT, 1, 1)
|
|
view_grid.attach_next_to(icon_size_combo, icon_size_label1, Gtk.PositionType.RIGHT, 1, 1)
|
|
view_grid.attach_next_to(icon_size_label2, icon_size_combo, Gtk.PositionType.RIGHT, 1, 1)
|
|
view_grid.attach_next_to(combo_settings["playlist-right"][1], combo_settings["playlist-right"][0], Gtk.PositionType.RIGHT, 1, 1)
|
|
|
|
#behavior grid
|
|
behavior_grid=Gtk.Grid()
|
|
behavior_grid.set_row_spacing(6)
|
|
behavior_grid.set_column_spacing(12)
|
|
behavior_grid.set_margin_start(12)
|
|
behavior_grid.add(combo_settings["sort-albums-by-year"][0])
|
|
behavior_grid.attach_next_to(combo_settings["sort-albums-by-year"][1], combo_settings["sort-albums-by-year"][0], Gtk.PositionType.RIGHT, 1, 1)
|
|
|
|
#connect
|
|
track_cover_size.connect("value-changed", self.on_int_changed, "track-cover")
|
|
album_cover_size.connect("value-changed", self.on_int_changed, "album-cover")
|
|
icon_size_combo.connect("changed", self.on_icon_size_changed)
|
|
|
|
#packing
|
|
self.pack_start(view_heading, True, True, 0)
|
|
self.pack_start(check_buttons["show-stop"], True, True, 0)
|
|
self.pack_start(check_buttons["show-initials"], True, True, 0)
|
|
self.pack_start(check_buttons["show-album-view-tooltips"], True, True, 0)
|
|
self.pack_start(view_grid, True, True, 0)
|
|
self.pack_start(behavior_heading, True, True, 0)
|
|
self.pack_start(check_buttons["use-album-artist"], True, True, 0)
|
|
self.pack_start(check_buttons["send-notify"], True, True, 0)
|
|
self.pack_start(check_buttons["stop-on-quit"], True, True, 0)
|
|
self.pack_start(check_buttons["force-mode"], True, True, 0)
|
|
self.pack_start(behavior_grid, True, True, 0)
|
|
|
|
def on_int_changed(self, widget, key):
|
|
self.settings.set_int(key, widget.get_int())
|
|
|
|
def on_icon_size_changed(self, box):
|
|
active_size=int(box.get_active_text())
|
|
self.settings.set_int("icon-size", active_size)
|
|
|
|
def on_combo_changed(self, box, key):
|
|
active=box.get_active()
|
|
if active == 0:
|
|
self.settings.set_boolean(key, False)
|
|
else:
|
|
self.settings.set_boolean(key, True)
|
|
|
|
def on_toggled(self, widget, key):
|
|
self.settings.set_boolean(key, widget.get_active())
|
|
|
|
|
|
class PlaylistSettings(Gtk.Box):
|
|
def __init__(self, settings):
|
|
Gtk.Box.__init__(self, orientation=Gtk.Orientation.VERTICAL, spacing=6)
|
|
self.set_property("border-width", 18)
|
|
|
|
#adding vars
|
|
self.settings = settings
|
|
|
|
#label
|
|
label=Gtk.Label(label=_("Choose the order of information to appear in the playlist:"))
|
|
label.set_line_wrap(True)
|
|
label.set_xalign(0)
|
|
|
|
#Store
|
|
#(toggle, header, index)
|
|
self.store = Gtk.ListStore(bool, str, int)
|
|
|
|
#TreeView
|
|
self.treeview = Gtk.TreeView(model=self.store)
|
|
self.treeview.set_search_column(-1)
|
|
self.treeview.set_reorderable(True)
|
|
self.treeview.set_headers_visible(False)
|
|
|
|
#selection
|
|
self.selection = self.treeview.get_selection()
|
|
|
|
#Column
|
|
renderer_text = Gtk.CellRendererText()
|
|
renderer_toggle = Gtk.CellRendererToggle()
|
|
|
|
column_toggle=Gtk.TreeViewColumn("", renderer_toggle, active=0)
|
|
self.treeview.append_column(column_toggle)
|
|
|
|
column_text=Gtk.TreeViewColumn("", renderer_text, text=1)
|
|
self.treeview.append_column(column_text)
|
|
|
|
#fill store
|
|
self.headers=[_("No"), _("Disc"), _("Title"), _("Artist"), _("Album"), _("Length"), _("Year"), _("Genre")]
|
|
visibilities=self.settings.get_value("column-visibilities").unpack()
|
|
|
|
for index in self.settings.get_value("column-permutation"):
|
|
self.store.append([visibilities[index], self.headers[index], index])
|
|
|
|
#scroll
|
|
scroll=Gtk.ScrolledWindow()
|
|
scroll.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
|
|
scroll.add(self.treeview)
|
|
frame=Gtk.Frame()
|
|
frame.add(scroll)
|
|
|
|
#Toolbar
|
|
toolbar=Gtk.Toolbar()
|
|
style_context=toolbar.get_style_context()
|
|
style_context.add_class("inline-toolbar")
|
|
self.up_button=Gtk.ToolButton.new(Gtk.Image.new_from_icon_name("go-up-symbolic", Gtk.IconSize.SMALL_TOOLBAR))
|
|
self.up_button.set_sensitive(False)
|
|
self.down_button=Gtk.ToolButton.new(Gtk.Image.new_from_icon_name("go-down-symbolic", Gtk.IconSize.SMALL_TOOLBAR))
|
|
self.down_button.set_sensitive(False)
|
|
toolbar.insert(self.up_button, 0)
|
|
toolbar.insert(self.down_button, 1)
|
|
|
|
#column chooser
|
|
column_chooser=Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
|
|
column_chooser.pack_start(frame, True, True, 0)
|
|
column_chooser.pack_start(toolbar, False, False, 0)
|
|
|
|
#connect
|
|
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)
|
|
|
|
#packing
|
|
self.pack_start(label, False, False, 0)
|
|
self.pack_start(column_chooser, True, True, 0)
|
|
|
|
def on_cell_toggled(self, widget, path):
|
|
self.store[path][0] = not self.store[path][0]
|
|
self.settings.array_modify('ab', "column-visibilities", self.store[path][2], self.store[path][0])
|
|
|
|
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))
|
|
|
|
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()
|
|
|
|
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()
|
|
|
|
def set_button_sensitivity(self, *args):
|
|
treeiter=self.selection.get_selected()[1]
|
|
path=self.store.get_path(treeiter)
|
|
if treeiter == None:
|
|
self.up_button.set_sensitive(False)
|
|
self.down_button.set_sensitive(False)
|
|
elif self.store.iter_next(treeiter) == 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)
|
|
|
|
class SettingsDialog(Gtk.Dialog):
|
|
def __init__(self, parent, settings):
|
|
Gtk.Dialog.__init__(self, title=_("Settings"), transient_for=parent)
|
|
self.add_button(Gtk.STOCK_OK, Gtk.ResponseType.OK)
|
|
self.set_default_size(500, 400)
|
|
|
|
#adding vars
|
|
self.settings = settings
|
|
|
|
#widgets
|
|
general=GeneralSettings(self.settings)
|
|
profiles=ProfileSettings(parent, self.settings)
|
|
playlist=PlaylistSettings(self.settings)
|
|
|
|
#packing
|
|
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")))
|
|
self.vbox.pack_start(tabs, True, True, 0) #vbox default widget of dialogs
|
|
self.vbox.set_spacing(6)
|
|
|
|
self.show_all()
|
|
|
|
class ClientControl(Gtk.ButtonBox):
|
|
def __init__(self, client, settings):
|
|
Gtk.ButtonBox.__init__(self, spacing=6)
|
|
self.set_property("layout-style", Gtk.ButtonBoxStyle.EXPAND)
|
|
|
|
#adding vars
|
|
self.client=client
|
|
self.settings=settings
|
|
self.icon_size=self.settings.get_gtk_icon_size("icon-size")
|
|
|
|
#widgets
|
|
self.play_button = Gtk.Button(image=Gtk.Image.new_from_icon_name("media-playback-start-symbolic", self.icon_size))
|
|
self.stop_button = Gtk.Button(image=Gtk.Image.new_from_icon_name("media-playback-stop-symbolic", self.icon_size))
|
|
self.prev_button = Gtk.Button(image=Gtk.Image.new_from_icon_name("media-skip-backward-symbolic", self.icon_size))
|
|
self.next_button = Gtk.Button(image=Gtk.Image.new_from_icon_name("media-skip-forward-symbolic", self.icon_size))
|
|
|
|
#connect
|
|
self.play_button.connect("clicked", self.on_play_clicked)
|
|
self.stop_button.connect("clicked", self.on_stop_clicked)
|
|
self.prev_button.connect("clicked", self.on_prev_clicked)
|
|
self.next_button.connect("clicked", self.on_next_clicked)
|
|
self.settings.connect("changed::show-stop", self.on_settings_changed)
|
|
self.client.emitter.connect("player", self.refresh)
|
|
|
|
#packing
|
|
self.pack_start(self.prev_button, True, True, 0)
|
|
self.pack_start(self.play_button, True, True, 0)
|
|
if self.settings.get_boolean("show-stop"):
|
|
self.pack_start(self.stop_button, True, True, 0)
|
|
self.pack_start(self.next_button, True, True, 0)
|
|
|
|
def refresh(self, *args):
|
|
status=self.client.status()
|
|
if status["state"] == "play":
|
|
self.play_button.set_image(Gtk.Image.new_from_icon_name("media-playback-pause-symbolic", self.icon_size))
|
|
self.prev_button.set_sensitive(True)
|
|
self.next_button.set_sensitive(True)
|
|
elif status["state"] == "pause":
|
|
self.play_button.set_image(Gtk.Image.new_from_icon_name("media-playback-start-symbolic", self.icon_size))
|
|
self.prev_button.set_sensitive(True)
|
|
self.next_button.set_sensitive(True)
|
|
else:
|
|
self.play_button.set_image(Gtk.Image.new_from_icon_name("media-playback-start-symbolic", self.icon_size))
|
|
self.prev_button.set_sensitive(False)
|
|
self.next_button.set_sensitive(False)
|
|
|
|
def on_play_clicked(self, widget):
|
|
if self.client.connected():
|
|
status=self.client.status()
|
|
if status["state"] == "play":
|
|
self.client.pause(1)
|
|
elif status["state"] == "pause":
|
|
self.client.pause(0)
|
|
else:
|
|
try:
|
|
self.client.play(status["song"])
|
|
except:
|
|
try:
|
|
self.client.play()
|
|
except:
|
|
pass
|
|
|
|
def on_stop_clicked(self, widget):
|
|
if self.client.connected():
|
|
self.client.stop()
|
|
|
|
def on_prev_clicked(self, widget):
|
|
if self.client.connected():
|
|
self.client.previous()
|
|
|
|
def on_next_clicked(self, widget):
|
|
if self.client.connected():
|
|
self.client.next()
|
|
|
|
def on_settings_changed(self, *args):
|
|
if self.settings.get_boolean("show-stop"):
|
|
self.pack_start(self.stop_button, True, True, 0)
|
|
self.reorder_child(self.stop_button, 2)
|
|
self.stop_button.show()
|
|
else:
|
|
self.remove(self.stop_button)
|
|
|
|
class SeekBar(Gtk.Box):
|
|
def __init__(self, client):
|
|
Gtk.Box.__init__(self)
|
|
self.set_hexpand(True)
|
|
|
|
#adding vars
|
|
self.client=client
|
|
self.seek_time="10" #seek increment in seconds
|
|
self.update=True
|
|
self.jumped=False
|
|
|
|
#labels
|
|
self.elapsed=Gtk.Label()
|
|
self.elapsed.set_width_chars(7)
|
|
self.rest=Gtk.Label()
|
|
self.rest.set_width_chars(8)
|
|
|
|
#progress bar
|
|
self.scale=Gtk.Scale.new_with_range(orientation=Gtk.Orientation.HORIZONTAL, min=0, max=100, step=0.001)
|
|
self.scale.set_show_fill_level(True)
|
|
self.scale.set_restrict_to_fill_level(False)
|
|
self.scale.set_draw_value(False)
|
|
|
|
#css (scale)
|
|
style_context=self.scale.get_style_context()
|
|
provider = Gtk.CssProvider()
|
|
css = b"""scale fill { background-color: @theme_selected_bg_color; }"""
|
|
provider.load_from_data(css)
|
|
style_context.add_provider(provider, 800)
|
|
|
|
#event boxes
|
|
self.elapsed_event_box=Gtk.EventBox()
|
|
self.rest_event_box=Gtk.EventBox()
|
|
|
|
#connect
|
|
self.elapsed_event_box.connect("button-press-event", self.on_elapsed_button_press_event)
|
|
self.rest_event_box.connect("button-press-event", self.on_rest_button_press_event)
|
|
self.scale.connect("change-value", self.on_change_value)
|
|
self.scale.connect("scroll-event", self.dummy) #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)
|
|
self.client.emitter.connect("disconnected", self.on_disconnected)
|
|
self.client.emitter.connect("reconnected", self.on_reconnected)
|
|
self.client.emitter.connect("player", self.on_player)
|
|
|
|
#timeouts
|
|
self.timeout_id=None
|
|
|
|
#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)
|
|
self.pack_start(self.scale, True, True, 0)
|
|
self.pack_end(self.rest_event_box, False, False, 0)
|
|
|
|
def dummy(self, *args):
|
|
return True
|
|
|
|
def on_scale_button_press_event(self, widget, event):
|
|
if event.button == 1 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
self.update=False
|
|
self.scale.set_has_origin(False)
|
|
if event.button == 3 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
self.jumped=False
|
|
|
|
def on_scale_button_release_event(self, widget, event):
|
|
if event.button == 1:
|
|
if self.jumped: #actual seek
|
|
status=self.client.status()
|
|
duration=float(status["duration"])
|
|
factor=(self.scale.get_value()/100)
|
|
pos=(duration*factor)
|
|
self.client.seekcur(pos)
|
|
self.jumped=False
|
|
self.scale.set_has_origin(True)
|
|
self.update=True
|
|
if self.timeout_id == None:
|
|
self.refresh()
|
|
|
|
def on_change_value(self, range, scroll, value): #value is inaccurate
|
|
if scroll == Gtk.ScrollType.STEP_BACKWARD:
|
|
self.seek_backward()
|
|
elif scroll == Gtk.ScrollType.STEP_FORWARD:
|
|
self.seek_forward()
|
|
elif scroll == Gtk.ScrollType.JUMP:
|
|
status=self.client.status()
|
|
duration=float(status["duration"])
|
|
factor=(value/100)
|
|
if factor > 1: #fix display error
|
|
factor=1
|
|
elapsed=(factor*duration)
|
|
self.elapsed.set_text(str(datetime.timedelta(seconds=int(elapsed))))
|
|
self.rest.set_text("-"+str(datetime.timedelta(seconds=int(duration-elapsed))))
|
|
self.jumped=True
|
|
|
|
def seek_forward(self):
|
|
self.client.seekcur("+"+self.seek_time)
|
|
|
|
def seek_backward(self):
|
|
self.client.seekcur("-"+self.seek_time)
|
|
|
|
def enable(self):
|
|
self.scale.set_sensitive(True)
|
|
self.scale.set_range(0, 100)
|
|
self.elapsed_event_box.set_sensitive(True)
|
|
self.rest_event_box.set_sensitive(True)
|
|
|
|
def disable(self):
|
|
self.scale.set_sensitive(False)
|
|
self.scale.set_value(0.0)
|
|
self.scale.set_range(0, 0)
|
|
self.elapsed_event_box.set_sensitive(False)
|
|
self.rest_event_box.set_sensitive(False)
|
|
self.elapsed.set_text("0:00:00")
|
|
self.rest.set_text("-0:00:00")
|
|
|
|
def on_elapsed_button_press_event(self, widget, event):
|
|
if event.button == 1 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
self.seek_backward()
|
|
elif event.button == 3 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
self.seek_forward()
|
|
|
|
def on_rest_button_press_event(self, widget, event):
|
|
if event.button == 1 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
self.seek_forward()
|
|
elif event.button == 3 and event.type == Gdk.EventType.BUTTON_PRESS:
|
|
self.seek_backward()
|
|
|
|
def on_reconnected(self, *args):
|
|
self.timeout_id=GLib.timeout_add(100, self.refresh)
|
|
self.enable()
|
|
|
|
def on_disconnected(self, *args):
|
|
if not self.timeout_id == None:
|
|
GLib.source_remove(self.timeout_id)
|
|
self.timeout_id=None
|
|
self.disable()
|
|
|
|
def on_player(self, *args):
|
|
status=self.client.status()
|
|
if status['state'] == "stop":
|
|
if not self.timeout_id == None:
|
|
GLib.source_remove(self.timeout_id)
|
|
self.timeout_id=None
|
|
self.disable()
|
|
elif status['state'] == "pause":
|
|
if not self.timeout_id == None:
|
|
GLib.source_remove(self.timeout_id)
|
|
self.timeout_id=None
|
|
self.refresh()
|
|
else:
|
|
if self.timeout_id == None:
|
|
self.timeout_id=GLib.timeout_add(100, self.refresh)
|
|
self.enable()
|
|
|
|
def refresh(self):
|
|
try:
|
|
status=self.client.status()
|
|
duration=float(status["duration"])
|
|
elapsed=float(status["elapsed"])
|
|
if elapsed > duration: #fix display error
|
|
elapsed=duration
|
|
fraction=(elapsed/duration)*100
|
|
if self.update:
|
|
self.scale.set_value(fraction)
|
|
self.elapsed.set_text(str(datetime.timedelta(seconds=int(elapsed))))
|
|
self.rest.set_text("-"+str(datetime.timedelta(seconds=int(duration-elapsed))))
|
|
self.scale.set_fill_level(fraction)
|
|
except:
|
|
self.disable()
|
|
return True
|
|
|
|
class PlaybackOptions(Gtk.Box):
|
|
def __init__(self, client, settings):
|
|
Gtk.Box.__init__(self, spacing=6)
|
|
|
|
#adding vars
|
|
self.client=client
|
|
self.settings=settings
|
|
self.icon_size=self.settings.get_gtk_icon_size("icon-size")
|
|
|
|
#widgets
|
|
self.random=Gtk.ToggleButton(image=Gtk.Image.new_from_icon_name("media-playlist-shuffle-symbolic", self.icon_size))
|
|
self.random.set_tooltip_text(_("Random mode"))
|
|
self.repeat=Gtk.ToggleButton(image=Gtk.Image.new_from_icon_name("media-playlist-repeat-symbolic", self.icon_size))
|
|
self.repeat.set_tooltip_text(_("Repeat mode"))
|
|
self.single=Gtk.ToggleButton(image=Gtk.Image.new_from_icon_name("zoom-original-symbolic", self.icon_size))
|
|
self.single.set_tooltip_text(_("Single mode"))
|
|
self.consume=Gtk.ToggleButton(image=Gtk.Image.new_from_icon_name("edit-cut-symbolic", self.icon_size))
|
|
self.consume.set_tooltip_text(_("Consume mode"))
|
|
self.volume=Gtk.VolumeButton()
|
|
self.volume.set_property("size", self.icon_size)
|
|
|
|
#connect
|
|
self.random_toggled=self.random.connect("toggled", self.set_random)
|
|
self.repeat_toggled=self.repeat.connect("toggled", self.set_repeat)
|
|
self.single_toggled=self.single.connect("toggled", self.set_single)
|
|
self.consume_toggled=self.consume.connect("toggled", self.set_consume)
|
|
self.volume_changed=self.volume.connect("value-changed", self.set_volume)
|
|
self.options_changed=self.client.emitter.connect("options", self.options_refresh)
|
|
self.mixer_changed=self.client.emitter.connect("mixer", self.mixer_refresh)
|
|
|
|
#packing
|
|
ButtonBox=Gtk.ButtonBox()
|
|
ButtonBox.set_property("layout-style", Gtk.ButtonBoxStyle.EXPAND)
|
|
ButtonBox.pack_start(self.repeat, True, True, 0)
|
|
ButtonBox.pack_start(self.random, True, True, 0)
|
|
ButtonBox.pack_start(self.single, True, True, 0)
|
|
ButtonBox.pack_start(self.consume, True, True, 0)
|
|
self.pack_start(ButtonBox, True, True, 0)
|
|
self.pack_start(self.volume, True, True, 0)
|
|
|
|
def set_random(self, widget):
|
|
if widget.get_active():
|
|
self.client.random("1")
|
|
else:
|
|
self.client.random("0")
|
|
|
|
def set_repeat(self, widget):
|
|
if widget.get_active():
|
|
self.client.repeat("1")
|
|
else:
|
|
self.client.repeat("0")
|
|
|
|
def set_single(self, widget):
|
|
if widget.get_active():
|
|
self.client.single("1")
|
|
else:
|
|
self.client.single("0")
|
|
|
|
def set_consume(self, widget):
|
|
if widget.get_active():
|
|
self.client.consume("1")
|
|
else:
|
|
self.client.consume("0")
|
|
|
|
def set_volume(self, widget, value):
|
|
self.client.setvol(str(int(value*100)))
|
|
|
|
def options_refresh(self, *args):
|
|
self.repeat.handler_block(self.repeat_toggled)
|
|
self.random.handler_block(self.random_toggled)
|
|
self.single.handler_block(self.single_toggled)
|
|
self.consume.handler_block(self.consume_toggled)
|
|
status=self.client.status()
|
|
if status["repeat"] == "0":
|
|
self.repeat.set_active(False)
|
|
else:
|
|
self.repeat.set_active(True)
|
|
if status["random"] == "0":
|
|
self.random.set_active(False)
|
|
else:
|
|
self.random.set_active(True)
|
|
if status["single"] == "0":
|
|
self.single.set_active(False)
|
|
else:
|
|
self.single.set_active(True)
|
|
if status["consume"] == "0":
|
|
self.consume.set_active(False)
|
|
else:
|
|
self.consume.set_active(True)
|
|
self.repeat.handler_unblock(self.repeat_toggled)
|
|
self.random.handler_unblock(self.random_toggled)
|
|
self.single.handler_unblock(self.single_toggled)
|
|
self.consume.handler_unblock(self.consume_toggled)
|
|
|
|
def mixer_refresh(self, *args):
|
|
self.volume.handler_block(self.volume_changed)
|
|
status=self.client.status()
|
|
try:
|
|
self.volume.set_value((int(status["volume"])/100))
|
|
except:
|
|
self.volume.set_value(0)
|
|
self.volume.handler_unblock(self.volume_changed)
|
|
|
|
class AudioType(Gtk.Button):
|
|
def __init__(self, client):
|
|
Gtk.Button.__init__(self)
|
|
self.set_relief(Gtk.ReliefStyle.NONE)
|
|
self.set_tooltip_text(_("Show additional information"))
|
|
|
|
#adding vars
|
|
self.client=client
|
|
|
|
#widgets
|
|
self.label=Gtk.Label()
|
|
self.label.set_xalign(1)
|
|
self.label.set_ellipsize(Pango.EllipsizeMode.END)
|
|
self.popover=Gtk.Popover()
|
|
self.popover.set_relative_to(self)
|
|
|
|
#Store
|
|
#(tag, value)
|
|
self.store = Gtk.ListStore(str, str)
|
|
|
|
#TreeView
|
|
self.treeview = Gtk.TreeView(model=self.store)
|
|
self.treeview.set_can_focus(False)
|
|
self.treeview.set_search_column(-1)
|
|
sel = self.treeview.get_selection()
|
|
sel.set_mode(Gtk.SelectionMode.NONE)
|
|
|
|
#Column
|
|
renderer_text = Gtk.CellRendererText()
|
|
|
|
self.column_tag = Gtk.TreeViewColumn(_("MPD-Tag"), renderer_text, text=0)
|
|
self.column_tag.set_property("resizable", False)
|
|
self.treeview.append_column(self.column_tag)
|
|
|
|
self.column_value = Gtk.TreeViewColumn(_("Value"), renderer_text, text=1)
|
|
self.column_value.set_property("resizable", False)
|
|
self.treeview.append_column(self.column_value)
|
|
|
|
#timeouts
|
|
GLib.timeout_add(1000, self.refresh)
|
|
|
|
#connect
|
|
self.connect("clicked", self.on_clicked)
|
|
|
|
#packing
|
|
self.popover.add(self.treeview)
|
|
self.add(self.label)
|
|
|
|
def refresh(self):
|
|
if self.client.connected():
|
|
status=self.client.status()
|
|
try:
|
|
file_type=self.client.playlistinfo(status["song"])[0]["file"].split('.')[-1]
|
|
freq, res, chan = status["audio"].split(':')
|
|
freq=str(float(freq)/1000)
|
|
brate = status["bitrate"]
|
|
string=_("%(bitrate)s kb/s, %(frequency)s kHz, %(resolution)s bit, %(channels)s channels, %(file_type)s") % {"bitrate": brate, "frequency": freq, "resolution": res, "channels": chan, "file_type": file_type}
|
|
self.label.set_text(string)
|
|
except:
|
|
self.label.set_text("-")
|
|
else:
|
|
self.label.set_text("-")
|
|
return True
|
|
|
|
def on_clicked(self, *args):
|
|
try:
|
|
self.store.clear()
|
|
song=self.client.status()["song"]
|
|
tags=self.client.playlistinfo(song)[0]
|
|
for key in tags:
|
|
if key == "time":
|
|
self.store.append([key, str(datetime.timedelta(seconds=int(tags[key])))])
|
|
else:
|
|
self.store.append([key, tags[key]])
|
|
self.popover.show_all()
|
|
self.treeview.queue_resize()
|
|
except:
|
|
pass
|
|
|
|
class ProfileSelect(Gtk.ComboBoxText):
|
|
def __init__(self, client, settings):
|
|
Gtk.ComboBoxText.__init__(self)
|
|
|
|
#adding vars
|
|
self.client=client
|
|
self.settings=settings
|
|
|
|
#connect
|
|
self.changed=self.connect("changed", self.on_changed)
|
|
self.settings.connect("changed::profiles", self.refresh)
|
|
self.settings.connect("changed::hosts", self.refresh)
|
|
self.settings.connect("changed::ports", self.refresh)
|
|
self.settings.connect("changed::passwords", self.refresh)
|
|
self.settings.connect("changed::paths", self.refresh)
|
|
|
|
self.refresh()
|
|
|
|
def refresh(self, *args):
|
|
self.handler_block(self.changed)
|
|
self.remove_all()
|
|
for profile in self.settings.get_value("profiles"):
|
|
self.append_text(profile)
|
|
self.set_active(self.settings.get_int("active-profile"))
|
|
self.handler_unblock(self.changed)
|
|
|
|
def on_changed(self, *args):
|
|
active=self.get_active()
|
|
self.settings.set_int("active-profile", active)
|
|
|
|
class ServerStats(Gtk.Dialog):
|
|
def __init__(self, parent, client):
|
|
Gtk.Dialog.__init__(self, title=_("Stats"), transient_for=parent)
|
|
|
|
#adding vars
|
|
self.client=client
|
|
|
|
#Store
|
|
#(tag, value)
|
|
self.store = Gtk.ListStore(str, str)
|
|
|
|
#TreeView
|
|
self.treeview = Gtk.TreeView(model=self.store)
|
|
self.treeview.set_can_focus(False)
|
|
self.treeview.set_search_column(-1)
|
|
sel = self.treeview.get_selection()
|
|
sel.set_mode(Gtk.SelectionMode.NONE)
|
|
|
|
#Column
|
|
renderer_text = Gtk.CellRendererText()
|
|
|
|
self.column_tag = Gtk.TreeViewColumn(_("Tag"), renderer_text, text=0)
|
|
self.treeview.append_column(self.column_tag)
|
|
|
|
self.column_value = Gtk.TreeViewColumn(_("Value"), renderer_text, text=1)
|
|
self.treeview.append_column(self.column_value)
|
|
|
|
stats=self.client.stats()
|
|
for key in stats:
|
|
if key == "uptime" or key == "playtime" or key == "db_playtime":
|
|
self.store.append([key, str(datetime.timedelta(seconds=int(stats[key])))])
|
|
elif key == "db_update":
|
|
self.store.append([key, str(datetime.datetime.fromtimestamp(int(stats[key])))])
|
|
else:
|
|
self.store.append([key, stats[key]])
|
|
|
|
self.vbox.pack_start(self.treeview, True, True, 0)
|
|
self.show_all()
|
|
|
|
class SearchWindow(Gtk.Window):
|
|
def __init__(self, client):
|
|
Gtk.Window.__init__(self, title=_("Search"))
|
|
self.set_icon_name("mpdevil")
|
|
self.set_default_size(800, 600)
|
|
|
|
#adding vars
|
|
self.client=client
|
|
|
|
#search entry
|
|
self.search_entry=Gtk.SearchEntry()
|
|
self.search_entry.set_margin_end(6)
|
|
self.search_entry.set_margin_start(6)
|
|
|
|
#label
|
|
self.label=Gtk.Label()
|
|
self.label.set_xalign(1)
|
|
self.label.set_margin_end(6)
|
|
|
|
#songs view
|
|
self.songs_view=SongsView(self.client)
|
|
|
|
#connect
|
|
self.search_entry.connect("search-changed", self.on_search_changed)
|
|
|
|
#packing
|
|
vbox=Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
|
|
vbox.pack_start(self.search_entry, False, False, 6)
|
|
vbox.pack_start(Gtk.Separator.new(orientation=Gtk.Orientation.HORIZONTAL), False, False, 0)
|
|
vbox.pack_start(self.songs_view, True, True, 0)
|
|
vbox.pack_start(Gtk.Separator.new(orientation=Gtk.Orientation.HORIZONTAL), False, False, 0)
|
|
vbox.pack_start(self.label, False, False, 6)
|
|
self.add(vbox)
|
|
|
|
self.show_all()
|
|
|
|
def on_search_changed(self, widget):
|
|
self.songs_view.clear()
|
|
self.songs_view.populate(self.client.search("any", self.search_entry.get_text()))
|
|
self.label.set_text(_("hits: %i") % (self.songs_view.count()))
|
|
|
|
class LyricsWindow(Gtk.Window):
|
|
def __init__(self, client, settings):
|
|
Gtk.Window.__init__(self, title=_("Lyrics"))
|
|
self.set_icon_name("mpdevil")
|
|
self.set_default_size(450, 800)
|
|
|
|
#adding vars
|
|
self.settings=settings
|
|
self.client=client
|
|
|
|
#widgets
|
|
self.scroll=Gtk.ScrolledWindow()
|
|
self.scroll.set_policy(Gtk.PolicyType.AUTOMATIC, Gtk.PolicyType.AUTOMATIC)
|
|
self.label=Gtk.Label()
|
|
self.label.set_selectable(True)
|
|
self.label.set_yalign(0)
|
|
self.label.set_xalign(0)
|
|
|
|
#connect
|
|
self.file_changed=self.client.emitter.connect("playing_file_changed", self.refresh)
|
|
self.connect("destroy", self.remove_handlers)
|
|
|
|
#packing
|
|
self.scroll.add(self.label)
|
|
self.add(self.scroll)
|
|
|
|
self.show_all()
|
|
|
|
self.refresh()
|
|
|
|
def remove_handlers(self, *args):
|
|
self.client.emitter.disconnect(self.file_changed)
|
|
|
|
def display_lyrics(self, current_song):
|
|
GLib.idle_add(self.label.set_text, _("searching..."))
|
|
try:
|
|
text=self.getLyrics(current_song["artist"], current_song["title"])
|
|
except:
|
|
text=_("not found")
|
|
GLib.idle_add(self.label.set_text, text)
|
|
|
|
def refresh(self, *args):
|
|
update_thread=threading.Thread(target=self.display_lyrics, kwargs={"current_song": self.client.currentsong()}, daemon=True)
|
|
update_thread.start()
|
|
|
|
def getLyrics(self, singer, song): #partially copied from PyLyrics 1.1.0
|
|
#Replace spaces with _
|
|
singer = singer.replace(' ', '_')
|
|
song = song.replace(' ', '_')
|
|
r = requests.get('http://lyrics.wikia.com/{0}:{1}'.format(singer,song))
|
|
s = BeautifulSoup(r.text)
|
|
#Get main lyrics holder
|
|
lyrics = s.find("div",{'class':'lyricbox'})
|
|
if lyrics is None:
|
|
raise ValueError("Song or Singer does not exist or the API does not have Lyrics")
|
|
return None
|
|
#Remove Scripts
|
|
[s.extract() for s in lyrics('script')]
|
|
#Remove Comments
|
|
comments = lyrics.findAll(text=lambda text:isinstance(text, Comment))
|
|
[comment.extract() for comment in comments]
|
|
#Remove span tag (Needed for instrumantal)
|
|
if not lyrics.span == None:
|
|
lyrics.span.extract()
|
|
#Remove unecessary tags
|
|
for tag in ['div','i','b','a']:
|
|
for match in lyrics.findAll(tag):
|
|
match.replaceWithChildren()
|
|
#Get output as a string and remove non unicode characters and replace <br> with newlines
|
|
output = str(lyrics).encode('utf-8', errors='replace')[22:-6:].decode("utf-8").replace('\n','').replace('<br/>','\n')
|
|
try:
|
|
return output
|
|
except:
|
|
return output.encode('utf-8')
|
|
|
|
class MainWindow(Gtk.ApplicationWindow):
|
|
def __init__(self, app, client, settings):
|
|
Gtk.ApplicationWindow.__init__(self, title=("mpdevil"), application=app)
|
|
Notify.init("mpdevil")
|
|
self.set_icon_name("mpdevil")
|
|
self.settings = settings
|
|
self.set_default_size(self.settings.get_int("width"), self.settings.get_int("height"))
|
|
|
|
#adding vars
|
|
self.app=app
|
|
self.client=client
|
|
self.icon_size=self.settings.get_gtk_icon_size("icon-size")
|
|
|
|
#MPRIS
|
|
DBusGMainLoop(set_as_default=True)
|
|
self.dbus_service = MPRISInterface(self, self.client, self.settings)
|
|
|
|
#actions
|
|
save_action = Gio.SimpleAction.new("save", None)
|
|
save_action.connect("activate", self.on_save)
|
|
self.add_action(save_action)
|
|
|
|
settings_action = Gio.SimpleAction.new("settings", None)
|
|
settings_action.connect("activate", self.on_settings)
|
|
self.add_action(settings_action)
|
|
|
|
stats_action = Gio.SimpleAction.new("stats", None)
|
|
stats_action.connect("activate", self.on_stats)
|
|
self.add_action(stats_action)
|
|
|
|
update_action = Gio.SimpleAction.new("update", None)
|
|
update_action.connect("activate", self.on_update)
|
|
self.add_action(update_action)
|
|
|
|
#widgets
|
|
self.browser=Browser(self.client, self.settings, self)
|
|
self.profiles=ProfileSelect(self.client, self.settings)
|
|
self.profiles.set_tooltip_text(_("Select profile"))
|
|
self.control=ClientControl(self.client, self.settings)
|
|
self.progress=SeekBar(self.client)
|
|
self.lyrics_button=Gtk.ToggleButton(image=Gtk.Image.new_from_icon_name("media-view-subtitles-symbolic", self.icon_size))
|
|
self.lyrics_button.set_tooltip_text(_("Show lyrics"))
|
|
self.play_opts=PlaybackOptions(self.client, self.settings)
|
|
|
|
#menu
|
|
menu = Gio.Menu()
|
|
menu.append(_("Save window layout"), "win.save")
|
|
menu.append(_("Settings"), "win.settings")
|
|
menu.append(_("Update database"), "win.update")
|
|
menu.append(_("Server stats"), "win.stats")
|
|
menu.append(_("About"), "app.about")
|
|
menu.append(_("Quit"), "app.quit")
|
|
|
|
menu_button = Gtk.MenuButton.new()
|
|
menu_popover = Gtk.Popover.new_from_model(menu_button, menu)
|
|
menu_button.set_popover(menu_popover)
|
|
menu_button.set_tooltip_text(_("Menu"))
|
|
menu_button.set_image(image=Gtk.Image.new_from_icon_name("open-menu-symbolic", self.icon_size))
|
|
|
|
#connect
|
|
self.lyrics_button.connect("toggled", self.on_lyrics_toggled)
|
|
self.settings.connect("changed::profiles", self.on_settings_changed)
|
|
self.client.emitter.connect("playing_file_changed", self.on_file_changed)
|
|
self.client.emitter.connect("disconnected", self.on_disconnected)
|
|
self.client.emitter.connect("reconnected", self.on_reconnected)
|
|
#unmap space
|
|
binding_set=Gtk.binding_set_find('GtkTreeView')
|
|
Gtk.binding_entry_remove(binding_set, 32, Gdk.ModifierType.MOD2_MASK)
|
|
#map space play/pause
|
|
self.connect("key-press-event", self.on_key_press_event)
|
|
|
|
#packing
|
|
self.vbox=Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
|
|
self.action_bar=Gtk.ActionBar()
|
|
self.vbox.pack_start(self.browser, True, True, 0)
|
|
self.vbox.pack_start(self.action_bar, False, False, 0)
|
|
self.action_bar.pack_start(self.control)
|
|
self.action_bar.pack_start(self.progress)
|
|
self.action_bar.pack_start(self.lyrics_button)
|
|
self.action_bar.pack_start(self.profiles)
|
|
self.action_bar.pack_start(self.play_opts)
|
|
self.action_bar.pack_end(menu_button)
|
|
|
|
self.add(self.vbox)
|
|
|
|
self.show_all()
|
|
self.on_settings_changed() #hide profiles button
|
|
|
|
def on_file_changed(self, *args):
|
|
try:
|
|
song=self.client.currentsong()
|
|
self.set_title(song["artist"]+" - "+song["title"]+" - "+song["album"])
|
|
if self.settings.get_boolean("send-notify"):
|
|
if not self.is_active() and self.client.status()["state"] == "play":
|
|
notify=Notify.Notification.new(song["title"], song["artist"]+"\n"+song["album"])
|
|
pixbuf=Cover(lib_path=self.settings.get_value("paths")[self.settings.get_int("active-profile")], song_file=song["file"]).get_pixbuf(400)
|
|
notify.set_image_from_pixbuf(pixbuf)
|
|
notify.show()
|
|
except:
|
|
self.set_title("mpdevil")
|
|
|
|
def on_reconnected(self, *args):
|
|
self.dbus_service.acquire_name()
|
|
self.progress.set_sensitive(True)
|
|
self.control.set_sensitive(True)
|
|
self.play_opts.set_sensitive(True)
|
|
self.lyrics_button.set_sensitive(True)
|
|
self.browser.back_to_album()
|
|
|
|
def on_disconnected(self, *args):
|
|
self.dbus_service.release_name()
|
|
self.lyrics_button.set_active(False)
|
|
self.set_title("mpdevil (not connected)")
|
|
self.songid_playing=None
|
|
self.progress.set_sensitive(False)
|
|
self.control.set_sensitive(False)
|
|
self.play_opts.set_sensitive(False)
|
|
self.lyrics_button.set_sensitive(False)
|
|
|
|
def on_lyrics_toggled(self, widget):
|
|
if widget.get_active():
|
|
if self.client.connected():
|
|
def set_active(*args):
|
|
self.lyrics_button.set_active(False)
|
|
self.lyrics_win = LyricsWindow(self.client, self.settings)
|
|
self.lyrics_win.connect("destroy", set_active)
|
|
else:
|
|
self.lyrics_win.destroy()
|
|
|
|
def on_key_press_event(self, widget, event):
|
|
if event.keyval == 32: #space
|
|
self.control.play_button.grab_focus()
|
|
if event.keyval == 269025044: #AudioPlay
|
|
self.control.play_button.grab_focus()
|
|
self.control.play_button.emit("clicked")
|
|
elif event.keyval == 269025047 or event.keyval == 43 or event.keyval == 65451: #AudioNext
|
|
self.control.next_button.grab_focus()
|
|
self.control.next_button.emit("clicked")
|
|
elif event.keyval == 269025046 or event.keyval == 45 or event.keyval == 65453: #AudioPrev
|
|
self.control.prev_button.grab_focus()
|
|
self.control.prev_button.emit("clicked")
|
|
elif event.keyval == 65307: #esc
|
|
self.browser.back_to_album()
|
|
elif event.keyval == 65450: #*
|
|
self.progress.scale.grab_focus()
|
|
self.progress.seek_forward()
|
|
elif event.keyval == 65455: #/
|
|
self.progress.scale.grab_focus()
|
|
self.progress.seek_backward()
|
|
|
|
def on_save(self, action, param):
|
|
size=self.get_size()
|
|
self.settings.set_int("width", size[0])
|
|
self.settings.set_int("height", size[1])
|
|
self.browser.save_settings()
|
|
|
|
def on_settings(self, action, param):
|
|
settings = SettingsDialog(self, self.settings)
|
|
settings.run()
|
|
settings.destroy()
|
|
|
|
def on_stats(self, action, param):
|
|
if self.client.connected():
|
|
stats = ServerStats(self, self.client)
|
|
stats.run()
|
|
stats.destroy()
|
|
|
|
def on_update(self, action, param):
|
|
if self.client.connected():
|
|
self.client.update()
|
|
|
|
def on_settings_changed(self, *args):
|
|
if len(self.settings.get_value("profiles")) > 1:
|
|
self.profiles.set_property("visible", True)
|
|
else:
|
|
self.profiles.set_property("visible", False)
|
|
|
|
class mpdevil(Gtk.Application):
|
|
def __init__(self, *args, **kwargs):
|
|
super().__init__(*args, application_id="org.mpdevil", flags=Gio.ApplicationFlags.FLAGS_NONE, **kwargs)
|
|
self.settings = Settings()
|
|
self.client=Client(self.settings)
|
|
self.window=None
|
|
|
|
def do_activate(self):
|
|
if not self.window: #allow just one instance
|
|
self.window = MainWindow(self, self.client, self.settings)
|
|
self.window.connect("delete-event", self.on_delete_event)
|
|
self.window.present()
|
|
|
|
def do_startup(self):
|
|
Gtk.Application.do_startup(self)
|
|
|
|
action = Gio.SimpleAction.new("about", None)
|
|
action.connect("activate", self.on_about)
|
|
self.add_action(action)
|
|
|
|
action = Gio.SimpleAction.new("quit", None)
|
|
action.connect("activate", self.on_quit)
|
|
self.add_action(action)
|
|
|
|
def on_delete_event(self, *args):
|
|
if self.settings.get_boolean("stop-on-quit") and self.client.connected():
|
|
self.client.stop()
|
|
self.quit()
|
|
|
|
def on_about(self, action, param):
|
|
dialog=Gtk.AboutDialog(transient_for=self.window, modal=True)
|
|
dialog.set_program_name(NAME)
|
|
dialog.set_version(VERSION)
|
|
dialog.set_comments(_("A small MPD client written in python"))
|
|
dialog.set_authors(["Martin Wagner"])
|
|
dialog.set_website("https://github.com/SoongNoonien/mpdevil")
|
|
dialog.set_copyright("\xa9 2020 Martin Wagner")
|
|
dialog.set_logo_icon_name(PACKAGE)
|
|
dialog.run()
|
|
dialog.destroy()
|
|
|
|
def on_quit(self, action, param):
|
|
if self.settings.get_boolean("stop-on-quit") and self.client.connected():
|
|
self.client.stop()
|
|
self.quit()
|
|
|
|
if __name__ == '__main__':
|
|
app = mpdevil()
|
|
app.run(sys.argv)
|
|
|