]> code.delx.au - notipod/blob - notipod_gui.py
Fixed UI
[notipod] / notipod_gui.py
1 #!/usr/bin/env python
2 # Copyright 2009 James Bunton <jamesbunton@fastmail.fm>
3 # Licensed for distribution under the GPL version 2, check COPYING for details
4
5 import logging
6
7 import objc
8 from Foundation import *
9 from AppKit import *
10 from PyObjCTools import AppHelper
11
12 import libnotipod
13
14
15 class PlaylistModel(NSObject):
16 outlineView = objc.IBOutlet()
17
18 def awakeFromNib(self):
19 self.root = []
20 self.playlists = {}
21 self.outlineView.setDataSource_(self)
22
23 def setPlaylists(self, playlists):
24 self.root = []
25 self.playlists = playlists
26 for playlist in self.playlists:
27 if playlist.parent is None:
28 self.root.append(playlist)
29 self.outlineView.reloadData()
30 self.outlineView.expandItem_expandChildren_(None, True)
31
32 def outlineView_child_ofItem_(self, _, childIndex, playlist):
33 if playlist == None:
34 return self.root[childIndex]
35 else:
36 return playlist.children[childIndex]
37
38 def outlineView_isItemExpandable_(self, _, playlist):
39 if playlist == None:
40 return True
41 else:
42 return len(playlist.children) > 0
43
44 def outlineView_numberOfChildrenOfItem_(self, _, playlist):
45 if playlist == None:
46 return len(self.root)
47 else:
48 return len(playlist.children)
49
50 def outlineView_objectValueForTableColumn_byItem_(self, _, col, playlist):
51 if not col:
52 return
53 col = col.identifier()
54
55 if col == "selected":
56 selected = NSApp.delegate().playlists()
57 return playlist.pid in selected
58 if col == "icon":
59 return NSImage.imageNamed_("playlist-" + playlist.ptype)
60 if col == "playlist":
61 return playlist.name
62
63 def outlineView_setObjectValue_forTableColumn_byItem_(self, _, v, col, playlist):
64 if not col:
65 return
66 col = col.identifier()
67
68
69 if col != "selected":
70 return
71 NSApp.delegate().setPlaylist_selected_(playlist.pid, v)
72
73
74 class FolderModel(NSObject):
75 window = objc.IBOutlet()
76 folderPopup = objc.IBOutlet()
77
78 def awakeFromNib(self):
79 folders = NSApp.delegate().folders()
80 self.folderPopup.addItemsWithTitles_(folders)
81 if len(folders) > 0:
82 self.folderPopup.selectItemAtIndex_(2)
83 self.lastIndex = 2
84 else:
85 self.lastIndex = 0
86
87 @objc.IBAction
88 def doSelectFolder_(self, sender):
89 currentIndex = self.folderPopup.indexOfSelectedItem()
90 if currentIndex >= 2:
91 self.lastIndex = currentIndex
92 NSApp.delegate().addFolder_(self.folderPopup.titleOfSelectedItem())
93 return
94 panel = NSOpenPanel.openPanel()
95 panel.setCanChooseFiles_(False)
96 panel.setCanChooseDirectories_(True)
97 panel.setAllowsMultipleSelection_(False)
98 panel.beginSheetForDirectory_file_types_modalForWindow_modalDelegate_didEndSelector_contextInfo_(
99 None, None, [], self.window, self, self.selectFolderEnd_returnCode_contextInfo_, None)
100
101 @objc.signature("v@:@ii")
102 def selectFolderEnd_returnCode_contextInfo_(self, panel, ret, _):
103 if ret == NSOKButton:
104 assert len(panel.filenames()) == 1
105 folder = panel.filenames()[0]
106 NSApp.delegate().addFolder_(folder)
107 self.folderPopup.insertItemWithTitle_atIndex_(folder, 2)
108 self.folderPopup.selectItemAtIndex_(2)
109 else:
110 self.folderPopup.selectItemAtIndex_(self.lastIndex)
111
112
113 class NotiPodController(NSObject):
114 window = objc.IBOutlet()
115 playlistModel = objc.IBOutlet()
116 folderModel = objc.IBOutlet()
117 loadingSheet = objc.IBOutlet()
118 loadingLabel = objc.IBOutlet()
119
120 def awakeFromNib(self):
121 self.runningGenerator = False
122
123 # Delegate methods
124 def applicationWillFinishLaunching_(self, _):
125 pass
126
127 def applicationDidFinishLaunching_(self, _):
128 self.library = libnotipod.ITunesLibrary.alloc().init()
129 def finish():
130 self.playlistModel.setPlaylists(self.library.get_playlists())
131 self.runGenerator(lambda: self.library.load_(None), finish)
132
133 def applicationWillTerminate_(self, _):
134 self.prefs().synchronize()
135
136 def applicationShouldTerminateAfterLastWindowClosed_(self, _):
137 return True
138
139
140 # Utility methods
141 def runGenerator(self, func, finish):
142 assert not self.runningGenerator
143 self.runningGenerator = True
144 NSApp.beginSheet_modalForWindow_modalDelegate_didEndSelector_contextInfo_(self.loadingSheet, self.window, None, None, None)
145 arg = (func(), finish)
146 self.performSelectorInBackground_withObject_(self.runGeneratorThread, arg)
147
148 def runGeneratorThread(self, (gen, finish)):
149 pool = NSAutoreleasePool.alloc().init()
150 for msg in gen:
151 if not self.runningGenerator:
152 break
153 self.loadingLabel.performSelectorOnMainThread_withObject_waitUntilDone_(
154 self.loadingLabel.setStringValue_, msg, True)
155 self.performSelectorOnMainThread_withObject_waitUntilDone_(
156 self.stopGenerator, finish, True)
157 self.runningGenerator = False
158 del pool
159
160 def stopGenerator(self, finish):
161 self.runningGenerator = False
162 NSApp.endSheet_(self.loadingSheet)
163 self.loadingSheet.orderOut_(self)
164 finish()
165
166 @objc.IBAction
167 def doCancel_(self, sender):
168 self.runningGenerator = False
169
170 @objc.IBAction
171 def doSync_(self, sender):
172 folder = self.folders()[0]
173 playlists = [self.library.get_playlist_pid(pid) for pid in self.playlists()]
174
175 all_tracks = []
176 for playlist in playlists:
177 all_tracks.extend(playlist.tracks)
178 libnotipod.export_m3u(dry_run=False, dest=folder, path_prefix="",
179 playlist_name=playlist.name, files=playlist.tracks)
180
181 def finish():
182 NSRunAlertPanel("Complete!", "Synchronisation is complete", "Ok", None, None)
183 self.runGenerator(
184 lambda:
185 libnotipod.sync(
186 dry_run=False,
187 source=self.library.folder,
188 dest=folder,
189 files_to_copy=all_tracks
190 )
191 ,
192 finish
193 )
194
195
196 # Public accessors
197
198 def prefs(self):
199 return NSUserDefaults.standardUserDefaults()
200
201 def _getArray(self, key):
202 res = self.prefs().stringArrayForKey_(key)
203 return list(res) if res else []
204
205 def _saveArray(self, key, array):
206 self.prefs().setObject_forKey_(array, key)
207
208 def playlists(self):
209 return self._getArray("playlists")
210
211 def folders(self):
212 return self._getArray("folders")
213
214 def addFolder_(self, folder):
215 folders = self.folders()
216 while folder in folders:
217 folders.remove(folder)
218 folders.insert(0, folder)
219 folders = folders[:10]
220 self._saveArray("folders", folders)
221
222 def setPlaylist_selected_(self, playlist, selected):
223 playlists = self.playlists()
224 if selected:
225 playlists.append(playlist)
226 else:
227 playlists.remove(playlist)
228 playlists = list(set(playlists))
229 self._saveArray("playlists", list(set(playlists)))
230
231
232 def main():
233 ### logging.basicConfig(format="%(levelname)s: %(message)s")
234 ### logging.getLogger().setLevel(logging.DEBUG)
235 AppHelper.runEventLoop()
236
237 if __name__ == "__main__":
238 main()
239