Audio plugin host https://kx.studio/carla
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

486 lines
18KB

  1. #!/usr/bin/env python3
  2. # SPDX-FileCopyrightText: 2011-2024 Filipe Coelho <falktx@falktx.com>
  3. # SPDX-License-Identifier: GPL-2.0-or-later
  4. # ---------------------------------------------------------------------------------------------------------------------
  5. # Imports (Global)
  6. from math import sqrt
  7. from qt_compat import qt_config
  8. if qt_config == 5:
  9. from PyQt5.QtCore import qCritical, Qt, QSize, QLineF, QRectF
  10. from PyQt5.QtGui import QColor, QLinearGradient, QPainter, QPen, QPixmap
  11. from PyQt5.QtWidgets import QWidget
  12. elif qt_config == 6:
  13. from PyQt6.QtCore import qCritical, Qt, QSize, QLineF, QRectF
  14. from PyQt6.QtGui import QColor, QLinearGradient, QPainter, QPen, QPixmap
  15. from PyQt6.QtWidgets import QWidget
  16. # ---------------------------------------------------------------------------------------------------------------------
  17. # Widget Class
  18. class DigitalPeakMeter(QWidget):
  19. # enum Color
  20. COLOR_GREEN = 1
  21. COLOR_BLUE = 2
  22. # enum Orientation
  23. HORIZONTAL = 1
  24. VERTICAL = 2
  25. # enum Style
  26. STYLE_DEFAULT = 1
  27. STYLE_OPENAV = 2
  28. STYLE_RNCBC = 3
  29. STYLE_CALF = 4
  30. STYLE_TUBE = 5
  31. # -----------------------------------------------------------------------------------------------------------------
  32. def __init__(self, parent):
  33. QWidget.__init__(self, parent)
  34. self.setAttribute(Qt.WA_OpaquePaintEvent)
  35. # defaults are VERTICAL, COLOR_GREEN, STYLE_DEFAULT
  36. self.fChannelCount = 0
  37. self.fChannelData = []
  38. self.fLastChannelData = []
  39. self.fMeterColor = self.COLOR_GREEN
  40. self.fMeterColorBase = QColor(93, 231, 61)
  41. self.fMeterColorBaseAlt = QColor(15, 110, 15, 100)
  42. self.fMeterLinesEnabled = True
  43. self.fMeterOrientation = self.VERTICAL
  44. self.fMeterStyle = self.STYLE_DEFAULT
  45. self.fMeterBackground = QColor("#070707")
  46. self.fMeterGradient = QLinearGradient(0, 0, 0, 0)
  47. self.fMeterPixmaps = ()
  48. self.fSmoothMultiplier = 2
  49. self.updateGrandient()
  50. # -----------------------------------------------------------------------------------------------------------------
  51. def channelCount(self):
  52. return self.fChannelCount
  53. def setChannelCount(self, count):
  54. if self.fChannelCount == count:
  55. return
  56. if count < 0:
  57. qCritical(f"DigitalPeakMeter::setChannelCount({count}) - channel count must be a positive integer or zero")
  58. return
  59. self.fChannelCount = count
  60. self.fChannelData = []
  61. self.fLastChannelData = []
  62. for _ in range(count):
  63. self.fChannelData.append(0.0)
  64. self.fLastChannelData.append(0.0)
  65. if self.fMeterStyle == self.STYLE_CALF:
  66. if self.fChannelCount > 0:
  67. self.setFixedSize(100, 12*self.fChannelCount)
  68. else:
  69. self.setMinimumSize(0, 0)
  70. self.setMaximumSize(9999, 9999)
  71. # -----------------------------------------------------------------------------------------------------------------
  72. def meterColor(self):
  73. return self.fMeterColor
  74. def setMeterColor(self, color):
  75. if self.fMeterColor == color:
  76. return
  77. if color not in (self.COLOR_GREEN, self.COLOR_BLUE):
  78. qCritical(f"DigitalPeakMeter::setMeterColor({color}) - invalid color")
  79. return
  80. if color == self.COLOR_GREEN:
  81. self.fMeterColorBase = QColor(93, 231, 61)
  82. self.fMeterColorBaseAlt = QColor(15, 110, 15, 100)
  83. elif color == self.COLOR_BLUE:
  84. self.fMeterColorBase = QColor(82, 238, 248)
  85. self.fMeterColorBaseAlt = QColor(15, 15, 110, 100)
  86. self.fMeterColor = color
  87. self.updateGrandient()
  88. # -----------------------------------------------------------------------------------------------------------------
  89. def meterLinesEnabled(self):
  90. return self.fMeterLinesEnabled
  91. def setMeterLinesEnabled(self, yesNo):
  92. if self.fMeterLinesEnabled == yesNo:
  93. return
  94. self.fMeterLinesEnabled = yesNo
  95. # -----------------------------------------------------------------------------------------------------------------
  96. def meterOrientation(self):
  97. return self.fMeterOrientation
  98. def setMeterOrientation(self, orientation):
  99. if self.fMeterOrientation == orientation:
  100. return
  101. if orientation not in (self.HORIZONTAL, self.VERTICAL):
  102. qCritical(f"DigitalPeakMeter::setMeterOrientation({orientation}) - invalid orientation")
  103. return
  104. self.fMeterOrientation = orientation
  105. self.updateGrandient()
  106. # -----------------------------------------------------------------------------------------------------------------
  107. def meterStyle(self):
  108. return self.fMeterStyle
  109. def setMeterStyle(self, style):
  110. if self.fMeterStyle == style:
  111. return
  112. if style not in (self.STYLE_DEFAULT, self.STYLE_OPENAV, self.STYLE_RNCBC, self.STYLE_CALF, self.STYLE_TUBE):
  113. qCritical(f"DigitalPeakMeter::setMeterStyle({style}) - invalid style")
  114. return
  115. if style == self.STYLE_DEFAULT:
  116. self.fMeterBackground = QColor("#070707")
  117. elif style == self.STYLE_OPENAV:
  118. self.fMeterBackground = QColor("#1A1A1A")
  119. elif style == self.STYLE_RNCBC:
  120. self.fMeterBackground = QColor("#070707")
  121. elif style in (self.STYLE_CALF, self.STYLE_TUBE):
  122. self.fMeterBackground = QColor("#000")
  123. if style == self.STYLE_CALF:
  124. self.fMeterPixmaps = (QPixmap(":/bitmaps/meter_calf_off.png"), QPixmap(":/bitmaps/meter_calf_on.png"))
  125. if self.fChannelCount > 0:
  126. self.setFixedSize(100, 12*self.fChannelCount)
  127. else:
  128. self.fMeterPixmaps = ()
  129. self.setMinimumSize(0, 0)
  130. self.setMaximumSize(9999, 9999)
  131. self.fMeterStyle = style
  132. self.updateGrandient()
  133. # -----------------------------------------------------------------------------------------------------------------
  134. def smoothMultiplier(self):
  135. return self.fSmoothMultiplier
  136. def setSmoothMultiplier(self, value):
  137. if self.fSmoothMultiplier == value:
  138. return
  139. if not isinstance(value, int):
  140. qCritical("DigitalPeakMeter::setSmoothMultiplier() - value must be an integer")
  141. return
  142. if value < 0:
  143. qCritical(f"DigitalPeakMeter::setSmoothMultiplier({value}) - value must be >= 0")
  144. return
  145. if value > 5:
  146. qCritical(f"DigitalPeakMeter::setSmoothMultiplier({value}) - value must be < 5")
  147. return
  148. self.fSmoothMultiplier = value
  149. # -----------------------------------------------------------------------------------------------------------------
  150. def displayMeter(self, meter, level, forced = False):
  151. if not isinstance(meter, int):
  152. qCritical("DigitalPeakMeter::displayMeter(,) - meter value must be an integer")
  153. return
  154. if not isinstance(level, float):
  155. qCritical(f"DigitalPeakMeter::displayMeter({meter},) - level value must be a float")
  156. return
  157. if meter <= 0 or meter > self.fChannelCount:
  158. qCritical(f"DigitalPeakMeter::displayMeter({meter}, {level}) - invalid meter number")
  159. return
  160. i = meter - 1
  161. if level < 0.001:
  162. level = 0.0
  163. elif level > 0.999:
  164. level = 1.0
  165. if self.fSmoothMultiplier > 0 and not forced:
  166. level = (
  167. (self.fLastChannelData[i] * float(self.fSmoothMultiplier) + level)
  168. / float(self.fSmoothMultiplier + 1)
  169. )
  170. self.fLastChannelData[i] = level
  171. # Discretize scale: for 10 points, first will lit at 5%,
  172. # then 15%, and last at 95% of normalized value.
  173. # We also win some CPU when not redraw at small changes.
  174. if (self.fMeterStyle == self.STYLE_TUBE) and (level > 0.0) and (level < 1.0):
  175. points = 20
  176. # Transform to Sq Root domain: our meters have Sqrt dynamic compression;
  177. # Discretize:
  178. level = int(sqrt(level) * points + 0.5) / points
  179. # Transform back from Square Root domain:
  180. level = level * level
  181. if self.fChannelData[i] != level:
  182. self.fChannelData[i] = level
  183. self.update()
  184. # -----------------------------------------------------------------------------------------------------------------
  185. def updateGrandient(self):
  186. self.fMeterGradient = QLinearGradient(0, 0, 1, 1)
  187. if self.fMeterStyle == self.STYLE_DEFAULT:
  188. if self.fMeterOrientation == self.HORIZONTAL:
  189. self.fMeterGradient.setColorAt(0.0, self.fMeterColorBase)
  190. self.fMeterGradient.setColorAt(0.2, self.fMeterColorBase)
  191. self.fMeterGradient.setColorAt(0.4, self.fMeterColorBase)
  192. self.fMeterGradient.setColorAt(0.6, self.fMeterColorBase)
  193. self.fMeterGradient.setColorAt(0.8, Qt.yellow)
  194. self.fMeterGradient.setColorAt(1.0, Qt.red)
  195. elif self.fMeterOrientation == self.VERTICAL:
  196. self.fMeterGradient.setColorAt(0.0, Qt.red)
  197. self.fMeterGradient.setColorAt(0.2, Qt.yellow)
  198. self.fMeterGradient.setColorAt(0.4, self.fMeterColorBase)
  199. self.fMeterGradient.setColorAt(0.6, self.fMeterColorBase)
  200. self.fMeterGradient.setColorAt(0.8, self.fMeterColorBase)
  201. self.fMeterGradient.setColorAt(1.0, self.fMeterColorBase)
  202. elif self.fMeterStyle == self.STYLE_RNCBC:
  203. if self.fMeterColor == self.COLOR_BLUE:
  204. c1 = QColor(40,160,160)
  205. c2 = QColor(60,220,160)
  206. elif self.fMeterColor == self.COLOR_GREEN:
  207. c1 = QColor( 40,160,40)
  208. c2 = QColor(160,220,20)
  209. if self.fMeterOrientation == self.HORIZONTAL:
  210. self.fMeterGradient.setColorAt(0.0, c1)
  211. self.fMeterGradient.setColorAt(0.2, c1)
  212. self.fMeterGradient.setColorAt(0.6, c2)
  213. self.fMeterGradient.setColorAt(0.7, QColor(220,220, 20))
  214. self.fMeterGradient.setColorAt(0.8, QColor(240,160, 20))
  215. self.fMeterGradient.setColorAt(0.9, QColor(240, 0, 20))
  216. self.fMeterGradient.setColorAt(1.0, QColor(240, 0, 20))
  217. elif self.fMeterOrientation == self.VERTICAL:
  218. self.fMeterGradient.setColorAt(0.0, QColor(240, 0, 20))
  219. self.fMeterGradient.setColorAt(0.1, QColor(240, 0, 20))
  220. self.fMeterGradient.setColorAt(0.2, QColor(240,160, 20))
  221. self.fMeterGradient.setColorAt(0.3, QColor(220,220, 20))
  222. self.fMeterGradient.setColorAt(0.4, c2)
  223. self.fMeterGradient.setColorAt(0.8, c1)
  224. self.fMeterGradient.setColorAt(1.0, c1)
  225. elif self.fMeterStyle in (self.STYLE_OPENAV, self.STYLE_CALF):
  226. self.fMeterGradient.setColorAt(0.0, self.fMeterColorBase)
  227. self.fMeterGradient.setColorAt(1.0, self.fMeterColorBase)
  228. elif self.fMeterStyle == self.STYLE_TUBE:
  229. color = QColor.fromHslF(0.9, 1, 0.6, 1) # Tuneon filled w/ neon + agron
  230. points = 20
  231. for i in range(points + 1):
  232. self.fMeterGradient.setColorAt(((i-0.3)/points % 1.0), color)
  233. self.fMeterGradient.setColorAt(( i /points ), Qt.black)
  234. self.fMeterGradient.setColorAt(((i+0.3)/points % 1.0), color)
  235. self.updateGrandientFinalStop()
  236. def updateGrandientFinalStop(self):
  237. if self.fMeterOrientation == self.HORIZONTAL:
  238. self.fMeterGradient.setFinalStop(self.width(), 0)
  239. elif self.fMeterOrientation == self.VERTICAL:
  240. self.fMeterGradient.setFinalStop(0, self.height())
  241. # -----------------------------------------------------------------------------------------------------------------
  242. def minimumSizeHint(self):
  243. return QSize(20, 10) if self.fMeterOrientation == self.HORIZONTAL else QSize(10, 20)
  244. def sizeHint(self):
  245. return QSize(self.width(), self.height())
  246. # -----------------------------------------------------------------------------------------------------------------
  247. def drawCalf(self, event):
  248. painter = QPainter(self)
  249. event.accept()
  250. # no channels, draw black
  251. if self.fChannelCount == 0:
  252. painter.setPen(QPen(Qt.black, 2))
  253. painter.setBrush(Qt.black)
  254. painter.drawRect(0, 0, self.width(), self.height())
  255. return
  256. for i in range(self.fChannelCount):
  257. painter.drawPixmap(0, 12*i, self.fMeterPixmaps[0])
  258. meterPos = 4
  259. meterSize = 12
  260. # draw levels
  261. for level in self.fChannelData:
  262. if level != 0.0:
  263. blevel = int(sqrt(level)*26.0)*3
  264. painter.drawPixmap(5, meterPos, blevel, 4, self.fMeterPixmaps[1], 0, 0, blevel, 4)
  265. meterPos += meterSize
  266. def paintEvent(self, event):
  267. if self.fMeterStyle == self.STYLE_CALF:
  268. self.drawCalf(event)
  269. return
  270. painter = QPainter(self)
  271. event.accept()
  272. width = self.width()
  273. height = self.height()
  274. # draw background
  275. painter.setPen(QPen(self.fMeterBackground, 2))
  276. painter.setBrush(self.fMeterBackground)
  277. painter.drawRect(0, 0, width, height)
  278. if self.fChannelCount == 0:
  279. return
  280. meterPad = 0
  281. meterPos = 0
  282. meterSize = (height if self.fMeterOrientation == self.HORIZONTAL else width)/self.fChannelCount
  283. # set pen/brush for levels
  284. if self.fMeterStyle == self.STYLE_OPENAV:
  285. colorTrans = QColor(self.fMeterColorBase)
  286. colorTrans.setAlphaF(0.5)
  287. painter.setBrush(colorTrans)
  288. painter.setPen(QPen(self.fMeterColorBase, 1))
  289. del colorTrans
  290. meterPad += 2
  291. meterSize -= 2
  292. elif self.fMeterStyle == self.STYLE_TUBE:
  293. painter.setPen(QPen(Qt.NoPen))
  294. painter.setBrush(self.fMeterGradient)
  295. meterPos += 3
  296. meterPad += 6
  297. meterSize -= 6
  298. else:
  299. painter.setPen(QPen(self.fMeterBackground, 0))
  300. painter.setBrush(self.fMeterGradient)
  301. # draw levels
  302. for level in self.fChannelData:
  303. if level == 0.0:
  304. pass
  305. elif self.fMeterOrientation == self.HORIZONTAL:
  306. painter.drawRect(QRectF(0, meterPos, sqrt(level) * float(width), meterSize))
  307. elif self.fMeterOrientation == self.VERTICAL:
  308. painter.drawRect(QRectF(meterPos, height - sqrt(level) * float(height), meterSize, height))
  309. meterPos += meterSize+meterPad
  310. if not self.fMeterLinesEnabled:
  311. return
  312. # draw lines
  313. if self.fMeterOrientation == self.HORIZONTAL:
  314. # Variables
  315. lsmall = float(width)
  316. lfull = float(height - 1)
  317. if self.fMeterStyle == self.STYLE_OPENAV:
  318. painter.setPen(QColor(37, 37, 37, 100))
  319. painter.drawLine(QLineF(lsmall * 0.25, 2, lsmall * 0.25, lfull-2.0))
  320. painter.drawLine(QLineF(lsmall * 0.50, 2, lsmall * 0.50, lfull-2.0))
  321. painter.drawLine(QLineF(lsmall * 0.75, 2, lsmall * 0.75, lfull-2.0))
  322. if self.fChannelCount > 1:
  323. painter.drawLine(QLineF(1, lfull/2-1, lsmall-1, lfull/2-1))
  324. else:
  325. # Base
  326. painter.setBrush(Qt.black)
  327. painter.setPen(QPen(self.fMeterColorBaseAlt, 1))
  328. painter.drawLine(QLineF(lsmall * 0.25, 2, lsmall * 0.25, lfull-2.0))
  329. painter.drawLine(QLineF(lsmall * 0.50, 2, lsmall * 0.50, lfull-2.0))
  330. # Yellow
  331. painter.setPen(QColor(110, 110, 15, 100))
  332. painter.drawLine(QLineF(lsmall * 0.70, 2, lsmall * 0.70, lfull-2.0))
  333. painter.drawLine(QLineF(lsmall * 0.83, 2, lsmall * 0.83, lfull-2.0))
  334. # Orange
  335. painter.setPen(QColor(180, 110, 15, 100))
  336. painter.drawLine(QLineF(lsmall * 0.90, 2, lsmall * 0.90, lfull-2.0))
  337. # Red
  338. painter.setPen(QColor(110, 15, 15, 100))
  339. painter.drawLine(QLineF(lsmall * 0.96, 2, lsmall * 0.96, lfull-2.0))
  340. elif self.fMeterOrientation == self.VERTICAL:
  341. # Variables
  342. lsmall = float(height)
  343. lfull = float(width - 1)
  344. if self.fMeterStyle == self.STYLE_OPENAV:
  345. painter.setPen(QColor(37, 37, 37, 100))
  346. painter.drawLine(QLineF(2, lsmall - (lsmall * 0.25), lfull-2.0, lsmall - (lsmall * 0.25)))
  347. painter.drawLine(QLineF(2, lsmall - (lsmall * 0.50), lfull-2.0, lsmall - (lsmall * 0.50)))
  348. painter.drawLine(QLineF(2, lsmall - (lsmall * 0.75), lfull-2.0, lsmall - (lsmall * 0.75)))
  349. if self.fChannelCount > 1:
  350. painter.drawLine(QLineF(lfull/2-1, 1, lfull/2-1, lsmall-1))
  351. else:
  352. # Base
  353. painter.setBrush(Qt.black)
  354. painter.setPen(QPen(self.fMeterColorBaseAlt, 1))
  355. painter.drawLine(QLineF(2, lsmall - (lsmall * 0.25), lfull-2.0, lsmall - (lsmall * 0.25)))
  356. painter.drawLine(QLineF(2, lsmall - (lsmall * 0.50), lfull-2.0, lsmall - (lsmall * 0.50)))
  357. # Yellow
  358. painter.setPen(QColor(110, 110, 15, 100))
  359. painter.drawLine(QLineF(2, lsmall - (lsmall * 0.70), lfull-2.0, lsmall - (lsmall * 0.70)))
  360. painter.drawLine(QLineF(2, lsmall - (lsmall * 0.82), lfull-2.0, lsmall - (lsmall * 0.82)))
  361. # Orange
  362. painter.setPen(QColor(180, 110, 15, 100))
  363. painter.drawLine(QLineF(2, lsmall - (lsmall * 0.90), lfull-2.0, lsmall - (lsmall * 0.90)))
  364. # Red
  365. painter.setPen(QColor(110, 15, 15, 100))
  366. painter.drawLine(QLineF(2, lsmall - (lsmall * 0.96), lfull-2.0, lsmall - (lsmall * 0.96)))
  367. # -----------------------------------------------------------------------------------------------------------------
  368. def resizeEvent(self, event):
  369. QWidget.resizeEvent(self, event)
  370. self.updateGrandientFinalStop()
  371. # ---------------------------------------------------------------------------------------------------------------------