window.py 14 KB

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