]> code.delx.au - notipod/blob - notipod_gui.py
Version 1.7
[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 import os
7 import sys
8 import traceback
9
10 import objc
11 from Foundation import *
12 from AppKit import *
13 from PyObjCTools import AppHelper
14
15 import libnotipod
16
17
18 class PlaylistModel(NSObject):
19 outlineView = objc.IBOutlet()
20
21 def awakeFromNib(self):
22 self.root = []
23 self.playlists = {}
24 self.outlineView.setDataSource_(self)
25
26 def setPlaylists(self, playlists):
27 self.root = []
28 self.playlists = playlists
29 for playlist in self.playlists:
30 if playlist.parent is None:
31 self.root.append(playlist)
32 self.outlineView.reloadData()
33 self.outlineView.expandItem_expandChildren_(None, True)
34
35 def outlineView_child_ofItem_(self, _, childIndex, playlist):
36 if playlist == None:
37 return self.root[childIndex]
38 else:
39 return playlist.children[childIndex]
40
41 def outlineView_isItemExpandable_(self, _, playlist):
42 if playlist == None:
43 return True
44 else:
45 return len(playlist.children) > 0
46
47 def outlineView_numberOfChildrenOfItem_(self, _, playlist):
48 if playlist == None:
49 return len(self.root)
50 else:
51 return len(playlist.children)
52
53 def outlineView_objectValueForTableColumn_byItem_(self, _, col, playlist):
54 if not col:
55 return
56 col = col.identifier()
57
58 if col == "selected":
59 selected = NSApp.delegate().playlists()
60 return playlist.pid in selected
61 if col == "icon":
62 return NSImage.imageNamed_("playlist-" + playlist.ptype)
63 if col == "playlist":
64 return playlist.name
65
66 def outlineView_setObjectValue_forTableColumn_byItem_(self, _, v, col, playlist):
67 if not col:
68 return
69 col = col.identifier()
70
71
72 if col != "selected":
73 return
74 NSApp.delegate().setPlaylist_selected_(playlist.pid, v)
75
76
77 class FolderModel(NSObject):
78 window = objc.IBOutlet()
79 folderPopup = objc.IBOutlet()
80
81 def awakeFromNib(self):
82 folders = NSApp.delegate().folders()
83 self.folderPopup.addItemsWithTitles_(folders)
84 if len(folders) > 0:
85 self.folderPopup.selectItemAtIndex_(2)
86 self.lastIndex = 2
87 else:
88 self.lastIndex = 0
89
90 @objc.IBAction
91 def doSelectFolder_(self, sender):
92 currentIndex = self.folderPopup.indexOfSelectedItem()
93 if currentIndex >= 2:
94 self.lastIndex = currentIndex
95 NSApp.delegate().addFolder_(self.folderPopup.titleOfSelectedItem())
96 return
97 panel = NSOpenPanel.openPanel()
98 panel.setCanChooseFiles_(False)
99 panel.setCanChooseDirectories_(True)
100 panel.setAllowsMultipleSelection_(False)
101 panel.beginSheetForDirectory_file_types_modalForWindow_modalDelegate_didEndSelector_contextInfo_(
102 None, None, [], self.window, self, self.selectFolderEnd_returnCode_contextInfo_, None)
103
104 @objc.signature("v@:@ii")
105 def selectFolderEnd_returnCode_contextInfo_(self, panel, ret, _):
106 if ret == NSOKButton:
107 assert len(panel.filenames()) == 1
108 folder = panel.filenames()[0]
109 NSApp.delegate().addFolder_(folder)
110 self.folderPopup.insertItemWithTitle_atIndex_(folder, 2)
111 self.folderPopup.selectItemAtIndex_(2)
112 else:
113 self.folderPopup.selectItemAtIndex_(self.lastIndex)
114
115
116 class NotiPodController(NSObject):
117 window = objc.IBOutlet()
118
119 loadingSheet = objc.IBOutlet()
120 loadingLabel = objc.IBOutlet()
121 loadingIndicator = objc.IBOutlet()
122
123 previewWindow = objc.IBOutlet()
124 previewText = objc.IBOutlet()
125
126 playlistModel = objc.IBOutlet()
127 folderModel = objc.IBOutlet()
128
129
130 def awakeFromNib(self):
131 self.runningGenerator = False
132 self.previewWindow.setReleasedWhenClosed_(False)
133
134 # Delegate methods
135 def applicationWillFinishLaunching_(self, _):
136 pass
137
138 def applicationDidFinishLaunching_(self, _):
139 self.library = libnotipod.ITunesLibrary.alloc().init()
140 def finish():
141 self.playlistModel.setPlaylists(self.library.get_playlists())
142 def fail():
143 sys.exit(0)
144 self.runGenerator(lambda: self.library.load_(None), finish, fail)
145
146 def applicationWillTerminate_(self, _):
147 self.prefs().synchronize()
148
149 def applicationShouldTerminateAfterLastWindowClosed_(self, _):
150 return True
151
152
153 # Utility methods
154 def runGenerator(self, func, finish, fail):
155 assert not self.runningGenerator
156 self.runningGenerator = True
157 self.loadingIndicator.startAnimation_(self)
158 NSApp.beginSheet_modalForWindow_modalDelegate_didEndSelector_contextInfo_(self.loadingSheet, self.window, None, None, None)
159 arg = (func(), finish, fail)
160 self.performSelectorInBackground_withObject_(self.runGeneratorThread, arg)
161
162 def runGeneratorThread(self, (gen, finish, fail)):
163 pool = NSAutoreleasePool.alloc().init()
164 try:
165 for msg in gen:
166 if not self.runningGenerator:
167 break
168 self.loadingLabel.performSelectorOnMainThread_withObject_waitUntilDone_(
169 self.loadingLabel.setStringValue_, msg, True)
170 except Exception, e:
171 NSRunAlertPanel("Error!", str(e), "Ok", None, None)
172 traceback.print_exc()
173 finish = fail
174 self.performSelectorOnMainThread_withObject_waitUntilDone_(
175 self.stopGenerator, finish, True)
176 self.runningGenerator = False
177
178 def stopGenerator(self, finish):
179 self.runningGenerator = False
180 NSApp.endSheet_(self.loadingSheet)
181 self.loadingSheet.orderOut_(self)
182 self.loadingIndicator.stopAnimation_(self)
183 if finish:
184 finish()
185
186 @objc.IBAction
187 def doCancel_(self, sender):
188 self.runningGenerator = False
189
190 def doPreviewThread(self):
191 yield "Calculating changes..."
192
193 folder = self.folders()[0]
194 if not os.path.isdir(folder.encode("utf-8")):
195 NSRunAlertPanel("Error!", "Destination " + folder + " does not exist, try mounting it first?", "Ok", None, None)
196 return
197
198 all_tracks = []
199 for playlist_id in self.playlists():
200 playlist = self.library.get_playlist_pid(playlist_id)
201 if playlist is not None:
202 all_tracks.extend(playlist.tracks)
203
204 gen = libnotipod.sync(
205 dry_run=True,
206 source=self.library.folder,
207 dest=folder,
208 files_to_copy=all_tracks
209 )
210 self.previewResult = "\n".join(gen)
211
212 @objc.IBAction
213 def doPreview_(self, sender):
214 self.previewResult = ""
215 self.previewWindow.orderOut_(self)
216
217 def finish():
218 self.previewText.textStorage().mutableString().setString_(self.previewResult)
219 self.previewWindow.center()
220 self.previewWindow.makeKeyAndOrderFront_(self)
221
222 self.runGenerator(self.doPreviewThread, finish, None)
223
224 @objc.IBAction
225 def doSync_(self, sender):
226 folder = self.folders()[0]
227 if not os.path.isdir(folder.encode("utf-8")):
228 NSRunAlertPanel("Error!", "Destination " + folder + " does not exist, try mounting it first?", "Ok", None, None)
229 return
230
231 all_tracks = []
232 for playlist_id in self.playlists():
233 playlist = self.library.get_playlist_pid(playlist_id)
234 if playlist is None:
235 print "Forgetting unknown playlist:", playlist_id
236 self.setPlaylist_selected_(playlist_id, False)
237 continue
238 all_tracks.extend(playlist.tracks)
239 libnotipod.export_m3u(dry_run=False, dest=folder, path_prefix="",
240 playlist_name=playlist.name, files=playlist.tracks)
241
242 def finish():
243 NSRunAlertPanel("Complete!", "Synchronisation is complete", "Ok", None, None)
244 self.runGenerator(
245 lambda:
246 libnotipod.sync(
247 dry_run=False,
248 source=self.library.folder,
249 dest=folder,
250 files_to_copy=all_tracks
251 )
252 ,
253 finish,
254 None
255 )
256
257
258 # Public accessors
259
260 def prefs(self):
261 return NSUserDefaults.standardUserDefaults()
262
263 def _getArray(self, key):
264 res = self.prefs().stringArrayForKey_(key)
265 return list(res) if res else []
266
267 def _saveArray(self, key, array):
268 self.prefs().setObject_forKey_(array, key)
269
270 def playlists(self):
271 return self._getArray("playlists")
272
273 def folders(self):
274 return self._getArray("folders")
275
276 def addFolder_(self, folder):
277 folders = self.folders()
278 while folder in folders:
279 folders.remove(folder)
280 folders.insert(0, folder)
281 folders = folders[:10]
282 self._saveArray("folders", folders)
283
284 def setPlaylist_selected_(self, playlist, selected):
285 playlists = self.playlists()
286 if selected:
287 playlists.append(playlist)
288 else:
289 playlists.remove(playlist)
290 playlists = list(set(playlists))
291 self._saveArray("playlists", list(set(playlists)))
292
293
294 def main():
295 ### logging.basicConfig(format="%(levelname)s: %(message)s")
296 ### logging.getLogger().setLevel(logging.DEBUG)
297 AppHelper.runEventLoop()
298
299 if __name__ == "__main__":
300 main()
301