diff --git a/app/musicmuster.py b/app/musicmuster.py index e2eff86..3d751bc 100755 --- a/app/musicmuster.py +++ b/app/musicmuster.py @@ -3,6 +3,7 @@ from log import log from os.path import basename import argparse +import matplotlib # type: ignore import os import stackprinter # type: ignore import subprocess @@ -11,6 +12,8 @@ import threading from datetime import datetime, timedelta from pygame import mixer +from matplotlib.backends.backend_qtagg import FigureCanvasQTAgg # type: ignore +from matplotlib.figure import Figure # type: ignore from time import sleep from typing import ( Callable, @@ -250,6 +253,17 @@ class ImportTrack(QObject): self.finished.emit(self.playlist) +class MplCanvas(FigureCanvasQTAgg): + """ + From https://www.pythonguis.com/tutorials/plotting-matplotlib/ + """ + + def __init__(self, parent=None, width=5, height=4, dpi=100): + fig = Figure(figsize=(width, height), dpi=dpi) + # self.axes = fig.add_subplot(111) + super(MplCanvas, self).__init__(fig) + + class MusicMusterSignals(QObject): """ Class for all MusicMuster signals. See: @@ -289,6 +303,7 @@ class Window(QMainWindow, Ui_MainWindow): self.txtSearch.setHidden(True) self.hide_played_tracks = False mixer.init() + matplotlib.use('QtAgg') self.visible_playlist_tab: Callable[[], PlaylistTab] = \ self.tabPlaylist.currentWidget @@ -775,21 +790,6 @@ class Window(QMainWindow, Ui_MainWindow): self.label_end_timer.setText("00:00") self.label_fade_timer.setText("00:00") self.label_silent_timer.setText("00:00") - self.label_start_time.setText("00:00:00") - self.label_end_time.setText("00:00:00") - - if self.next_track.track_id: - self.label_track_length.setText( - helpers.ms_to_mmss(self.next_track.duration) - ) - if self.next_track.silence_at and self.next_track.fade_at: - self.label_fade_length.setText(helpers.ms_to_mmss( - self.next_track.silence_at - self.next_track.fade_at)) - else: - self.label_fade_length.setText("0:00") - else: - self.label_track_length.setText("0:00") - self.label_fade_length.setText("0:00") # Update headers self.update_headers() @@ -1249,6 +1249,11 @@ class Window(QMainWindow, Ui_MainWindow): break sleep(0.1) + # Show closing volume graph + sc = MplCanvas(self, width=2, height=2, dpi=100) + # sc.axes.plot([0, 1, 2, 3, 4], [10, 1, 20, 3, 40]) + self.horizontalLayout_graph.addWidget(sc) + # Tell database to record it as played Playdates(session, self.current_track.track_id) @@ -1265,21 +1270,6 @@ class Window(QMainWindow, Ui_MainWindow): # Update headers self.update_headers() - # Update clocks - self.label_track_length.setText( - helpers.ms_to_mmss(self.current_track.duration) - ) - self.label_fade_length.setText( - helpers.ms_to_mmss(self.current_track.fade_length)) - if self.current_track.start_time: - self.label_start_time.setText( - self.current_track.start_time.strftime( - Config.TRACK_TIME_FORMAT)) - if self.current_track.end_time: - self.label_end_time.setText( - self.current_track.end_time.strftime( - Config.TRACK_TIME_FORMAT)) - def preview(self) -> None: """ Preview selected or next track. We use a different mechanism to diff --git a/app/ui/main_window.ui b/app/ui/main_window.ui index 2812cba..7a8f26c 100644 --- a/app/ui/main_window.ui +++ b/app/ui/main_window.ui @@ -365,8 +365,14 @@ padding-left: 8px; - 321 - 0 + 152 + 112 + + + + + 184 + 16777215 @@ -376,28 +382,20 @@ padding-left: 8px; QFrame::Raised - - - - Fade length: - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - Start: - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - + + + + 132 + 36 + + + + + 164 + 16777215 + + Fade @@ -413,92 +411,14 @@ padding-left: 8px; - - - - - FreeSans - 16 - - - - 00:00:00 - - - false - - - - - - - - FreeSans - 16 - - - - 00:00:00 - - - false - - - - - - - Track length: - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - FreeSans - 16 - - - - 0:00 - - - false - - - - - - - End: - - - Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter - - - - - - - - FreeSans - 16 - - - - 0:00 - - - false - - - - + + + + 132 + 36 + + Preview @@ -520,6 +440,13 @@ padding-left: 8px; + + + + QLayout::SetDefaultConstraint + + + diff --git a/app/ui/main_window_ui.py b/app/ui/main_window_ui.py index ffab8ae..03c6614 100644 --- a/app/ui/main_window_ui.py +++ b/app/ui/main_window_ui.py @@ -188,76 +188,35 @@ class Ui_MainWindow(object): self.horizontalLayout = QtWidgets.QHBoxLayout() self.horizontalLayout.setObjectName("horizontalLayout") self.FadeStopInfoFrame = QtWidgets.QFrame(parent=self.InfoFooterFrame) - self.FadeStopInfoFrame.setMinimumSize(QtCore.QSize(321, 0)) + self.FadeStopInfoFrame.setMinimumSize(QtCore.QSize(152, 112)) + self.FadeStopInfoFrame.setMaximumSize(QtCore.QSize(184, 16777215)) self.FadeStopInfoFrame.setFrameShape(QtWidgets.QFrame.Shape.StyledPanel) self.FadeStopInfoFrame.setFrameShadow(QtWidgets.QFrame.Shadow.Raised) self.FadeStopInfoFrame.setObjectName("FadeStopInfoFrame") self.gridLayout = QtWidgets.QGridLayout(self.FadeStopInfoFrame) self.gridLayout.setObjectName("gridLayout") - self.label_7 = QtWidgets.QLabel(parent=self.FadeStopInfoFrame) - self.label_7.setAlignment(QtCore.Qt.AlignmentFlag.AlignRight|QtCore.Qt.AlignmentFlag.AlignTrailing|QtCore.Qt.AlignmentFlag.AlignVCenter) - self.label_7.setObjectName("label_7") - self.gridLayout.addWidget(self.label_7, 1, 0, 1, 1) - self.label_x_2 = QtWidgets.QLabel(parent=self.FadeStopInfoFrame) - self.label_x_2.setAlignment(QtCore.Qt.AlignmentFlag.AlignRight|QtCore.Qt.AlignmentFlag.AlignTrailing|QtCore.Qt.AlignmentFlag.AlignVCenter) - self.label_x_2.setObjectName("label_x_2") - self.gridLayout.addWidget(self.label_x_2, 0, 2, 1, 1) self.btnFade = QtWidgets.QPushButton(parent=self.FadeStopInfoFrame) + self.btnFade.setMinimumSize(QtCore.QSize(132, 36)) + self.btnFade.setMaximumSize(QtCore.QSize(164, 16777215)) icon1 = QtGui.QIcon() icon1.addPixmap(QtGui.QPixmap(":/icons/fade"), QtGui.QIcon.Mode.Normal, QtGui.QIcon.State.Off) self.btnFade.setIcon(icon1) self.btnFade.setIconSize(QtCore.QSize(30, 30)) self.btnFade.setObjectName("btnFade") - self.gridLayout.addWidget(self.btnFade, 2, 0, 1, 2) - self.label_start_time = QtWidgets.QLabel(parent=self.FadeStopInfoFrame) - font = QtGui.QFont() - font.setFamily("FreeSans") - font.setPointSize(16) - self.label_start_time.setFont(font) - self.label_start_time.setScaledContents(False) - self.label_start_time.setObjectName("label_start_time") - self.gridLayout.addWidget(self.label_start_time, 0, 3, 1, 1) - self.label_end_time = QtWidgets.QLabel(parent=self.FadeStopInfoFrame) - font = QtGui.QFont() - font.setFamily("FreeSans") - font.setPointSize(16) - self.label_end_time.setFont(font) - self.label_end_time.setScaledContents(False) - self.label_end_time.setObjectName("label_end_time") - self.gridLayout.addWidget(self.label_end_time, 1, 3, 1, 1) - self.label_x = QtWidgets.QLabel(parent=self.FadeStopInfoFrame) - self.label_x.setAlignment(QtCore.Qt.AlignmentFlag.AlignRight|QtCore.Qt.AlignmentFlag.AlignTrailing|QtCore.Qt.AlignmentFlag.AlignVCenter) - self.label_x.setObjectName("label_x") - self.gridLayout.addWidget(self.label_x, 0, 0, 1, 1) - self.label_fade_length = QtWidgets.QLabel(parent=self.FadeStopInfoFrame) - font = QtGui.QFont() - font.setFamily("FreeSans") - font.setPointSize(16) - self.label_fade_length.setFont(font) - self.label_fade_length.setScaledContents(False) - self.label_fade_length.setObjectName("label_fade_length") - self.gridLayout.addWidget(self.label_fade_length, 1, 1, 1, 1) - self.label_8 = QtWidgets.QLabel(parent=self.FadeStopInfoFrame) - self.label_8.setAlignment(QtCore.Qt.AlignmentFlag.AlignRight|QtCore.Qt.AlignmentFlag.AlignTrailing|QtCore.Qt.AlignmentFlag.AlignVCenter) - self.label_8.setObjectName("label_8") - self.gridLayout.addWidget(self.label_8, 1, 2, 1, 1) - self.label_track_length = QtWidgets.QLabel(parent=self.FadeStopInfoFrame) - font = QtGui.QFont() - font.setFamily("FreeSans") - font.setPointSize(16) - self.label_track_length.setFont(font) - self.label_track_length.setScaledContents(False) - self.label_track_length.setObjectName("label_track_length") - self.gridLayout.addWidget(self.label_track_length, 0, 1, 1, 1) + self.gridLayout.addWidget(self.btnFade, 0, 0, 1, 2) self.btnPreview = QtWidgets.QPushButton(parent=self.FadeStopInfoFrame) + self.btnPreview.setMinimumSize(QtCore.QSize(132, 36)) icon2 = QtGui.QIcon() icon2.addPixmap(QtGui.QPixmap(":/icons/headphones"), QtGui.QIcon.Mode.Normal, QtGui.QIcon.State.Off) self.btnPreview.setIcon(icon2) self.btnPreview.setIconSize(QtCore.QSize(30, 30)) self.btnPreview.setCheckable(True) self.btnPreview.setObjectName("btnPreview") - self.gridLayout.addWidget(self.btnPreview, 2, 2, 1, 2) + self.gridLayout.addWidget(self.btnPreview, 1, 0, 1, 2) self.horizontalLayout.addWidget(self.FadeStopInfoFrame) + self.horizontalLayout_graph = QtWidgets.QHBoxLayout() + self.horizontalLayout_graph.setObjectName("horizontalLayout_graph") + self.horizontalLayout.addLayout(self.horizontalLayout_graph) self.frame_elapsed = QtWidgets.QFrame(parent=self.InfoFooterFrame) self.frame_elapsed.setMinimumSize(QtCore.QSize(0, 112)) self.frame_elapsed.setStyleSheet("") @@ -574,15 +533,7 @@ class Ui_MainWindow(object): self.current_track_2.setText(_translate("MainWindow", "Current track:")) self.next_track_2.setText(_translate("MainWindow", "Next track:")) self.lblTOD.setText(_translate("MainWindow", "00:00:00")) - self.label_7.setText(_translate("MainWindow", "Fade length:")) - self.label_x_2.setText(_translate("MainWindow", "Start:")) self.btnFade.setText(_translate("MainWindow", " Fade")) - self.label_start_time.setText(_translate("MainWindow", "00:00:00")) - self.label_end_time.setText(_translate("MainWindow", "00:00:00")) - self.label_x.setText(_translate("MainWindow", "Track length:")) - self.label_fade_length.setText(_translate("MainWindow", "0:00")) - self.label_8.setText(_translate("MainWindow", "End:")) - self.label_track_length.setText(_translate("MainWindow", "0:00")) self.btnPreview.setText(_translate("MainWindow", " Preview")) self.label.setText(_translate("MainWindow", "Elapsed time")) self.label_elapsed_timer.setText(_translate("MainWindow", "00:00")) diff --git a/poetry.lock b/poetry.lock index a77ef19..8f4f27d 100644 --- a/poetry.lock +++ b/poetry.lock @@ -631,6 +631,44 @@ files = [ {file = "mysqlclient-2.1.1.tar.gz", hash = "sha256:828757e419fb11dd6c5ed2576ec92c3efaa93a0f7c39e263586d1ee779c3d782"}, ] +[[package]] +name = "numpy" +version = "1.24.3" +description = "Fundamental package for array computing in Python" +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "numpy-1.24.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:3c1104d3c036fb81ab923f507536daedc718d0ad5a8707c6061cdfd6d184e570"}, + {file = "numpy-1.24.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:202de8f38fc4a45a3eea4b63e2f376e5f2dc64ef0fa692838e31a808520efaf7"}, + {file = "numpy-1.24.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8535303847b89aa6b0f00aa1dc62867b5a32923e4d1681a35b5eef2d9591a463"}, + {file = "numpy-1.24.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2d926b52ba1367f9acb76b0df6ed21f0b16a1ad87c6720a1121674e5cf63e2b6"}, + {file = "numpy-1.24.3-cp310-cp310-win32.whl", hash = "sha256:f21c442fdd2805e91799fbe044a7b999b8571bb0ab0f7850d0cb9641a687092b"}, + {file = "numpy-1.24.3-cp310-cp310-win_amd64.whl", hash = "sha256:ab5f23af8c16022663a652d3b25dcdc272ac3f83c3af4c02eb8b824e6b3ab9d7"}, + {file = "numpy-1.24.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:9a7721ec204d3a237225db3e194c25268faf92e19338a35f3a224469cb6039a3"}, + {file = "numpy-1.24.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:d6cc757de514c00b24ae8cf5c876af2a7c3df189028d68c0cb4eaa9cd5afc2bf"}, + {file = "numpy-1.24.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:76e3f4e85fc5d4fd311f6e9b794d0c00e7002ec122be271f2019d63376f1d385"}, + {file = "numpy-1.24.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a1d3c026f57ceaad42f8231305d4653d5f05dc6332a730ae5c0bea3513de0950"}, + {file = "numpy-1.24.3-cp311-cp311-win32.whl", hash = "sha256:c91c4afd8abc3908e00a44b2672718905b8611503f7ff87390cc0ac3423fb096"}, + {file = "numpy-1.24.3-cp311-cp311-win_amd64.whl", hash = "sha256:5342cf6aad47943286afa6f1609cad9b4266a05e7f2ec408e2cf7aea7ff69d80"}, + {file = "numpy-1.24.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:7776ea65423ca6a15255ba1872d82d207bd1e09f6d0894ee4a64678dd2204078"}, + {file = "numpy-1.24.3-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:ae8d0be48d1b6ed82588934aaaa179875e7dc4f3d84da18d7eae6eb3f06c242c"}, + {file = "numpy-1.24.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ecde0f8adef7dfdec993fd54b0f78183051b6580f606111a6d789cd14c61ea0c"}, + {file = "numpy-1.24.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:4749e053a29364d3452c034827102ee100986903263e89884922ef01a0a6fd2f"}, + {file = "numpy-1.24.3-cp38-cp38-win32.whl", hash = "sha256:d933fabd8f6a319e8530d0de4fcc2e6a61917e0b0c271fded460032db42a0fe4"}, + {file = "numpy-1.24.3-cp38-cp38-win_amd64.whl", hash = "sha256:56e48aec79ae238f6e4395886b5eaed058abb7231fb3361ddd7bfdf4eed54289"}, + {file = "numpy-1.24.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:4719d5aefb5189f50887773699eaf94e7d1e02bf36c1a9d353d9f46703758ca4"}, + {file = "numpy-1.24.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:0ec87a7084caa559c36e0a2309e4ecb1baa03b687201d0a847c8b0ed476a7187"}, + {file = "numpy-1.24.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ea8282b9bcfe2b5e7d491d0bf7f3e2da29700cec05b49e64d6246923329f2b02"}, + {file = "numpy-1.24.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:210461d87fb02a84ef243cac5e814aad2b7f4be953b32cb53327bb49fd77fbb4"}, + {file = "numpy-1.24.3-cp39-cp39-win32.whl", hash = "sha256:784c6da1a07818491b0ffd63c6bbe5a33deaa0e25a20e1b3ea20cf0e43f8046c"}, + {file = "numpy-1.24.3-cp39-cp39-win_amd64.whl", hash = "sha256:d5036197ecae68d7f491fcdb4df90082b0d4960ca6599ba2659957aafced7c17"}, + {file = "numpy-1.24.3-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:352ee00c7f8387b44d19f4cada524586f07379c0d49270f87233983bc5087ca0"}, + {file = "numpy-1.24.3-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:1a7d6acc2e7524c9955e5c903160aa4ea083736fde7e91276b0e5d98e6332812"}, + {file = "numpy-1.24.3-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:35400e6a8d102fd07c71ed7dcadd9eb62ee9a6e84ec159bd48c28235bbb0f8e4"}, + {file = "numpy-1.24.3.tar.gz", hash = "sha256:ab344f1bf21f140adab8e47fdbc7c35a477dc01408791f8ba00d018dd0bc5155"}, +] + [[package]] name = "obsws-python" version = "1.4.2" @@ -1134,6 +1172,21 @@ files = [ {file = "PyQt6_WebEngine_Qt6-6.5.0-py3-none-win_amd64.whl", hash = "sha256:5acadcc6608df8d9eba385e04ced2fc88e7eb92e366556ee4ac3c57a02c00088"}, ] +[[package]] +name = "pyqtgraph" +version = "0.13.3" +description = "Scientific Graphics and GUI Library for Python" +category = "main" +optional = false +python-versions = ">=3.8" +files = [ + {file = "pyqtgraph-0.13.3-py3-none-any.whl", hash = "sha256:fdcc04ac4b32a7bedf1bf3cf74cbb93ab3ba5687791712bbfa8d0712377d2f2b"}, + {file = "pyqtgraph-0.13.3.tar.gz", hash = "sha256:58108d8411c7054e0841d8b791ee85e101fc296b9b359c0e01dde38a98ff2ace"}, +] + +[package.dependencies] +numpy = ">=1.20.0" + [[package]] name = "pyqtwebengine" version = "5.15.6" @@ -1606,4 +1659,4 @@ test = ["websockets"] [metadata] lock-version = "2.0" python-versions = "^3.9" -content-hash = "5b335197cebe3355472e3d0024a978e4853efce33802008987351b5196d8746d" +content-hash = "e195143fc7756994b10a3c4472a17f2288829ebd6a75ef75f6b683905cb8a543" diff --git a/pyproject.toml b/pyproject.toml index 043a455..673841f 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -26,6 +26,7 @@ obsws-python = "^1.4.2" pyqt6 = "^6.5.0" pyqt6-webengine = "^6.5.0" pygame = "^2.4.0" +pyqtgraph = "^0.13.3" [tool.poetry.dev-dependencies] ipdb = "^0.13.9"