2 # Copyright 2009 James Bunton <jamesbunton@fastmail.fm>
3 # Licensed for distribution under the GPL version 2, check COPYING for details
11 from Foundation
import *
13 from PyObjCTools
import AppHelper
18 class PlaylistModel(NSObject
):
19 outlineView
= objc
.IBOutlet()
21 def awakeFromNib(self
):
24 self
.outlineView
.setDataSource_(self
)
26 def setPlaylists(self
, playlists
):
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)
35 def outlineView_child_ofItem_(self
, _
, childIndex
, playlist
):
37 return self
.root
[childIndex
]
39 return playlist
.children
[childIndex
]
41 def outlineView_isItemExpandable_(self
, _
, playlist
):
45 return len(playlist
.children
) > 0
47 def outlineView_numberOfChildrenOfItem_(self
, _
, playlist
):
51 return len(playlist
.children
)
53 def outlineView_objectValueForTableColumn_byItem_(self
, _
, col
, playlist
):
56 col
= col
.identifier()
59 selected
= NSApp
.delegate().playlists()
60 return playlist
.pid
in selected
62 return NSImage
.imageNamed_("playlist-" + playlist
.ptype
)
66 def outlineView_setObjectValue_forTableColumn_byItem_(self
, _
, v
, col
, playlist
):
69 col
= col
.identifier()
74 NSApp
.delegate().setPlaylist_selected_(playlist
.pid
, v
)
77 class FolderModel(NSObject
):
78 window
= objc
.IBOutlet()
79 folderPopup
= objc
.IBOutlet()
81 def awakeFromNib(self
):
82 folders
= NSApp
.delegate().folders()
83 self
.folderPopup
.addItemsWithTitles_(folders
)
85 self
.folderPopup
.selectItemAtIndex_(2)
91 def doSelectFolder_(self
, sender
):
92 currentIndex
= self
.folderPopup
.indexOfSelectedItem()
94 self
.lastIndex
= currentIndex
95 NSApp
.delegate().addFolder_(self
.folderPopup
.titleOfSelectedItem())
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)
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)
113 self
.folderPopup
.selectItemAtIndex_(self
.lastIndex
)
116 class NotiPodController(NSObject
):
117 window
= objc
.IBOutlet()
119 loadingSheet
= objc
.IBOutlet()
120 loadingLabel
= objc
.IBOutlet()
121 loadingIndicator
= objc
.IBOutlet()
123 previewWindow
= objc
.IBOutlet()
124 previewText
= objc
.IBOutlet()
126 playlistModel
= objc
.IBOutlet()
127 folderModel
= objc
.IBOutlet()
130 def awakeFromNib(self
):
131 self
.runningGenerator
= False
134 def applicationWillFinishLaunching_(self
, _
):
137 def applicationDidFinishLaunching_(self
, _
):
138 self
.library
= libnotipod
.ITunesLibrary
.alloc().init()
140 self
.playlistModel
.setPlaylists(self
.library
.get_playlists())
143 self
.runGenerator(lambda: self
.library
.load_(None), finish
, fail
)
145 def applicationWillTerminate_(self
, _
):
146 self
.prefs().synchronize()
148 def applicationShouldTerminateAfterLastWindowClosed_(self
, _
):
153 def runGenerator(self
, func
, finish
, fail
):
154 assert not self
.runningGenerator
155 self
.runningGenerator
= True
156 self
.loadingIndicator
.startAnimation_(self
)
157 NSApp
.beginSheet_modalForWindow_modalDelegate_didEndSelector_contextInfo_(self
.loadingSheet
, self
.window
, None, None, None)
158 arg
= (func(), finish
, fail
)
159 self
.performSelectorInBackground_withObject_(self
.runGeneratorThread
, arg
)
161 def runGeneratorThread(self
, (gen
, finish
, fail
)):
162 pool
= NSAutoreleasePool
.alloc().init()
165 if not self
.runningGenerator
:
167 self
.loadingLabel
.performSelectorOnMainThread_withObject_waitUntilDone_(
168 self
.loadingLabel
.setStringValue_
, msg
, True)
170 NSRunAlertPanel("Error!", str(e
), "Ok", None, None)
171 traceback
.print_exc()
173 self
.performSelectorOnMainThread_withObject_waitUntilDone_(
174 self
.stopGenerator
, finish
, True)
175 self
.runningGenerator
= False
177 def stopGenerator(self
, finish
):
178 self
.runningGenerator
= False
179 NSApp
.endSheet_(self
.loadingSheet
)
180 self
.loadingSheet
.orderOut_(self
)
181 self
.loadingIndicator
.stopAnimation_(self
)
186 def doCancel_(self
, sender
):
187 self
.runningGenerator
= False
189 def doPreviewThread(self
):
190 yield "Calculating changes..."
192 folder
= self
.folders()[0]
193 if not os
.path
.isdir(folder
.encode("utf-8")):
194 NSRunAlertPanel("Error!", "Destination " + folder
+ " does not exist, try mounting it first?", "Ok", None, None)
198 for playlist_id
in self
.playlists():
199 playlist
= self
.library
.get_playlist_pid(playlist_id
)
200 if playlist
is not None:
201 all_tracks
.extend(playlist
.tracks
)
203 gen
= libnotipod
.sync(
205 source
=self
.library
.folder
,
207 files_to_copy
=all_tracks
209 self
.previewResult
= "\n".join(gen
)
212 def doPreview_(self
, sender
):
213 self
.previewResult
= ""
214 self
.previewWindow
.orderOut_(self
)
217 self
.previewText
.textStorage().mutableString().setString_(self
.previewResult
)
218 self
.previewWindow
.center()
219 self
.previewWindow
.makeKeyAndOrderFront_(self
)
221 self
.runGenerator(self
.doPreviewThread
, finish
, None)
224 def doSync_(self
, sender
):
225 folder
= self
.folders()[0]
226 if not os
.path
.isdir(folder
.encode("utf-8")):
227 NSRunAlertPanel("Error!", "Destination " + folder
+ " does not exist, try mounting it first?", "Ok", None, None)
231 for playlist_id
in self
.playlists():
232 playlist
= self
.library
.get_playlist_pid(playlist_id
)
234 print "Forgetting unknown playlist:", playlist_id
235 self
.setPlaylist_selected_(playlist_id
, False)
237 all_tracks
.extend(playlist
.tracks
)
238 libnotipod
.export_m3u(dry_run
=False, dest
=folder
, path_prefix
="",
239 playlist_name
=playlist
.name
, files
=playlist
.tracks
)
242 NSRunAlertPanel("Complete!", "Synchronisation is complete", "Ok", None, None)
247 source
=self
.library
.folder
,
249 files_to_copy
=all_tracks
260 return NSUserDefaults
.standardUserDefaults()
262 def _getArray(self
, key
):
263 res
= self
.prefs().stringArrayForKey_(key
)
264 return list(res
) if res
else []
266 def _saveArray(self
, key
, array
):
267 self
.prefs().setObject_forKey_(array
, key
)
270 return self
._getArray
("playlists")
273 return self
._getArray
("folders")
275 def addFolder_(self
, folder
):
276 folders
= self
.folders()
277 while folder
in folders
:
278 folders
.remove(folder
)
279 folders
.insert(0, folder
)
280 folders
= folders
[:10]
281 self
._saveArray
("folders", folders
)
283 def setPlaylist_selected_(self
, playlist
, selected
):
284 playlists
= self
.playlists()
286 playlists
.append(playlist
)
288 playlists
.remove(playlist
)
289 playlists
= list(set(playlists
))
290 self
._saveArray
("playlists", list(set(playlists
)))
294 ### logging.basicConfig(format="%(levelname)s: %(message)s")
295 ### logging.getLogger().setLevel(logging.DEBUG)
296 AppHelper
.runEventLoop()
298 if __name__
== "__main__":