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.

315 lines
11KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # A piano roll viewer/editor
  4. # Copyright (C) 2012-2015 Filipe Coelho <falktx@falktx.com>
  5. # Copyright (C) 2014-2015 Perry Nguyen
  6. #
  7. # This program is free software; you can redistribute it and/or
  8. # modify it under the terms of the GNU General Public License as
  9. # published by the Free Software Foundation; either version 2 of
  10. # the License, or any later version.
  11. #
  12. # This program is distributed in the hope that it will be useful,
  13. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  14. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  15. # GNU General Public License for more details.
  16. #
  17. # For a full copy of the GNU General Public License see the doc/GPL.txt file.
  18. # ------------------------------------------------------------------------------------------------------------
  19. # Imports (Config)
  20. from carla_config import *
  21. # ------------------------------------------------------------------------------------------------------------
  22. # Imports (Global)
  23. if config_UseQt5:
  24. from PyQt5.QtCore import pyqtSlot, Qt, QEvent
  25. from PyQt5.QtGui import QKeyEvent
  26. from PyQt5.QtWidgets import QMainWindow
  27. else:
  28. from PyQt4.QtCore import pyqtSlot, Qt, QEvent
  29. from PyQt4.QtGui import QKeyEvent, QMainWindow
  30. # ------------------------------------------------------------------------------------------------------------
  31. # Imports (Custom)
  32. from carla_shared import *
  33. from carla_utils import *
  34. from pianoroll import *
  35. import ui_midipattern
  36. # ------------------------------------------------------------------------------------------------------------
  37. # Imports (ExternalUI)
  38. from carla_app import CarlaApplication
  39. from externalui import ExternalUI
  40. # ------------------------------------------------------------------------------------------------------------
  41. class MidiPatternW(ExternalUI, QMainWindow):
  42. PPQ = 48.0
  43. def __init__(self):
  44. ExternalUI.__init__(self)
  45. QMainWindow.__init__(self)
  46. self.ui = ui_midipattern.Ui_MidiPatternW()
  47. self.ui.setupUi(self)
  48. self.ui.piano = self.ui.graphicsView.piano
  49. # to be filled with note-on events, while waiting for their matching note-off
  50. self.fPendingNoteOns = [] # (channel, note, velocity, time)
  51. self.fTransportInfo = {
  52. "playing": False,
  53. "frame": 0,
  54. "bar": 0,
  55. "beat": 0,
  56. "tick": 0,
  57. "bpm": 120.0,
  58. "sigNum": 4.0,
  59. "sigDenom": 4.0
  60. }
  61. self.ui.act_edit_insert.triggered.connect(self.slot_editInsertMode)
  62. self.ui.act_edit_velocity.triggered.connect(self.slot_editVelocityMode)
  63. self.ui.act_edit_select_all.triggered.connect(self.slot_editSelectAll)
  64. self.ui.piano.midievent.connect(self.sendMsg)
  65. self.ui.piano.measureupdate.connect(self.updateMeasureBox)
  66. self.ui.piano.modeupdate.connect(self.ui.modeIndicator.changeMode)
  67. self.ui.piano.modeupdate.connect(self.slot_modeChanged)
  68. self.ui.timeSigBox.currentIndexChanged[str].connect(self.ui.piano.setTimeSig)
  69. self.ui.measureBox.currentIndexChanged[str].connect(self.ui.piano.setMeasures)
  70. self.ui.defaultLengthBox.currentIndexChanged[str].connect(self.ui.piano.setDefaultLength)
  71. self.ui.quantizeBox.currentIndexChanged[str].connect(self.ui.piano.setGridDiv)
  72. self.ui.hSlider.valueChanged.connect(self.ui.graphicsView.setZoomX)
  73. self.ui.vSlider.valueChanged.connect(self.ui.graphicsView.setZoomY)
  74. self.ui.graphicsView.setFocus()
  75. self.fIdleTimer = self.startTimer(30)
  76. self.setWindowTitle(self.fUiName)
  77. self.ready()
  78. def slot_editInsertMode(self):
  79. ev = QKeyEvent(QEvent.User, Qt.Key_F, Qt.NoModifier)
  80. self.ui.piano.keyPressEvent(ev)
  81. def slot_editVelocityMode(self):
  82. ev = QKeyEvent(QEvent.User, Qt.Key_D, Qt.NoModifier)
  83. self.ui.piano.keyPressEvent(ev)
  84. def slot_editSelectAll(self):
  85. ev = QKeyEvent(QEvent.User, Qt.Key_A, Qt.NoModifier)
  86. self.ui.piano.keyPressEvent(ev)
  87. def slot_modeChanged(self, mode):
  88. if mode == "insert_mode":
  89. self.ui.act_edit_insert.setChecked(True)
  90. self.ui.act_edit_velocity.setChecked(False)
  91. elif mode == "velocity_mode":
  92. self.ui.act_edit_insert.setChecked(False)
  93. self.ui.act_edit_velocity.setChecked(True)
  94. else:
  95. self.ui.act_edit_insert.setChecked(False)
  96. self.ui.act_edit_velocity.setChecked(False)
  97. # -------------------------------------------------------------------
  98. # DSP Callbacks
  99. def dspParameterChanged(self, index, value):
  100. pass
  101. def dspStateChanged(self, key, value):
  102. pass
  103. # -------------------------------------------------------------------
  104. # ExternalUI Callbacks
  105. def uiShow(self):
  106. self.show()
  107. def uiFocus(self):
  108. self.setWindowState((self.windowState() & ~Qt.WindowMinimized) | Qt.WindowActive)
  109. self.show()
  110. self.raise_()
  111. self.activateWindow()
  112. def uiHide(self):
  113. self.hide()
  114. def uiQuit(self):
  115. self.closeExternalUI()
  116. self.close()
  117. app.quit()
  118. def uiTitleChanged(self, uiTitle):
  119. self.setWindowTitle(uiTitle)
  120. # -------------------------------------------------------------------
  121. # Qt events
  122. def timerEvent(self, event):
  123. if event.timerId() == self.fIdleTimer:
  124. self.idleExternalUI()
  125. QMainWindow.timerEvent(self, event)
  126. def closeEvent(self, event):
  127. self.closeExternalUI()
  128. QMainWindow.closeEvent(self, event)
  129. # there might be other qt windows open which will block the UI from quitting
  130. app.quit()
  131. # -------------------------------------------------------------------
  132. # Custom callback
  133. def updateMeasureBox(self, index):
  134. self.measureBox.setCurrentIndex(index-1)
  135. def sendMsg(self, data):
  136. msg = data[0]
  137. if msg == "midievent-remove":
  138. note, start, length, vel = data[1:5]
  139. note_start = start * 60. / self.fTransportInfo["bpm"] * 4. / self.fTransportInfo["sigDenom"] * self.PPQ
  140. note_stop = note_start + length * 60. / self.fTransportInfo["bpm"] * 4. * self.fTransportInfo["sigNum"] / self.fTransportInfo["sigDenom"] * self.PPQ
  141. self.send([msg, note_start, 3, MIDI_STATUS_NOTE_ON, note, vel])
  142. self.send([msg, note_stop, 3, MIDI_STATUS_NOTE_OFF, note, vel])
  143. elif msg == "midievent-add":
  144. note, start, length, vel = data[1:5]
  145. note_start = start * 60. / self.fTransportInfo["bpm"] * self.PPQ
  146. note_stop = note_start + length * 60. / self.fTransportInfo["bpm"] * 4. * self.fTransportInfo["sigNum"] / self.fTransportInfo["sigDenom"] * self.PPQ
  147. self.send([msg, note_start, 3, MIDI_STATUS_NOTE_ON, note, vel])
  148. self.send([msg, note_stop, 3, MIDI_STATUS_NOTE_OFF, note, vel])
  149. def msgCallback(self, msg):
  150. #try:
  151. self.msgCallback2(msg)
  152. #except:
  153. #print("Custom msgCallback error, skipped for", msg)
  154. def msgCallback2(self, msg):
  155. msg = charPtrToString(msg)
  156. if msg == "midi-clear-all":
  157. # clear all notes
  158. self.ui.piano.clearNotes()
  159. elif msg == "midievent-add":
  160. # adds single midi event
  161. time = int(self.readlineblock())
  162. size = int(self.readlineblock())
  163. data = []
  164. for x in range(size):
  165. data.append(int(self.readlineblock()))
  166. self.handleMidiEvent(time, size, data)
  167. elif msg == "transport":
  168. playing = bool(self.readlineblock() == "true")
  169. frame, bar, beat, tick = [int(i) for i in self.readlineblock().split(":")]
  170. bpm, sigNum, sigDenom = [float(i) for i in self.readlineblock().split(":")]
  171. if beat != self.fTransportInfo["beat"]:
  172. print(beat)
  173. old_frame = self.fTransportInfo['frame']
  174. self.fTransportInfo = {
  175. "playing": playing,
  176. "frame": frame,
  177. "bar": bar,
  178. "beat": beat,
  179. "tick": tick,
  180. "bpm": bpm,
  181. "sigNum": sigNum,
  182. "sigDenom": sigDenom
  183. }
  184. if old_frame != frame:
  185. self.ui.piano.movePlayHead(self.fTransportInfo)
  186. elif msg == "show":
  187. self.uiShow()
  188. elif msg == "focus":
  189. self.uiFocus()
  190. elif msg == "hide":
  191. self.uiHide()
  192. elif msg == "quit":
  193. self.fQuitReceived = True
  194. self.uiQuit()
  195. elif msg == "uiTitle":
  196. uiTitle = self.readlineblock()
  197. self.uiTitleChanged(uiTitle)
  198. else:
  199. print("unknown message: \"" + msg + "\"")
  200. # -------------------------------------------------------------------
  201. # Internal stuff
  202. def handleMidiEvent(self, time, size, data):
  203. #print("Got MIDI Event on UI", time, size, data)
  204. # NOTE: for now time comes in frames, which might not be desirable
  205. # we'll convert it to a smaller value for now (seconds)
  206. # later on we can have time as PPQ or similar
  207. time /= self.PPQ
  208. status = MIDI_GET_STATUS_FROM_DATA(data)
  209. channel = MIDI_GET_CHANNEL_FROM_DATA(data)
  210. if status == MIDI_STATUS_NOTE_ON:
  211. note = data[1]
  212. velo = data[2]
  213. # append (channel, note, velo, time) for later
  214. self.fPendingNoteOns.append((channel, note, velo, time))
  215. elif status == MIDI_STATUS_NOTE_OFF:
  216. note = data[1]
  217. velo = data[2]
  218. # find previous note-on that matches this note and channel
  219. for noteOnMsg in self.fPendingNoteOns:
  220. channel_, note_, velo_, time_ = noteOnMsg
  221. if channel_ != channel:
  222. continue
  223. if note_ != note:
  224. continue
  225. # found it
  226. #print("{} {} {} {}\n".format(note, time_, time-time_, velo_))
  227. start = time_ / 60. * self.fTransportInfo["bpm"] / 4. * self.fTransportInfo["sigDenom"]
  228. length = (time - time_) / 60. * self.fTransportInfo["bpm"] / 4. / self.fTransportInfo["sigNum"] * self.fTransportInfo["sigDenom"]
  229. self.ui.piano.drawNote(note, start, length, velo_)
  230. # remove from list
  231. self.fPendingNoteOns.remove(noteOnMsg)
  232. break
  233. #--------------- main ------------------
  234. if __name__ == '__main__':
  235. import resources_rc
  236. pathBinaries, pathResources = getPaths()
  237. gCarla.utils = CarlaUtils(os.path.join(pathBinaries, "libcarla_utils." + DLL_EXTENSION))
  238. gCarla.utils.set_process_name("MidiPattern")
  239. app = CarlaApplication("MidiPattern")
  240. gui = MidiPatternW()
  241. app.exit_exec()