Grob umstrukturiert und an Python-Styleguide angepasst

This commit is contained in:
gotik 2012-04-19 18:28:17 +02:00
parent 70949ae1cd
commit 8a4bb3e41e
2 changed files with 291 additions and 140 deletions

View file

@ -3,7 +3,7 @@
from mpd import MPDClient import mpd
import os import os
from threading import Thread from threading import Thread
@ -11,120 +11,205 @@ from threading import Thread
class MPDCoverGrid: class MPDCoverGrid:
client = MPDClient() SIGNAL_CONNECT = 'connect'
albums = {} SIGNAL_IDLE = 'idle'
SIGNAL_IDLE_PLAYER = 'idlePlayer'
SIGNAL_UPDATE = 'update'
def __init__(self, host='localhost', port='6600', password=None): def __init__(self, host="localhost", port=6600, password=None):
self._host = host self._host = host
self._port = port self._port = port
self._password = password self._password = password
self.updateCallback = None self._connected = False
self._callbacks = {}
self._threads = {}
self._client = mpd.MPDClient()
self._albums = {}
def connect(self): def connect(self):
try: self._start_thread(self.SIGNAL_CONNECT, self._connect)
self.client.connect(self._host, self._port)
if self._password:
self.client.password(self._password)
except IOError as e:
# TODO Error
print(e)
def disconnect(self): def disconnect(self):
self._disconnect()
def is_connected(self):
return self._connected
def _connect(self):
try: try:
self.client.disconnect() self._client.connect(self._host, self._port)
if self._password:
self._client.password(self._password)
# TODO Verbindung testen
self._connected = True
self._callback(self.SIGNAL_CONNECT, self._connected, None)
#self._start_idle()
self.update()
except IOError as e: except IOError as e:
# TODO Error self._connected = False
print(e) self._callback(self.SIGNAL_CONNECT, self._connected, e)
self.client = MPDClient()
def getAlbums(self): def _disconnect(self):
self.update() if not self.is_connected():
return self.albums return
self._stop_idle()
try:
#self._client.close()
self._client.disconnect()
except:
self._client = mpd.MPDClient()
self._connected = False
self._callback(self.SIGNAL_CONNECT, self._connected, None)
def connectUpdate(self, updateCallback): def _start_idle(self):
self.updateCallback = updateCallback self._start_thread(self.SIGNAL_IDLE, self._idle)
def _stop_idle(self):
if not self._is_doing(self.SIGNAL_IDLE):
return
try:
del self._threads[self.SIGNAL_IDLE]
self._client.noidle()
except TypeError as e:
pass
def _idle(self):
while not self._is_doing(self.SIGNAL_IDLE):
pass
while self._client is not None and self._connected and self._is_doing(self.SIGNAL_IDLE):
self._client.send_idle()
if self._is_doing(self.SIGNAL_IDLE):
modules = self._client.fetch_idle()
if 'player' in modules:
self._idlePlayer()
if 'database' in modules:
# TODO update DB
# self.update()?
pass
if 'update' in modules:
# TODO update
#self._idleUpdate()
pass
if 'mixer' in modules:
pass
def _idlePlayer(self):
if not self._has_callback(self.SIGNAL_IDLE_PLAYER):
return
status = self._client.status()
state = status['state']
song = self._client.currentsong()
album = MCGAlbum(song['artist'], song['album'], os.path.dirname(song['file']))
self._callback(self.SIGNAL_IDLE_PLAYER, state, album)
def update(self): def update(self):
Thread(target=self._update, args=()).start() self._start_thread(self.SIGNAL_UPDATE, self._update)
def _update(self): def _update(self):
for song in self.client.listallinfo(): self._stop_idle()
for song in self._client.listallinfo():
try: try:
new = False if song['album'] not in self._albums:
if song['album'] not in self.albums: album = MCGAlbum(song['artist'], song['album'], os.path.dirname(song['file']))
self.albums[song['album']] = MCGAlbum(song['artist'], song['album'], os.path.dirname(song['file'])) self._albums[song['album']] = album
new = True self._callback(self.SIGNAL_UPDATE, album)
album = self.albums[song['album']]
album.addTrack(song['title'])
if new and self.updateCallback is not None:
self.updateCallback(album)
except KeyError: except KeyError:
pass pass
self._start_idle()
class MCGAlbum():
fileNames = ['folder', 'cover'] def connect_signal(self, signal, callback):
fileExts = ['jpg', 'jpeg', 'png'] self._callbacks[signal] = callback
def _has_callback(self, signal):
return signal in self._callbacks
def _callback(self, signal, *args):
if self._has_callback(signal):
callback = self._callbacks[signal]
callback(*args)
def _start_thread(self, signal, method):
self._threads[signal] = Thread(target=method, args=()).start()
def _is_doing(self, signal):
return signal in self._threads
def play(self):
# TODO play()
pass
class MCGAlbum:
_file_names = ['folder', 'cover']
_file_exts = ['jpg', 'jpeg', 'png']
def __init__(self, artist, title, path): def __init__(self, artist, title, path):
self.artist = artist self._artist = artist
if type(self.artist) is list: if type(self._artist) is list:
self.artist = self.artist[0] self._artist = self._artist[0]
self.title = title self._title = title
self.path = path self._path = path
self.tracks = [] self._cover = None
self.cover = None self._find_cover()
self._findCover()
def getArtist(self): def get_artist(self):
return self.artist return self._artist
def getTitle(self): def get_title(self):
return self.title return self._title
def getPath(self): def get_path(self):
return self.path return self._path
def addTrack(self, track): def get_cover(self):
self.tracks.append(track) return self._cover
def getTracks(self): def _find_cover(self):
return self.tracks names = list(self._file_names)
names.append(self._title)
names.append(' - '.join((self._artist, self._title)))
def getCover(self):
return self.cover
def _findCover(self):
names = list(self.fileNames)
names.append(self.title)
names.append(' - '.join((self.artist, self.title)))
for name in names: for name in names:
for ext in self.fileExts: for ext in self._file_exts:
filename = os.path.join('/home/oliver/Musik/', self.path, '.'.join([name, ext])) filename = os.path.join('/home/oliver/Musik/', self._path, '.'.join([name, ext]))
if os.path.isfile(filename): if os.path.isfile(filename):
self.cover = filename self._cover = filename
break break
if self.cover is not None: if self._cover is not None:
break break

