window.py 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387
  1. """
  2. [Module documentaion here]
  3. @author:[author], [year]
  4. """
  5. from PyQt5 import QtWidgets
  6. from path import Path
  7. from PyQt5.QtGui import QIcon
  8. from PyQt5.QtWidgets import QMainWindow, QListWidgetItem, QTableWidgetItem, QFileDialog, QMessageBox
  9. from core.logging_ import Logger
  10. from core.models import MusicFolder, TrackTag
  11. from core.repositories import MusicFolderRepository, TrackRepository, SessionTrackRepository, SessionRepository, \
  12. TrackTagRepository
  13. from ui.qt.dlg_meta_editor import DlgMetaEditor
  14. from ui.qt.dlg_playlist import DlgPlaylist
  15. from ui.qt.dlg_select_playlist import DlgSelectPlaylist
  16. from ui.qt.dlg_tag_editor import DlgTagEditor
  17. from ui.qt.main_ui import Ui_mainWindow
  18. from ui.qt.widgets.explorertable import ExplorerTable
  19. from ui.qt.widgets.playlist_table import PlaylistTable
  20. logger = Logger.get()
  21. class MainWindow(QMainWindow):
  22. def __init__(self, settings=None):
  23. super().__init__()
  24. self.settings = settings or {}
  25. self.selected_playlist = None
  26. self.playing_queue = []
  27. self.selected_track = None # /!\ the selected track is not necessarily the one being played!
  28. self.createWidgets()
  29. def createWidgets(self):
  30. self.ui = Ui_mainWindow()
  31. self.ui.setupUi(self)
  32. # Stack and menus
  33. self.ui.stack.setCurrentIndex(0)
  34. menu_items = {
  35. 0: (':/img/rsc/dance.png', 'Ma séance'),
  36. # 1: (':/img/rsc/writing-tool.png', 'Préparer'),
  37. 2: (':/img/rsc/map.png', 'Explorer'),
  38. # 3: (':/img/rsc/calendar.png', 'Agenda'),
  39. 4: (':/img/rsc/settings.png', 'Paramètres'),
  40. }
  41. for i, item in menu_items.items():
  42. icon, lbl = item
  43. item = QListWidgetItem(QIcon(icon), lbl)
  44. item.index = i
  45. self.ui.menu.addItem(item)
  46. QtWidgets.qApp.focusChanged.connect(self.focus_changed)
  47. # Menu item clicked
  48. self.ui.menu.itemClicked.connect(self.menu_item_selected)
  49. # Page 1 - Homepage
  50. self.ui.sessionPlaylist.trackDoubleClicked.connect(self.play_playlist)
  51. self.ui.sessionBtnChange.clicked.connect(self.selectPlaylist)
  52. self.ui.btnSessionStart.clicked.connect(self.play_playlist)
  53. # Page 3 - explorer
  54. self.ui.explorerTable.populate()
  55. self.ui.explorerTable.trackSelected.connect(self.newTrackSelected)
  56. self.ui.explorerTable.trackDoubleClicked.connect(self.play_track)
  57. self.ui.btnExplorerRefresh.clicked.connect(self.refresh_explorer_tree)
  58. self.ui.explorerLineSearch.editingFinished.connect(self.explorerSearchChanged)
  59. self.ui.explorerLineSearch.textChanged.connect(lambda s: (s or self.explorerSearchChanged())) # when search bar is cleared
  60. self.ui.explorerBtnSearch.clicked.connect(self.explorerSearchChanged)
  61. self.ui.explorerTrackEdit.clicked.connect(self.showMetaEditor)
  62. # self.ui.btnEditTags.clicked.connect(self.showTagsEditor)
  63. self.ui.explorerTrackTagsTable.tagChecked.connect(self.addTrackTags)
  64. self.ui.explorerTrackTagsTable.tagUnchecked.connect(self.removeTrackTags)
  65. self.ui.explorerTrackMetaStack.setCurrentIndex(0)
  66. self.ui.explorerTrackPlay.clicked.connect(self.explorerPlaySelected)
  67. self.ui.btnSelectPlaylist.clicked.connect(self.selectPlaylist)
  68. self.ui.explorerAddToPlaylist.clicked.connect(self.add_to_playlist)
  69. self.ui.explorerRemoveFromPlaylist.clicked.connect(self.remove_from_playlist)
  70. self.ui.explorerPlaylist.trackSelected.connect(self.newTrackSelected)
  71. self.ui.explorerPlaylist.trackDoubleClicked.connect(self.play_playlist)
  72. # Page 5 - settings
  73. self.ui.settingsMusicFoldersTable.setColumnHidden(0, 1)
  74. self.ui.musicFoldersAddButton.clicked.connect(self.add_music_folder)
  75. self.ui.musicFoldersRemoveButton.clicked.connect(self.remove_music_folder)
  76. self.populate_music_folders_table()
  77. self.ui.tagsTableAddButton.clicked.connect(self.addTag)
  78. self.ui.tableTagsRemoveButton.clicked.connect(self.removeTag)
  79. self.ui.settingsTagsTableWidget.selectable = False
  80. self.ui.settingsTagsTableWidget.populate()
  81. self.ui.settingsTagsTableWidget.contentChanged.connect(self.ui.explorerTrackTagsTable.populate)
  82. # vlc frame
  83. self.ui.vlcFrame.trackStarted.connect(self.vlcTrackStarted)
  84. self.ui.vlcFrame.playlistEnded.connect(self.vlcPlaylistEnded)
  85. # Apply settings
  86. if self.settings:
  87. self.ui.vlcFrame.set_volume(int(self.settings['volume']))
  88. if self.settings['muted']:
  89. self.ui.vlcFrame.toggle_muted()
  90. playlist_id = self.settings['playlist']
  91. if playlist_id is not None and playlist_id > 0:
  92. playlist_repo = SessionRepository()
  93. playlist = playlist_repo.get_by_id(playlist_id)
  94. self.loadPlaylist(playlist)
  95. def focus_changed(self, old, new):
  96. if old is self.ui.explorerTrackNotepad:
  97. self.saveTrackNotes()
  98. def menu_item_selected(self, e):
  99. self.ui.stack.setCurrentIndex(e.index)
  100. def indexation_started(self):
  101. self.ui.statusbar.setStatusTip("Indexation en cours...")
  102. def indexation_ended(self):
  103. self.ui.statusbar.setStatusTip("Indexation terminée.")
  104. def refresh_explorer_tree(self, tracks=None):
  105. if tracks:
  106. self.ui.explorerTable.update_tracks(tracks)
  107. else:
  108. self.ui.explorerTable.populate()
  109. self.ui.explorerLineSearch.clear()
  110. def newTrackSelected(self, track=None):
  111. sender = self.sender()
  112. self.selected_track = None
  113. self.ui.explorerTrackMetaStack.setCurrentIndex(0)
  114. self.ui.explorerTrackPlay.setEnabled(False)
  115. self.ui.explorerAddToPlaylist.setEnabled(False)
  116. self.ui.explorerRemoveFromPlaylist.setEnabled(False)
  117. self.ui.explorerTrackNotepad.setText("")
  118. if not track:
  119. return
  120. self.selected_track = track
  121. # track infos
  122. self.update_meta()
  123. self.ui.explorerTrackNotepad.setHtml(track.note)
  124. self.ui.explorerTrackTagsTable.populate(track)
  125. self.ui.explorerTrackMetaStack.setCurrentIndex(1)
  126. self.ui.explorerTrackPlay.setEnabled(True)
  127. if type(sender) is ExplorerTable:
  128. self.ui.explorerAddToPlaylist.setEnabled(True)
  129. elif type(sender) is PlaylistTable:
  130. self.ui.explorerRemoveFromPlaylist.setEnabled(True)
  131. else:
  132. raise RuntimeError("Unknown sender")
  133. self.ui.explorerTable.viewport().repaint()
  134. self.ui.explorerPlaylist.viewport().repaint()
  135. self.ui.sessionPlaylist.viewport().repaint()
  136. def update_meta(self):
  137. if self.selected_track:
  138. title, artist, album, track_num = self.selected_track.title, \
  139. self.selected_track.artist, \
  140. self.selected_track.album, \
  141. self.selected_track.track_num
  142. else:
  143. title, artist, album, track_num = "", "", "", ""
  144. self.ui.explorerLblTrackTitle.setText(title)
  145. self.ui.explorerLblTrackArtist.setText(artist)
  146. self.ui.explorerLblTrackAlbum.setText(album)
  147. self.ui.explorerLblTrackNumber.setText(
  148. str(track_num if track_num is not None else "")
  149. )
  150. def populate_music_folders_table(self):
  151. music_folders = MusicFolderRepository().get_all()
  152. self.ui.settingsMusicFoldersTable.setRowCount(0)
  153. self.ui.settingsMusicFoldersTable.setRowCount(len(music_folders))
  154. music_folder_statuses = [
  155. ('Inconnu', ':/img/rsc/unknown.png'),
  156. ('Valide', ':/img/rsc/valid.png'),
  157. ('Inaccessible', ':/img/rsc/invalid.png')
  158. ]
  159. for i, music_folder in enumerate(music_folders):
  160. self.ui.settingsMusicFoldersTable.setItem(i, 0, QTableWidgetItem(music_folder.id))
  161. status_lbl, status_pic = music_folder_statuses[music_folder.status]
  162. self.ui.settingsMusicFoldersTable.setItem(i, 1, QTableWidgetItem(QIcon(status_pic), status_lbl))
  163. self.ui.settingsMusicFoldersTable.setItem(i, 2, QTableWidgetItem(music_folder.path))
  164. def addTag(self):
  165. self.ui.settingsTagsTableWidget.add()
  166. def removeTag(self):
  167. if QMessageBox.question(self, "Confirmer", "Êtes-vous sûr(e) de vouloir supprimer cette étiquette?") != QMessageBox.Yes:
  168. return
  169. self.ui.settingsTagsTableWidget.removeSelected()
  170. def explorerPlaySelected(self):
  171. track = self.ui.explorerTable.selected_track()
  172. if track is None:
  173. return
  174. self.play_track(track)
  175. def play_track(self, track):
  176. logger.info("Start playing: %s" % track)
  177. self.ui.vlcFrame.load_track(track)
  178. self.ui.vlcFrame.play()
  179. def showMetaEditor(self):
  180. if not self.selected_track:
  181. return
  182. r = DlgMetaEditor.edit(self, self.selected_track)
  183. if r:
  184. self.update_meta()
  185. def add_music_folder(self):
  186. path = QFileDialog.getExistingDirectory(self, "Sélectionnez le dossier à ajouter")
  187. if not path:
  188. return
  189. path = Path(path)
  190. repo = MusicFolderRepository()
  191. music_folders = repo.get_all()
  192. # for folder in music_folders:
  193. # if path == Path(folder.path):
  194. # QMessageBox.warning(self, "Ajout invalide", "Ce dossier a déjà été ajouté")
  195. # return
  196. #
  197. # if is_subdir_of(path, Path(folder.path)):
  198. # QMessageBox.warning(self, "Ajout invalide", "Ce dossier est contenu dans un dossier existant")
  199. # return
  200. folder = MusicFolder(path=path)
  201. repo.create(folder, True)
  202. self.populate_music_folders_table()
  203. def remove_music_folder(self):
  204. pass
  205. def createOrEditPlaylist(self):
  206. r = DlgPlaylist.edit(self)
  207. def selectPlaylist(self):
  208. playlist = DlgSelectPlaylist.select(self)
  209. if playlist:
  210. self.loadPlaylist(playlist)
  211. def loadPlaylist(self, playlist):
  212. # home page
  213. self.ui.sessionPlaylist.setEnabled(True)
  214. self.ui.frameNotes.setEnabled(True)
  215. self.ui.sessionLblTitle.setText(playlist.name)
  216. self.ui.sessionPlaylist.populate(playlist)
  217. self.ui.frameNotes.set_playlist(playlist)
  218. # explorer page
  219. self.ui.explorerPlaylist.setEnabled(True)
  220. self.ui.explorerLblPlaylistTitle.setText(playlist.name)
  221. self.ui.explorerPlaylist.populate(playlist)
  222. self.selected_playlist = playlist
  223. def add_to_playlist(self):
  224. track = self.ui.explorerTable.selected_track()
  225. if track is None:
  226. return
  227. playlist = self.ui.explorerPlaylist.playlist
  228. if playlist is None:
  229. return
  230. track_repo = TrackRepository()
  231. track_repo.add_to_session(
  232. track.id,
  233. playlist.id
  234. )
  235. self.ui.explorerPlaylist.populate(playlist)
  236. self.ui.sessionPlaylist.populate(playlist)
  237. def remove_from_playlist(self):
  238. session_track = self.ui.explorerPlaylist.selected_session_track()
  239. if session_track is None:
  240. return
  241. session_track_repo = SessionTrackRepository()
  242. session_track_repo.delete(session_track, True)
  243. self.ui.explorerPlaylist.populate()
  244. self.ui.sessionPlaylist.populate()
  245. def play_playlist(self, track=None):
  246. playlist = self.selected_playlist
  247. track_repo = TrackRepository()
  248. tracks = track_repo.get_by_session_id(playlist.id)
  249. start_at = tracks.index(track) if track else 0
  250. logger.info("Start playing playlist: %s from index %s" % (playlist, start_at))
  251. self.ui.vlcFrame.load_playlist(playlist, tracks, start_at)
  252. self.ui.vlcFrame.play()
  253. def vlcTrackStarted(self, track):
  254. if self.selected_playlist:
  255. self.ui.explorerPlaylist.set_is_playing(track)
  256. self.ui.sessionPlaylist.set_is_playing(track)
  257. else:
  258. self.ui.explorerPlaylist.set_is_playing(None)
  259. self.ui.sessionPlaylist.set_is_playing(None)
  260. def vlcPlaylistEnded(self):
  261. self.ui.explorerPlaylist.set_is_playing(None)
  262. self.ui.sessionPlaylist.set_is_playing(None)
  263. def addTrackTags(self, tag_id):
  264. track_id = self.selected_track.id
  265. track_tag_repo = TrackTagRepository()
  266. tt = TrackTag(track_id=track_id, tag_id=tag_id)
  267. track_tag_repo.create(tt)
  268. track_tag_repo.commit()
  269. def removeTrackTags(self, tag_id):
  270. track_id = self.selected_track.id
  271. track_tag_repo = TrackTagRepository()
  272. track_tag_repo.query().filter(TrackTag.track_id == track_id).filter(TrackTag.tag_id == tag_id).delete()
  273. track_tag_repo.commit()
  274. def saveTrackNotes(self):
  275. if not self.selected_track:
  276. return
  277. note = self.ui.explorerTrackNotepad.toHtml()
  278. track_repo = TrackRepository()
  279. self.selected_track.note = note
  280. track_repo.commit()
  281. print('track notes saved')
  282. def explorerSearchChanged(self):
  283. searchText = self.ui.explorerLineSearch.text()
  284. self.ui.explorerTable.filterBySearchText(searchText)
  285. def filesIndexed(self, tracks):
  286. self.statusBar().showMessage(f"{len(tracks)} fichiers indexés", 3000)
  287. self.refresh_explorer_tree(tracks)
  288. def musicFolderStatusChanged(self, music_folder_id):
  289. self.populate_music_folders_table()
  290. def currentSettings(self):
  291. volume = self.ui.vlcFrame.volume
  292. playlist_id = self.selected_playlist.id if self.selected_playlist else None
  293. muted = self.ui.vlcFrame.is_muted
  294. return {
  295. "playlist": playlist_id,
  296. "volume": volume,
  297. "muted": muted,
  298. }
  299. def __del__(self):
  300. if self.qdb.isOpen():
  301. self.qdb.close()