2 # Copyright 2009 James Bunton <jamesbunton@fastmail.fm>
3 # Licensed for distribution under the GPL version 2, check COPYING for details
8 from Foundation
import *
10 from PyObjCTools
import AppHelper
15 class PlaylistModel(NSObject
):
16 outlineView
= objc
.IBOutlet()
18 def awakeFromNib(self
):
21 self
.outlineView
.setDataSource_(self
)
23 def setPlaylists(self
, playlists
):
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)
32 def outlineView_child_ofItem_(self
, _
, childIndex
, playlist
):
34 return self
.root
[childIndex
]
36 return playlist
.children
[childIndex
]
38 def outlineView_isItemExpandable_(self
, _
, playlist
):
42 return len(playlist
.children
) > 0
44 def outlineView_numberOfChildrenOfItem_(self
, _
, playlist
):
48 return len(playlist
.children
)
50 def outlineView_objectValueForTableColumn_byItem_(self
, _
, col
, playlist
):
51 col
= col
.identifier() if col
else "playlist"
54 selected
= NSApp
.delegate().playlists()
55 return playlist
.pid
in selected
56 if col
== None or col
== "playlist":
59 def outlineView_setObjectValue_forTableColumn_byItem_(self
, _
, v
, col
, playlist
):
60 col
= col
.identifier() if col
else "playlist"
64 NSApp
.delegate().setPlaylist_selected_(playlist
.pid
, v
)
67 class FolderModel(NSObject
):
68 window
= objc
.IBOutlet()
69 folderPopup
= objc
.IBOutlet()
71 def awakeFromNib(self
):
72 self
.folderPopup
.addItemsWithTitles_(NSApp
.delegate().folders())
73 self
.folderPopup
.selectItemAtIndex_(2)
77 def doSelectFolder_(self
, sender
):
78 currentIndex
= self
.folderPopup
.indexOfSelectedItem()
80 self
.lastIndex
= currentIndex
81 NSApp
.delegate().addFolder_(self
.folderPopup
.titleOfSelectedItem())
83 panel
= NSOpenPanel
.openPanel()
84 panel
.setCanChooseFiles_(False)
85 panel
.setCanChooseDirectories_(True)
86 panel
.setAllowsMultipleSelection_(False)
87 panel
.beginSheetForDirectory_file_types_modalForWindow_modalDelegate_didEndSelector_contextInfo_(
88 None, None, [], self
.window
, self
, self
.selectFolderEnd_returnCode_contextInfo_
, None)
90 @objc.signature("v@:@ii")
91 def selectFolderEnd_returnCode_contextInfo_(self
, panel
, ret
, _
):
93 assert len(panel
.filenames()) == 1
94 folder
= panel
.filenames()[0]
95 NSApp
.delegate().addFolder_(folder
)
96 self
.folderPopup
.insertItemWithTitle_atIndex_(folder
, 2)
97 self
.folderPopup
.selectItemAtIndex_(2)
99 self
.folderPopup
.selectItemAtIndex_(self
.lastIndex
)
102 class NotiPodAppDelegate(NSObject
):
103 window
= objc
.IBOutlet()
104 playlistModel
= objc
.IBOutlet()
105 folderModel
= objc
.IBOutlet()
106 loadingSheet
= objc
.IBOutlet()
107 synchronizingSheet
= objc
.IBOutlet()
111 def applicationWillFinishLaunching_(self
, _
):
114 def applicationDidFinishLaunching_(self
, _
):
115 NSApp
.beginSheet_modalForWindow_modalDelegate_didEndSelector_contextInfo_(self
.loadingSheet
, self
.window
, None, None, None)
116 self
.performSelectorInBackground_withObject_(self
.loadFromThread
, None)
118 def applicationWillTerminate_(self
, _
):
119 self
.prefs().synchronize()
121 def applicationShouldTerminateAfterLastWindowClosed_(self
, _
):
125 def loadFromThread(self
):
126 pool
= NSAutoreleasePool
.alloc().init()
127 self
.library
= libsyncitunes
.ITunesLibrary
.alloc().init()
128 self
.performSelectorOnMainThread_withObject_waitUntilDone_(self
.finishLoading
, None, True)
131 def finishLoading(self
):
132 self
.playlistModel
.setPlaylists(self
.library
.get_playlists())
133 NSApp
.endSheet_(self
.loadingSheet
)
134 self
.loadingSheet
.close()
137 def doSync_(self
, sender
):
138 NSApp
.beginSheet_modalForWindow_modalDelegate_didEndSelector_contextInfo_(self
.synchronizingSheet
, self
.window
, None, None, None)
139 self
.performSelectorInBackground_withObject_(self
.syncFromThread
, None)
141 def syncFromThread(self
):
142 pool
= NSAutoreleasePool
.alloc().init()
144 folder
= self
.folders()[0]
145 playlists
= [self
.library
.get_playlist_pid(pid
) for pid
in self
.playlists()]
148 for playlist
in playlists
:
149 all_tracks
.extend(playlist
.tracks
)
150 libsyncitunes
.export_m3u(dry_run
=False, dest
=folder
, path_prefix
="",
151 playlist_name
=playlist
.name
, files
=playlist
.tracks
)
153 libsyncitunes
.sync(dry_run
=False,
154 source
=self
.library
.folder
, dest
=folder
, files
=all_tracks
)
156 self
.performSelectorOnMainThread_withObject_waitUntilDone_(self
.finishSync
, None, True)
159 def finishSync(self
):
160 NSApp
.endSheet_(self
.synchronizingSheet
)
161 self
.synchronizingSheet
.close()
167 return NSUserDefaults
.standardUserDefaults()
169 def _getArray(self
, key
):
170 res
= self
.prefs().stringArrayForKey_(key
)
171 return list(res
) if res
else []
173 def _saveArray(self
, key
, array
):
174 self
.prefs().setObject_forKey_(array
, key
)
177 return self
._getArray
("playlists")
180 return self
._getArray
("folders")
182 def addFolder_(self
, folder
):
183 folders
= self
.folders()
184 while folder
in folders
:
185 folders
.remove(folder
)
186 folders
.insert(0, folder
)
187 folders
= folders
[:10]
188 self
._saveArray
("folders", folders
)
190 def setPlaylist_selected_(self
, playlist
, selected
):
191 playlists
= self
.playlists()
193 playlists
.append(playlist
)
195 playlists
.remove(playlist
)
196 playlists
= list(set(playlists
))
197 self
._saveArray
("playlists", list(set(playlists
)))
201 ### logging.basicConfig(format="%(levelname)s: %(message)s")
202 ### logging.getLogger().setLevel(logging.DEBUG)
203 AppHelper
.runEventLoop()
205 if __name__
== "__main__":