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"