View file

@ -5,121 +5,186 @@
from gi.repository import Gtk, Gdk, GdkPixbuf, GObject from gi.repository import Gtk, Gdk, GdkPixbuf, GObject
from MPDCoverGrid import MPDCoverGrid from MPDCoverGrid import MPDCoverGrid
import inspect
UI_INFO = """
<ui>
<toolbar name='ToolBar'>
<toolitem action='Connect' />
</toolbar>
</ui>
"""
class MPDCoverGridGTK(Gtk.Window): class MPDCoverGridGTK(Gtk.Window):
size = 128 _default_cover_size = 128
def __init__(self): def __init__(self):
Gtk.Window.__init__(self, title="MPDCoverGridGTK") Gtk.Window.__init__(self, title="MPDCoverGridGTK")
self.set_default_size(600, 400) self.set_default_size(600, 400)
self.connect("focus", self.updateSignal) self.connect("focus", self._focus)
self.connect("delete-event", self._destroy) self.connect("delete-event", self._destroy)
GObject.threads_init() self._cover_pixbuf = None
# VPaned # Box
VPaned = Gtk.VPaned() _main_box = Gtk.Box(orientation=Gtk.Orientation.VERTICAL)
self.add(VPaned)
# HPaned # HPaned
HPaned = Gtk.HPaned() hpaned = Gtk.HPaned()
VPaned.pack1(HPaned, resize=True)
# UIManager
action_group = Gtk.ActionGroup("toolbar")
ui_manager = Gtk.UIManager()
ui_manager.add_ui_from_string(UI_INFO)
accel_group = ui_manager.get_accel_group()
self.add_accel_group(accel_group)
ui_manager.insert_action_group(action_group)
self._action_connect = Gtk.Action("Connect", "_Connect", "Connect to server", Gtk.STOCK_DISCONNECT)
self._action_connect.connect("activate", self._on_toolbar_connect)
action_group.add_action_with_accel(self._action_connect, None)
# Toolbar
toolbar = ui_manager.get_widget("/ToolBar")
toolbar_context = toolbar.get_style_context()
toolbar_context.add_class(Gtk.STYLE_CLASS_PRIMARY_TOOLBAR)
_main_box.pack_start(toolbar, False, False, 0)
# Image # Image
self.coverImage = Gtk.Image() self._cover_image = Gtk.Image()
self._cover_image.connect('size-allocate', self._on_resize)
# EventBox # EventBox
self.coverBox = Gtk.EventBox() self._cover_box = Gtk.EventBox()
self.coverBox.add(self.coverImage) self._cover_box.add(self._cover_image)
# Viewport hpaned.pack1(self._cover_box, resize=True)
self.coverView = Gtk.Viewport()
self.coverView.add(self.coverBox)
HPaned.pack1(self.coverView, resize=True)
# GridModel # GridModel
self.coverGridModel = Gtk.ListStore(GdkPixbuf.Pixbuf, str, str) self._cover_grid_model = Gtk.ListStore(GdkPixbuf.Pixbuf, str, str)
# GridView # GridView
self.coverGrid = Gtk.IconView.new_with_model(self.coverGridModel) self._cover_grid = Gtk.IconView.new_with_model(self._cover_grid_model)
self.coverGrid.set_pixbuf_column(0) self._cover_grid.set_pixbuf_column(0)
self.coverGrid.set_text_column(-1) self._cover_grid.set_text_column(-1)
self.coverGrid.set_tooltip_column(2) self._cover_grid.set_tooltip_column(2)
self.coverGrid.set_columns(-1) self._cover_grid.set_columns(-1)
self.coverGrid.set_margin(0) self._cover_grid.set_margin(0)
self.coverGrid.set_row_spacing(0) self._cover_grid.set_row_spacing(0)
self.coverGrid.set_column_spacing(0) self._cover_grid.set_column_spacing(0)
self.coverGrid.set_item_padding(0) self._cover_grid.set_item_padding(0)
self.coverGrid.set_reorderable(False) self._cover_grid.set_reorderable(False)
self.coverGrid.set_selection_mode(Gtk.SelectionMode.SINGLE) self._cover_grid.set_selection_mode(Gtk.SelectionMode.SINGLE)
#color = self.get_style_context().lookup_color('bg_color')[1] #color = self.get_style_context().lookup_color('bg_color')[1]
#self.coverGrid.override_background_color(Gtk.StateFlags.NORMAL, Gdk.RGBA(color.red, color.green, color.blue, 1)) #self._cover_grid.override_background_color(Gtk.StateFlags.NORMAL, Gdk.RGBA(color.red, color.green, color.blue, 1))
# Scroll # Scroll
coverGridScroll = Gtk.ScrolledWindow() _cover_grid_scroll = Gtk.ScrolledWindow()
coverGridScroll.add_with_viewport(self.coverGrid) _cover_grid_scroll.add_with_viewport(self._cover_grid)
HPaned.pack2(coverGridScroll, resize=False) hpaned.pack2(_cover_grid_scroll, resize=False)
# ListModel _main_box.pack_start(hpaned, True, True, 0)
self.songListModel = Gtk.ListStore(str, str) self.add(_main_box)
# ListView
self.songList = Gtk.TreeView(self.songListModel) self._mcg = MPDCoverGrid()
renderer = Gtk.CellRendererText()
column1 = Gtk.TreeViewColumn("Artist", renderer, text=0)
column2 = Gtk.TreeViewColumn("Album", renderer, text=0)
self.songList.append_column(column1)
self.songList.append_column(column2)
self.songList.set_headers_visible(True)
VPaned.pack2(self.songList, resize=False)
# Signals # Signals
self.coverGrid.connect("selection-changed", self.coverGridShow) #self.coverGrid.connect("selection-changed", self.coverGridShow)
self.coverGrid.connect("item-activated", self.coverGridPlay) self._cover_grid.connect("item-activated", self._cover_grid_play)
self._mcg.connect_signal(MPDCoverGrid.SIGNAL_CONNECT, self._connect_callback)
self._initClient() self._mcg.connect_signal(MPDCoverGrid.SIGNAL_IDLE_PLAYER, self._idle_player_callback)
self.mcg.connectUpdate(self.updateCallback) self._mcg.connect_signal(MPDCoverGrid.SIGNAL_UPDATE, self._update_callback)
def _initClient(self): def _on_toolbar_connect(self, widget):
self.mcg = MPDCoverGrid() if self._mcg.is_connected():
self.mcg.connect() self._mcg.disconnect()
else:
self._mcg.connect()
def _connect_callback(self, connected, message):
if connected:
self._action_connect.set_stock_id(Gtk.STOCK_CONNECT)
else:
self._action_connect.set_stock_id(Gtk.STOCK_DISCONNECT)
def _idle_player_callback(self, state, album):
self._set_album(album.get_cover())
def _destroy(self, widget, state): def _destroy(self, widget, state):
if self.mcg is not None: if self._mcg is not None:
self.mcg.disconnect() self._mcg.disconnect()
Gtk.main_quit() Gtk.main_quit()
def updateSignal(self, widget, state): def _on_resize(self, widget, allocation):
self.update() self._resize_image()
def update(self): def _set_album(self, url):
if self.mcg is None: # Pfad überprüfen
if url is not None and url != "":
# Bild laden und zeichnen
self._cover_pixbuf = GdkPixbuf.Pixbuf.new_from_file(url)
self._resize_image()
else:
# Bild zurücksetzen
self._cover_pixbuf = None
self._cover_image.clear()
def _resize_image(self):
"""Diese Methode skaliert das geladene Bild aus dem Pixelpuffer
auf die Größe des Fensters unter Beibehalt der Seitenverhältnisse
"""
pixbuf = self._cover_pixbuf
size = self._cover_image.get_allocation()
## Pixelpuffer überprüfen
if pixbuf is None:
return return
self.mcg.update()
# Skalierungswert für Breite und Höhe ermitteln
ratioW = float(size.width) / float(pixbuf.get_width())
ratioH = float(size.height) / float(pixbuf.get_height())
# Kleineren beider Skalierungswerte nehmen, nicht Hochskalieren
ratio = min(ratioW, ratioH)
ratio = min(ratio, 1)
# Neue Breite und Höhe berechnen
width = int(round(pixbuf.get_width()*ratio))
height = int(round(pixbuf.get_height()*ratio))
# Pixelpuffer auf Oberfläche zeichnen
self._cover_image.set_from_pixbuf(pixbuf.scale_simple(width, height, GdkPixbuf.InterpType.HYPER))
def updateCallback(self, album):
if album.getCover() is not None:
pixbuf = GdkPixbuf.Pixbuf.new_from_file_at_size(album.getCover(), self.size, self.size)
def _focus(self, widget, state):
self._update()
def _update(self):
if self._mcg is None:
return
self._mcg.update()
def _update_callback(self, album):
if album.get_cover() is not None:
pixbuf = GdkPixbuf.Pixbuf.new_from_file_at_size(album.get_cover(), self._default_cover_size, self._default_cover_size)
if pixbuf is not None: if pixbuf is not None:
self.coverGridModel.append([pixbuf, album.getTitle(), ' von '.join([album.getTitle(), album.getArtist()])]) self._cover_grid_model.append([pixbuf, album.get_title(), "\n".join([album.get_title(), album.get_artist()])])
else: else:
print("pixbuf none: "+album.getTitle()) print("pixbuf none: "+album.get_title())
def coverGridShow(self, widget): def _player_callback(self, state):
# TODO coverGridShow() print(state)
pass
def coverGridSelected(self, widget, index, data): def _cover_grid_play(self, widget, item):
# TODO coverGridSelected()
pass
def coverGridPlay(self, widget, item):
# TODO coverGridPlay() # TODO coverGridPlay()
pass pass
@ -127,6 +192,7 @@ class MPDCoverGridGTK(Gtk.Window):
if __name__ == "__main__": if __name__ == "__main__":
GObject.threads_init()
mcgg = MPDCoverGridGTK() mcgg = MPDCoverGridGTK()
mcgg.show_all() mcgg.show_all()
Gtk.main() Gtk.main()