Collection of tools useful for audio production
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.

1446 lines
54KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # JACK Patchbay
  4. # Copyright (C) 2010-2018 Filipe Coelho <falktx@falktx.com>
  5. #
  6. # This program is free software; you can redistribute it and/or modify
  7. # it under the terms of the GNU General Public License as published by
  8. # the Free Software Foundation; either version 2 of the License, or
  9. # any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # For a full copy of the GNU General Public License see the COPYING file
  17. # ------------------------------------------------------------------------------------------------------------
  18. # Imports (Custom Stuff)
  19. import ui_catia
  20. from shared_canvasjack import *
  21. from shared_settings import *
  22. # ------------------------------------------------------------------------------------------------------------
  23. # Try Import DBus
  24. try:
  25. import dbus
  26. from dbus.mainloop.pyqt5 import DBusQtMainLoop
  27. haveDBus = True
  28. except:
  29. haveDBus = False
  30. # ------------------------------------------------------------------------------------------------------------
  31. # Try Import OpenGL
  32. try:
  33. from PyQt5.QtOpenGL import QGLWidget
  34. hasGL = True
  35. except:
  36. hasGL = False
  37. # ------------------------------------------------------------------------------------------------------------
  38. # Check for ALSA-MIDI
  39. haveALSA = False
  40. if LINUX:
  41. for iPATH in PATH:
  42. if not os.path.exists(os.path.join(iPATH, "aconnect")):
  43. continue
  44. haveALSA = True
  45. if sys.version_info >= (3, 0):
  46. from subprocess import getoutput
  47. else:
  48. from commands import getoutput
  49. if DEBUG:
  50. print("Using experimental ALSA-MIDI support")
  51. break
  52. # ------------------------------------------------------------------------------------------------------------
  53. # Global Variables
  54. global gA2JClientName
  55. gA2JClientName = None
  56. # ------------------------------------------------------------------------------------------------------------
  57. # Static Variables
  58. GROUP_TYPE_NULL = 0
  59. GROUP_TYPE_ALSA = 1
  60. GROUP_TYPE_JACK = 2
  61. iGroupId = 0
  62. iGroupName = 1
  63. iGroupType = 2
  64. iPortId = 0
  65. iPortName = 1
  66. iPortNameR = 2
  67. iPortGroupName = 3
  68. iConnId = 0
  69. iConnOutput = 1
  70. iConnInput = 2
  71. URI_CANVAS_ICON = "http://kxstudio.sf.net/ns/canvas/icon"
  72. # ------------------------------------------------------------------------------------------------------------
  73. # Catia Main Window
  74. class CatiaMainW(AbstractCanvasJackClass):
  75. def __init__(self, parent=None):
  76. AbstractCanvasJackClass.__init__(self, "Catia", ui_catia.Ui_CatiaMainW, parent)
  77. self.fGroupList = []
  78. self.fGroupSplitList = []
  79. self.fPortList = []
  80. self.fConnectionList = []
  81. self.fLastGroupId = 1
  82. self.fLastPortId = 1
  83. self.fLastConnectionId = 1
  84. self.loadSettings(True)
  85. # -------------------------------------------------------------
  86. # Set-up GUI
  87. setIcons(self, ["canvas", "jack", "transport", "misc"])
  88. self.ui.act_quit.setIcon(getIcon("application-exit"))
  89. self.ui.act_configure.setIcon(getIcon("configure"))
  90. self.ui.cb_buffer_size.clear()
  91. self.ui.cb_sample_rate.clear()
  92. for bufferSize in BUFFER_SIZE_LIST:
  93. self.ui.cb_buffer_size.addItem(str(bufferSize))
  94. for sampleRate in SAMPLE_RATE_LIST:
  95. self.ui.cb_sample_rate.addItem(str(sampleRate))
  96. self.ui.act_jack_bf_list = (self.ui.act_jack_bf_16, self.ui.act_jack_bf_32, self.ui.act_jack_bf_64, self.ui.act_jack_bf_128,
  97. self.ui.act_jack_bf_256, self.ui.act_jack_bf_512, self.ui.act_jack_bf_1024, self.ui.act_jack_bf_2048,
  98. self.ui.act_jack_bf_4096, self.ui.act_jack_bf_8192)
  99. if not haveALSA:
  100. self.ui.act_settings_show_alsa.setChecked(False)
  101. self.ui.act_settings_show_alsa.setEnabled(False)
  102. # -------------------------------------------------------------
  103. # Set-up Canvas
  104. self.scene = patchcanvas.PatchScene(self, self.ui.graphicsView)
  105. self.ui.graphicsView.setScene(self.scene)
  106. self.ui.graphicsView.setRenderHint(QPainter.Antialiasing, bool(self.fSavedSettings["Canvas/Antialiasing"] == patchcanvas.ANTIALIASING_FULL))
  107. if self.fSavedSettings["Canvas/UseOpenGL"] and hasGL:
  108. self.ui.graphicsView.setViewport(QGLWidget(self.ui.graphicsView))
  109. self.ui.graphicsView.setRenderHint(QPainter.HighQualityAntialiasing, self.fSavedSettings["Canvas/HighQualityAntialiasing"])
  110. pOptions = patchcanvas.options_t()
  111. pOptions.theme_name = self.fSavedSettings["Canvas/Theme"]
  112. pOptions.auto_hide_groups = self.fSavedSettings["Canvas/AutoHideGroups"]
  113. pOptions.use_bezier_lines = self.fSavedSettings["Canvas/UseBezierLines"]
  114. pOptions.antialiasing = self.fSavedSettings["Canvas/Antialiasing"]
  115. pOptions.eyecandy = self.fSavedSettings["Canvas/EyeCandy"]
  116. pFeatures = patchcanvas.features_t()
  117. pFeatures.group_info = False
  118. pFeatures.group_rename = False
  119. pFeatures.port_info = True
  120. pFeatures.port_rename = bool(self.fSavedSettings["Main/JackPortAlias"] > 0)
  121. pFeatures.handle_group_pos = True
  122. patchcanvas.setOptions(pOptions)
  123. patchcanvas.setFeatures(pFeatures)
  124. patchcanvas.init("Catia", self.scene, self.canvasCallback, DEBUG)
  125. # -------------------------------------------------------------
  126. # Try to connect to jack
  127. if self.jackStarted():
  128. self.initAlsaPorts()
  129. # -------------------------------------------------------------
  130. # Check DBus
  131. if haveDBus:
  132. if gDBus.jack:
  133. pass
  134. else:
  135. self.ui.act_tools_jack_start.setEnabled(False)
  136. self.ui.act_tools_jack_stop.setEnabled(False)
  137. self.ui.act_jack_configure.setEnabled(False)
  138. self.ui.b_jack_configure.setEnabled(False)
  139. if gDBus.a2j:
  140. if gDBus.a2j.is_started():
  141. self.a2jStarted()
  142. else:
  143. self.a2jStopped()
  144. else:
  145. self.ui.act_tools_a2j_start.setEnabled(False)
  146. self.ui.act_tools_a2j_stop.setEnabled(False)
  147. self.ui.act_tools_a2j_export_hw.setEnabled(False)
  148. self.ui.menu_A2J_Bridge.setEnabled(False)
  149. else:
  150. # No DBus
  151. self.ui.act_tools_jack_start.setEnabled(False)
  152. self.ui.act_tools_jack_stop.setEnabled(False)
  153. self.ui.act_jack_configure.setEnabled(False)
  154. self.ui.b_jack_configure.setEnabled(False)
  155. self.ui.act_tools_a2j_start.setEnabled(False)
  156. self.ui.act_tools_a2j_stop.setEnabled(False)
  157. self.ui.act_tools_a2j_export_hw.setEnabled(False)
  158. self.ui.menu_A2J_Bridge.setEnabled(False)
  159. # -------------------------------------------------------------
  160. # Set-up Timers
  161. self.fTimer120 = self.startTimer(self.fSavedSettings["Main/RefreshInterval"])
  162. self.fTimer600 = self.startTimer(self.fSavedSettings["Main/RefreshInterval"] * 5)
  163. # -------------------------------------------------------------
  164. # Set-up Connections
  165. self.setCanvasConnections()
  166. self.setJackConnections(["jack", "buffer-size", "transport", "misc"])
  167. self.ui.act_tools_jack_start.triggered.connect(self.slot_JackServerStart)
  168. self.ui.act_tools_jack_stop.triggered.connect(self.slot_JackServerStop)
  169. self.ui.act_tools_a2j_start.triggered.connect(self.slot_A2JBridgeStart)
  170. self.ui.act_tools_a2j_stop.triggered.connect(self.slot_A2JBridgeStop)
  171. self.ui.act_tools_a2j_export_hw.triggered.connect(self.slot_A2JBridgeExportHW)
  172. self.ui.act_settings_show_alsa.triggered.connect(self.slot_showAlsaMIDI)
  173. self.ui.act_configure.triggered.connect(self.slot_configureCatia)
  174. self.ui.act_help_about.triggered.connect(self.slot_aboutCatia)
  175. self.ui.act_help_about_qt.triggered.connect(app.aboutQt)
  176. self.XRunCallback.connect(self.slot_XRunCallback)
  177. self.BufferSizeCallback.connect(self.slot_BufferSizeCallback)
  178. self.SampleRateCallback.connect(self.slot_SampleRateCallback)
  179. self.ClientRenameCallback.connect(self.slot_ClientRenameCallback)
  180. self.PortRegistrationCallback.connect(self.slot_PortRegistrationCallback)
  181. self.PortConnectCallback.connect(self.slot_PortConnectCallback)
  182. self.PortRenameCallback.connect(self.slot_PortRenameCallback)
  183. self.ShutdownCallback.connect(self.slot_ShutdownCallback)
  184. # -------------------------------------------------------------
  185. # Set-up DBus
  186. if gDBus.jack or gDBus.a2j:
  187. gDBus.bus.add_signal_receiver(self.DBusSignalReceiver, destination_keyword="dest", path_keyword="path",
  188. member_keyword="member", interface_keyword="interface", sender_keyword="sender")
  189. # -------------------------------------------------------------
  190. def canvasCallback(self, action, value1, value2, valueStr):
  191. if action == patchcanvas.ACTION_GROUP_INFO:
  192. pass
  193. elif action == patchcanvas.ACTION_GROUP_RENAME:
  194. pass
  195. elif action == patchcanvas.ACTION_GROUP_SPLIT:
  196. groupId = value1
  197. patchcanvas.splitGroup(groupId)
  198. elif action == patchcanvas.ACTION_GROUP_JOIN:
  199. groupId = value1
  200. patchcanvas.joinGroup(groupId)
  201. elif action == patchcanvas.ACTION_PORT_INFO:
  202. portId = value1
  203. for port in self.fPortList:
  204. if port[iPortId] == portId:
  205. portNameR = port[iPortNameR]
  206. portNameG = port[iPortGroupName]
  207. break
  208. else:
  209. return
  210. if portNameR.startswith("[ALSA-"):
  211. portId, portName = portNameR.split("] ", 1)[1].split(" ", 1)
  212. flags = []
  213. if portNameR.startswith("[ALSA-Input] "):
  214. flags.append(self.tr("Input"))
  215. elif portNameR.startswith("[ALSA-Output] "):
  216. flags.append(self.tr("Output"))
  217. flagsText = " | ".join(flags)
  218. typeText = self.tr("ALSA MIDI")
  219. info = self.tr(""
  220. "<table>"
  221. "<tr><td align='right'><b>Group Name:</b></td><td>&nbsp;%s</td></tr>"
  222. "<tr><td align='right'><b>Port Id:</b></td><td>&nbsp;%s</td></tr>"
  223. "<tr><td align='right'><b>Port Name:</b></td><td>&nbsp;%s</td></tr>"
  224. "<tr><td colspan='2'>&nbsp;</td></tr>"
  225. "<tr><td align='right'><b>Port Flags:</b></td><td>&nbsp;%s</td></tr>"
  226. "<tr><td align='right'><b>Port Type:</b></td><td>&nbsp;%s</td></tr>"
  227. "</table>" % (portNameG, portId, portName, flagsText, typeText))
  228. else:
  229. portPtr = jacklib.port_by_name(gJack.client, portNameR)
  230. portFlags = jacklib.port_flags(portPtr)
  231. groupName = portNameR.split(":", 1)[0]
  232. portShortName = str(jacklib.port_short_name(portPtr), encoding="utf-8")
  233. aliases = jacklib.port_get_aliases(portPtr)
  234. if aliases[0] == 1:
  235. alias1text = aliases[1]
  236. alias2text = "(none)"
  237. elif aliases[0] == 2:
  238. alias1text = aliases[1]
  239. alias2text = aliases[2]
  240. else:
  241. alias1text = "(none)"
  242. alias2text = "(none)"
  243. flags = []
  244. if portFlags & jacklib.JackPortIsInput:
  245. flags.append(self.tr("Input"))
  246. if portFlags & jacklib.JackPortIsOutput:
  247. flags.append(self.tr("Output"))
  248. if portFlags & jacklib.JackPortIsPhysical:
  249. flags.append(self.tr("Physical"))
  250. if portFlags & jacklib.JackPortCanMonitor:
  251. flags.append(self.tr("Can Monitor"))
  252. if portFlags & jacklib.JackPortIsTerminal:
  253. flags.append(self.tr("Terminal"))
  254. flagsText = " | ".join(flags)
  255. portTypeStr = str(jacklib.port_type(portPtr), encoding="utf-8")
  256. if portTypeStr == jacklib.JACK_DEFAULT_AUDIO_TYPE:
  257. typeText = self.tr("JACK Audio")
  258. elif portTypeStr == jacklib.JACK_DEFAULT_MIDI_TYPE:
  259. typeText = self.tr("JACK MIDI")
  260. else:
  261. typeText = self.tr("Unknown")
  262. portLatency = jacklib.port_get_latency(portPtr)
  263. portTotalLatency = jacklib.port_get_total_latency(gJack.client, portPtr)
  264. latencyText = self.tr("%.1f ms (%i frames)" % (portLatency * 1000 / int(self.fSampleRate), portLatency))
  265. latencyTotalText = self.tr("%.1f ms (%i frames)" % (portTotalLatency * 1000 / int(self.fSampleRate), portTotalLatency))
  266. info = self.tr(""
  267. "<table>"
  268. "<tr><td align='right'><b>Group Name:</b></td><td>&nbsp;%s</td></tr>"
  269. "<tr><td align='right'><b>Port Name:</b></td><td>&nbsp;%s</td></tr>"
  270. "<tr><td align='right'><b>Full Port Name:</b></td><td>&nbsp;%s</td></tr>"
  271. "<tr><td align='right'><b>Port Alias #1:</b></td><td>&nbsp;%s</td></tr>"
  272. "<tr><td align='right'><b>Port Alias #2:</b></td><td>&nbsp;%s</td></tr>"
  273. "<tr><td colspan='2'>&nbsp;</td></tr>"
  274. "<tr><td align='right'><b>Port Flags:</b></td><td>&nbsp;%s</td></tr>"
  275. "<tr><td align='right'><b>Port Type:</b></td><td>&nbsp;%s</td></tr>"
  276. "<tr><td colspan='2'>&nbsp;</td></tr>"
  277. "<tr><td align='right'><b>Port Latency:</b></td><td>&nbsp;%s</td></tr>"
  278. "<tr><td align='right'><b>Total Port Latency:</b></td><td>&nbsp;%s</td></tr>"
  279. "</table>" % (groupName, portShortName, portNameR, alias1text, alias2text, flagsText, typeText, latencyText, latencyTotalText))
  280. QMessageBox.information(self, self.tr("Port Information"), info)
  281. elif action == patchcanvas.ACTION_PORT_RENAME:
  282. global gA2JClientName
  283. portId = value1
  284. portShortName = asciiString(valueStr)
  285. for port in self.fPortList:
  286. if port[iPortId] == portId:
  287. portNameR = port[iPortNameR]
  288. if portNameR.startswith("[ALSA-"):
  289. QMessageBox.warning(self, self.tr("Cannot continue"), self.tr(""
  290. "Rename functions rely on JACK aliases and cannot be done in ALSA ports"))
  291. return
  292. if portNameR.split(":", 1)[0] == gA2JClientName:
  293. a2jSplit = portNameR.split(":", 3)
  294. portName = "%s:%s: %s" % (a2jSplit[0], a2jSplit[1], portShortName)
  295. else:
  296. portName = "%s:%s" % (port[iPortGroupName], portShortName)
  297. break
  298. else:
  299. return
  300. portPtr = jacklib.port_by_name(gJack.client, portNameR)
  301. aliases = jacklib.port_get_aliases(portPtr)
  302. if aliases[0] == 2:
  303. # JACK only allows 2 aliases, remove 2nd
  304. jacklib.port_unset_alias(portPtr, aliases[2])
  305. # If we're going for 1st alias, unset it too
  306. if self.fSavedSettings["Main/JackPortAlias"] == 1:
  307. jacklib.port_unset_alias(portPtr, aliases[1])
  308. elif aliases[0] == 1 and self.fSavedSettings["Main/JackPortAlias"] == 1:
  309. jacklib.port_unset_alias(portPtr, aliases[1])
  310. if aliases[0] == 0 and self.fSavedSettings["Main/JackPortAlias"] == 2:
  311. # If 2nd alias is enabled and port had no previous aliases, set the 1st alias now
  312. jacklib.port_set_alias(portPtr, portName)
  313. if jacklib.port_set_alias(portPtr, portName) == 0:
  314. patchcanvas.renamePort(portId, portShortName)
  315. elif action == patchcanvas.ACTION_PORTS_CONNECT:
  316. portIdA = value1
  317. portIdB = value2
  318. portRealNameA = ""
  319. portRealNameB = ""
  320. for port in self.fPortList:
  321. if port[iPortId] == portIdA:
  322. portRealNameA = port[iPortNameR]
  323. if port[iPortId] == portIdB:
  324. portRealNameB = port[iPortNameR]
  325. if portRealNameA.startswith("[ALSA-"):
  326. portIdAlsaA = portRealNameA.split(" ", 2)[1]
  327. portIdAlsaB = portRealNameB.split(" ", 2)[1]
  328. if os.system("aconnect %s %s" % (portIdAlsaA, portIdAlsaB)) == 0:
  329. self.canvas_connectPorts(portIdA, portIdB)
  330. elif portRealNameA and portRealNameB:
  331. jacklib.connect(gJack.client, portRealNameA, portRealNameB)
  332. elif action == patchcanvas.ACTION_PORTS_DISCONNECT:
  333. connectionId = value1
  334. for connection in self.fConnectionList:
  335. if connection[iConnId] == connectionId:
  336. portIdA = connection[iConnOutput]
  337. portIdB = connection[iConnInput]
  338. break
  339. else:
  340. return
  341. portRealNameA = ""
  342. portRealNameB = ""
  343. for port in self.fPortList:
  344. if port[iPortId] == portIdA:
  345. portRealNameA = port[iPortNameR]
  346. if port[iPortId] == portIdB:
  347. portRealNameB = port[iPortNameR]
  348. if portRealNameA.startswith("[ALSA-"):
  349. portIdAlsaA = portRealNameA.split(" ", 2)[1]
  350. portIdAlsaB = portRealNameB.split(" ", 2)[1]
  351. if os.system("aconnect -d %s %s" % (portIdAlsaA, portIdAlsaB)) == 0:
  352. self.canvas_disconnectPorts(portIdA, portIdB)
  353. elif portRealNameA and portRealNameB:
  354. jacklib.disconnect(gJack.client, portRealNameA, portRealNameB)
  355. def initPorts(self):
  356. self.fGroupList = []
  357. self.fGroupSplitList = []
  358. self.fPortList = []
  359. self.fConnectionList = []
  360. self.fLastGroupId = 1
  361. self.fLastPortId = 1
  362. self.fLastConnectionId = 1
  363. self.initJackPorts()
  364. self.initAlsaPorts()
  365. def initJack(self):
  366. self.fXruns = 0
  367. self.fNextSampleRate = 0.0
  368. self.fLastBPM = None
  369. self.fLastTransportState = None
  370. bufferSize = int(jacklib.get_buffer_size(gJack.client))
  371. sampleRate = int(jacklib.get_sample_rate(gJack.client))
  372. realtime = bool(int(jacklib.is_realtime(gJack.client)))
  373. self.ui_setBufferSize(bufferSize)
  374. self.ui_setSampleRate(sampleRate)
  375. self.ui_setRealTime(realtime)
  376. self.ui_setXruns(0)
  377. self.refreshDSPLoad()
  378. self.refreshTransport()
  379. self.initJackCallbacks()
  380. self.initJackPorts()
  381. self.scene.zoom_fit()
  382. self.scene.zoom_reset()
  383. jacklib.activate(gJack.client)
  384. def initJackCallbacks(self):
  385. jacklib.set_buffer_size_callback(gJack.client, self.JackBufferSizeCallback, None)
  386. jacklib.set_sample_rate_callback(gJack.client, self.JackSampleRateCallback, None)
  387. jacklib.set_xrun_callback(gJack.client, self.JackXRunCallback, None)
  388. jacklib.set_port_registration_callback(gJack.client, self.JackPortRegistrationCallback, None)
  389. jacklib.set_port_connect_callback(gJack.client, self.JackPortConnectCallback, None)
  390. jacklib.set_session_callback(gJack.client, self.JackSessionCallback, None)
  391. jacklib.on_shutdown(gJack.client, self.JackShutdownCallback, None)
  392. jacklib.set_client_rename_callback(gJack.client, self.JackClientRenameCallback, None)
  393. jacklib.set_port_rename_callback(gJack.client, self.JackPortRenameCallback, None)
  394. def initJackPorts(self):
  395. if not gJack.client:
  396. return
  397. global gA2JClientName
  398. # Get all jack ports, put a2j ones to the bottom of the list
  399. a2jNameList = []
  400. portNameList = c_char_p_p_to_list(jacklib.get_ports(gJack.client, "", "", 0))
  401. h = 0
  402. for i in range(len(portNameList)):
  403. if portNameList[i - h].split(":")[0] == gA2JClientName:
  404. portName = portNameList.pop(i - h)
  405. a2jNameList.append(portName)
  406. h += 1
  407. for a2jName in a2jNameList:
  408. portNameList.append(a2jName)
  409. del a2jNameList
  410. # Add jack ports
  411. for portName in portNameList:
  412. portPtr = jacklib.port_by_name(gJack.client, portName)
  413. self.canvas_addJackPort(portPtr, portName)
  414. # Add jack connections
  415. for portName in portNameList:
  416. portPtr = jacklib.port_by_name(gJack.client, portName)
  417. # Only make connections from an output port
  418. if jacklib.port_flags(portPtr) & jacklib.JackPortIsInput:
  419. continue
  420. portConnectionNames = c_char_p_p_to_list(jacklib.port_get_all_connections(gJack.client, portPtr))
  421. for portConName in portConnectionNames:
  422. self.canvas_connectPortsByName(portName, portConName)
  423. def initAlsaPorts(self):
  424. if not (haveALSA and self.ui.act_settings_show_alsa.isChecked()):
  425. return
  426. # Get ALSA MIDI ports (outputs)
  427. output = getoutput("env LANG=C LC_ALL=C aconnect -i").split("\n")
  428. lastGroupId = -1
  429. lastGroupName = ""
  430. for line in output:
  431. # Make 'System' match JACK's 'system'
  432. if line == "client 0: 'System' [type=kernel]":
  433. line = "client 0: 'system' [type=kernel]"
  434. if line.startswith("client "):
  435. lineSplit = line.split(": ", 1)
  436. lineSplit2 = lineSplit[1].replace("'", "", 1).split("' [type=", 1)
  437. groupId = int(lineSplit[0].replace("client ", ""))
  438. groupName = lineSplit2[0]
  439. groupType = lineSplit2[1].rsplit("]", 1)[0]
  440. lastGroupId = self.canvas_getGroupId(groupName)
  441. if lastGroupId == -1:
  442. # Group doesn't exist yet
  443. lastGroupId = self.canvas_addAlsaGroup(groupId, groupName, bool(groupType == "kernel"))
  444. lastGroupName = groupName
  445. elif line.startswith(" ") and lastGroupId >= 0 and lastGroupName:
  446. lineSplit = line.split(" '", 1)
  447. portId = int(lineSplit[0].strip())
  448. portName = lineSplit[1].rsplit("'", 1)[0].strip()
  449. self.canvas_addAlsaPort(lastGroupId, lastGroupName, portName, "%i:%i %s" % (groupId, portId, portName), False)
  450. else:
  451. lastGroupId = -1
  452. lastGroupName = ""
  453. # Get ALSA MIDI ports (inputs)
  454. output = getoutput("env LANG=C LC_ALL=C aconnect -o").split("\n")
  455. lastGroupId = -1
  456. lastGroupName = ""
  457. for line in output:
  458. # Make 'System' match JACK's 'system'
  459. if line == "client 0: 'System' [type=kernel]":
  460. line = "client 0: 'system' [type=kernel]"
  461. if line.startswith("client "):
  462. lineSplit = line.split(": ", 1)
  463. lineSplit2 = lineSplit[1].replace("'", "", 1).split("' [type=", 1)
  464. groupId = int(lineSplit[0].replace("client ", ""))
  465. groupName = lineSplit2[0]
  466. groupType = lineSplit2[1].rsplit("]", 1)[0]
  467. lastGroupId = self.canvas_getGroupId(groupName)
  468. if lastGroupId == -1:
  469. # Group doesn't exist yet
  470. lastGroupId = self.canvas_addAlsaGroup(groupId, groupName, bool(groupType == "kernel"))
  471. lastGroupName = groupName
  472. elif line.startswith(" ") and lastGroupId >= 0 and lastGroupName:
  473. lineSplit = line.split(" '", 1)
  474. portId = int(lineSplit[0].strip())
  475. portName = lineSplit[1].rsplit("'", 1)[0].strip()
  476. self.canvas_addAlsaPort(lastGroupId, lastGroupName, portName, "%i:%i %s" % (groupId, portId, portName), True)
  477. else:
  478. lastGroupId = -1
  479. lastGroupName = ""
  480. # Get ALSA MIDI connections
  481. output = getoutput("env LANG=C LC_ALL=C aconnect -ol").split("\n")
  482. lastGroupId = -1
  483. lastPortId = -1
  484. for line in output:
  485. # Make 'System' match JACK's 'system'
  486. if line == "client 0: 'System' [type=kernel]":
  487. line = "client 0: 'system' [type=kernel]"
  488. if line.startswith("client "):
  489. lineSplit = line.split(": ", 1)
  490. lineSplit2 = lineSplit[1].replace("'", "", 1).split("' [type=", 1)
  491. groupId = int(lineSplit[0].replace("client ", ""))
  492. groupName = lineSplit2[0]
  493. lastGroupId = self.canvas_getGroupId(groupName)
  494. elif line.startswith(" ") and lastGroupId >= 0:
  495. lineSplit = line.split(" '", 1)
  496. portId = int(lineSplit[0].strip())
  497. portName = lineSplit[1].rsplit("'", 1)[0].strip()
  498. for port in self.fPortList:
  499. if port[iPortNameR] == "[ALSA-Input] %i:%i %s" % (groupId, portId, portName):
  500. lastPortId = port[iPortId]
  501. break
  502. else:
  503. lastPortId = -1
  504. elif line.startswith("\tConnect") and lastGroupId >= 0 and lastPortId >= 0:
  505. if line.startswith("\tConnected From"):
  506. lineSplit = line.split(": ", 1)[1]
  507. lineConns = lineSplit.split(", ")
  508. for lineConn in lineConns:
  509. lineConnSplit = lineConn.replace("'","").split(":", 1)
  510. alsaGroupId = int(lineConnSplit[0].split("[real:",1)[0])
  511. alsaPortId = int(lineConnSplit[1].split("[real:",1)[0])
  512. portNameRtest = "[ALSA-Output] %i:%i " % (alsaGroupId, alsaPortId)
  513. for port in self.fPortList:
  514. if port[iPortNameR].startswith(portNameRtest):
  515. self.canvas_connectPorts(port[iPortId], lastPortId)
  516. break
  517. else:
  518. lastGroupId = -1
  519. lastPortId = -1
  520. def canvas_getGroupId(self, groupName):
  521. for group in self.fGroupList:
  522. if group[iGroupName] == groupName:
  523. return group[iGroupId]
  524. return -1
  525. def canvas_addAlsaGroup(self, alsaGroupId, groupName, hwSplit):
  526. groupId = self.fLastGroupId
  527. if hwSplit:
  528. patchcanvas.addGroup(groupId, groupName, patchcanvas.SPLIT_YES, patchcanvas.ICON_HARDWARE)
  529. else:
  530. patchcanvas.addGroup(groupId, groupName)
  531. groupObj = [None, None, None]
  532. groupObj[iGroupId] = groupId
  533. groupObj[iGroupName] = groupName
  534. groupObj[iGroupType] = GROUP_TYPE_ALSA
  535. self.fGroupList.append(groupObj)
  536. self.fLastGroupId += 1
  537. return groupId
  538. def canvas_addJackGroup(self, groupName):
  539. ret, data, dataSize = jacklib.custom_get_data(gJack.client, groupName, URI_CANVAS_ICON)
  540. groupId = self.fLastGroupId
  541. groupSplit = patchcanvas.SPLIT_UNDEF
  542. groupIcon = patchcanvas.ICON_APPLICATION
  543. if ret == 0:
  544. iconName = voidptr2str(data)
  545. jacklib.free(data)
  546. if iconName == "hardware":
  547. groupSplit = patchcanvas.SPLIT_YES
  548. groupIcon = patchcanvas.ICON_HARDWARE
  549. #elif iconName =="carla":
  550. #groupIcon = patchcanvas.ICON_CARLA
  551. elif iconName =="distrho":
  552. groupIcon = patchcanvas.ICON_DISTRHO
  553. elif iconName =="file":
  554. groupIcon = patchcanvas.ICON_FILE
  555. elif iconName =="plugin":
  556. groupIcon = patchcanvas.ICON_PLUGIN
  557. patchcanvas.addGroup(groupId, groupName, groupSplit, groupIcon)
  558. groupObj = [None, None, None]
  559. groupObj[iGroupId] = groupId
  560. groupObj[iGroupName] = groupName
  561. groupObj[iGroupType] = GROUP_TYPE_JACK
  562. self.fGroupList.append(groupObj)
  563. self.fLastGroupId += 1
  564. return groupId
  565. def canvas_removeGroup(self, groupName):
  566. groupId = -1
  567. for group in self.fGroupList:
  568. if group[iGroupName] == groupName:
  569. groupId = group[iGroupId]
  570. self.fGroupList.remove(group)
  571. break
  572. else:
  573. print("Catia - remove group failed")
  574. return
  575. patchcanvas.removeGroup(groupId)
  576. def canvas_addAlsaPort(self, groupId, groupName, portName, portNameR, isPortInput):
  577. portId = self.fLastPortId
  578. portMode = patchcanvas.PORT_MODE_INPUT if isPortInput else patchcanvas.PORT_MODE_OUTPUT
  579. portType = patchcanvas.PORT_TYPE_MIDI_ALSA
  580. patchcanvas.addPort(groupId, portId, portName, portMode, portType)
  581. portObj = [None, None, None, None]
  582. portObj[iPortId] = portId
  583. portObj[iPortName] = portName
  584. portObj[iPortNameR] = "[ALSA-%s] %s" % ("Input" if isPortInput else "Output", portNameR)
  585. portObj[iPortGroupName] = groupName
  586. self.fPortList.append(portObj)
  587. self.fLastPortId += 1
  588. return portId
  589. def canvas_addJackPort(self, portPtr, portName):
  590. global gA2JClientName
  591. portId = self.fLastPortId
  592. groupId = -1
  593. portNameR = portName
  594. aliasN = self.fSavedSettings["Main/JackPortAlias"]
  595. if aliasN in (1, 2):
  596. aliases = jacklib.port_get_aliases(portPtr)
  597. if aliases[0] == 2 and aliasN == 2:
  598. portName = aliases[2]
  599. elif aliases[0] >= 1 and aliasN == 1:
  600. portName = aliases[1]
  601. portFlags = jacklib.port_flags(portPtr)
  602. groupName = portName.split(":", 1)[0]
  603. if portFlags & jacklib.JackPortIsInput:
  604. portMode = patchcanvas.PORT_MODE_INPUT
  605. elif portFlags & jacklib.JackPortIsOutput:
  606. portMode = patchcanvas.PORT_MODE_OUTPUT
  607. else:
  608. portMode = patchcanvas.PORT_MODE_NULL
  609. if groupName == gA2JClientName:
  610. portType = patchcanvas.PORT_TYPE_MIDI_A2J
  611. groupName = portName.replace("%s:" % gA2JClientName, "", 1).split(" [", 1)[0]
  612. portShortName = portName.split("): ", 1)[1]
  613. else:
  614. portShortName = portName.replace("%s:" % groupName, "", 1)
  615. portTypeStr = str(jacklib.port_type(portPtr), encoding="utf-8")
  616. if portTypeStr == jacklib.JACK_DEFAULT_AUDIO_TYPE:
  617. portType = patchcanvas.PORT_TYPE_AUDIO_JACK
  618. elif portTypeStr == jacklib.JACK_DEFAULT_MIDI_TYPE:
  619. portType = patchcanvas.PORT_TYPE_MIDI_JACK
  620. else:
  621. portType = patchcanvas.PORT_TYPE_NULL
  622. for group in self.fGroupList:
  623. if group[iGroupName] == groupName:
  624. groupId = group[iGroupId]
  625. break
  626. else:
  627. # For ports with no group
  628. groupId = self.canvas_addJackGroup(groupName)
  629. patchcanvas.addPort(groupId, portId, portShortName, portMode, portType)
  630. portObj = [None, None, None, None]
  631. portObj[iPortId] = portId
  632. portObj[iPortName] = portName
  633. portObj[iPortNameR] = portNameR
  634. portObj[iPortGroupName] = groupName
  635. self.fPortList.append(portObj)
  636. self.fLastPortId += 1
  637. if groupId not in self.fGroupSplitList and (portFlags & jacklib.JackPortIsPhysical) > 0:
  638. patchcanvas.splitGroup(groupId)
  639. patchcanvas.setGroupIcon(groupId, patchcanvas.ICON_HARDWARE)
  640. self.fGroupSplitList.append(groupId)
  641. return portId
  642. def canvas_removeJackPort(self, portId):
  643. patchcanvas.removePort(portId)
  644. for port in self.fPortList:
  645. if port[iPortId] == portId:
  646. groupName = port[iPortGroupName]
  647. self.fPortList.remove(port)
  648. break
  649. else:
  650. return
  651. # Check if group has no more ports; if yes remove it
  652. for port in self.fPortList:
  653. if port[iPortGroupName] == groupName:
  654. break
  655. else:
  656. self.canvas_removeGroup(groupName)
  657. def canvas_renamePort(self, portId, portShortName):
  658. patchcanvas.renamePort(portId, portShortName)
  659. def canvas_connectPorts(self, portOutId, portInId):
  660. connectionId = self.fLastConnectionId
  661. patchcanvas.connectPorts(connectionId, portOutId, portInId)
  662. connObj = [None, None, None]
  663. connObj[iConnId] = connectionId
  664. connObj[iConnOutput] = portOutId
  665. connObj[iConnInput] = portInId
  666. self.fConnectionList.append(connObj)
  667. self.fLastConnectionId += 1
  668. return connectionId
  669. def canvas_connectPortsByName(self, portOutName, portInName):
  670. portOutId = -1
  671. portInId = -1
  672. for port in self.fPortList:
  673. if port[iPortNameR] == portOutName:
  674. portOutId = port[iPortId]
  675. elif port[iPortNameR] == portInName:
  676. portInId = port[iPortId]
  677. if portOutId >= 0 and portInId >= 0:
  678. break
  679. else:
  680. print("Catia - connect jack ports failed")
  681. return -1
  682. return self.canvas_connectPorts(portOutId, portInId)
  683. def canvas_disconnectPorts(self, portOutId, portInId):
  684. for connection in self.fConnectionList:
  685. if connection[iConnOutput] == portOutId and connection[iConnInput] == portInId:
  686. patchcanvas.disconnectPorts(connection[iConnId])
  687. self.fConnectionList.remove(connection)
  688. break
  689. def canvas_disconnectPortsByName(self, portOutName, portInName):
  690. portOutId = -1
  691. portInId = -1
  692. for port in self.fPortList:
  693. if port[iPortNameR] == portOutName:
  694. portOutId = port[iPortId]
  695. elif port[iPortNameR] == portInName:
  696. portInId = port[iPortId]
  697. if portOutId == -1 or portInId == -1:
  698. print("Catia - disconnect ports failed")
  699. return
  700. self.canvas_disconnectPorts(portOutId, portInId)
  701. def jackStarted(self):
  702. if not gJack.client:
  703. gJack.client = jacklib.client_open("catia", jacklib.JackNoStartServer | jacklib.JackSessionID, None)
  704. if not gJack.client:
  705. self.jackStopped()
  706. return False
  707. canRender = render.canRender()
  708. self.ui.act_jack_render.setEnabled(canRender)
  709. self.ui.b_jack_render.setEnabled(canRender)
  710. self.menuJackServer(True)
  711. self.menuJackTransport(True)
  712. self.ui.cb_buffer_size.setEnabled(True)
  713. self.ui.cb_sample_rate.setEnabled(bool(gDBus.jack)) # DBus.jack and jacksettings.getSampleRate() != -1
  714. self.ui.menu_Jack_Buffer_Size.setEnabled(True)
  715. self.ui.pb_dsp_load.setMaximum(100)
  716. self.ui.pb_dsp_load.setValue(0)
  717. self.ui.pb_dsp_load.update()
  718. self.initJack()
  719. return True
  720. def jackStopped(self):
  721. if haveDBus:
  722. self.DBusReconnect()
  723. # client already closed
  724. gJack.client = None
  725. # refresh canvas (remove jack ports)
  726. patchcanvas.clear()
  727. self.initPorts()
  728. if self.fNextSampleRate:
  729. self.jack_setSampleRate(self.fNextSampleRate)
  730. if gDBus.jack:
  731. bufferSize = jacksettings.getBufferSize()
  732. sampleRate = jacksettings.getSampleRate()
  733. bufferSizeTest = bool(bufferSize != -1)
  734. sampleRateTest = bool(sampleRate != -1)
  735. if bufferSizeTest:
  736. self.ui_setBufferSize(bufferSize)
  737. if sampleRateTest:
  738. self.ui_setSampleRate(sampleRate)
  739. self.ui_setRealTime(jacksettings.isRealtime())
  740. self.ui.cb_buffer_size.setEnabled(bufferSizeTest)
  741. self.ui.cb_sample_rate.setEnabled(sampleRateTest)
  742. self.ui.menu_Jack_Buffer_Size.setEnabled(bufferSizeTest)
  743. else:
  744. self.ui.cb_buffer_size.setEnabled(False)
  745. self.ui.cb_sample_rate.setEnabled(False)
  746. self.ui.menu_Jack_Buffer_Size.setEnabled(False)
  747. self.ui.act_jack_render.setEnabled(False)
  748. self.ui.b_jack_render.setEnabled(False)
  749. self.menuJackServer(False)
  750. self.menuJackTransport(False)
  751. self.ui_setXruns(-1)
  752. if self.fCurTransportView == TRANSPORT_VIEW_HMS:
  753. self.ui.label_time.setText("00:00:00")
  754. elif self.fCurTransportView == TRANSPORT_VIEW_BBT:
  755. self.ui.label_time.setText("000|0|0000")
  756. elif self.fCurTransportView == TRANSPORT_VIEW_FRAMES:
  757. self.ui.label_time.setText("000'000'000")
  758. self.ui.pb_dsp_load.setValue(0)
  759. self.ui.pb_dsp_load.setMaximum(0)
  760. self.ui.pb_dsp_load.update()
  761. def a2jStarted(self):
  762. self.menuA2JBridge(True)
  763. def a2jStopped(self):
  764. self.menuA2JBridge(False)
  765. def menuJackServer(self, started):
  766. if gDBus.jack:
  767. self.ui.act_tools_jack_start.setEnabled(not started)
  768. self.ui.act_tools_jack_stop.setEnabled(started)
  769. self.menuA2JBridge(False)
  770. def menuJackTransport(self, enabled):
  771. self.ui.act_transport_play.setEnabled(enabled)
  772. self.ui.act_transport_stop.setEnabled(enabled)
  773. self.ui.act_transport_backwards.setEnabled(enabled)
  774. self.ui.act_transport_forwards.setEnabled(enabled)
  775. self.ui.menu_Transport.setEnabled(enabled)
  776. self.ui.group_transport.setEnabled(enabled)
  777. def menuA2JBridge(self, started):
  778. if gDBus.jack and not gDBus.jack.IsStarted():
  779. self.ui.act_tools_a2j_start.setEnabled(False)
  780. self.ui.act_tools_a2j_stop.setEnabled(False)
  781. self.ui.act_tools_a2j_export_hw.setEnabled(gDBus.a2j and not gDBus.a2j.is_started())
  782. else:
  783. self.ui.act_tools_a2j_start.setEnabled(not started)
  784. self.ui.act_tools_a2j_stop.setEnabled(started)
  785. self.ui.act_tools_a2j_export_hw.setEnabled(not started)
  786. def DBusSignalReceiver(self, *args, **kwds):
  787. if kwds["interface"] == "org.freedesktop.DBus" and kwds["path"] == "/org/freedesktop/DBus" and kwds["member"] == "NameOwnerChanged":
  788. appInterface, appId, newId = args
  789. if not newId:
  790. # Something crashed
  791. if appInterface == "org.gna.home.a2jmidid":
  792. QTimer.singleShot(0, self.slot_handleCrash_a2j)
  793. elif appInterface == "org.jackaudio.service":
  794. QTimer.singleShot(0, self.slot_handleCrash_jack)
  795. elif kwds['interface'] == "org.jackaudio.JackControl":
  796. if kwds['member'] == "ServerStarted":
  797. self.jackStarted()
  798. elif kwds['member'] == "ServerStopped":
  799. self.jackStopped()
  800. elif kwds['interface'] == "org.gna.home.a2jmidid.control":
  801. if kwds['member'] == "bridge_started":
  802. self.a2jStarted()
  803. elif kwds['member'] == "bridge_stopped":
  804. self.a2jStopped()
  805. def DBusReconnect(self):
  806. global gA2JClientName
  807. try:
  808. gDBus.jack = gDBus.bus.get_object("org.jackaudio.service", "/org/jackaudio/Controller")
  809. jacksettings.initBus(gDBus.bus)
  810. except:
  811. gDBus.jack = None
  812. try:
  813. gDBus.a2j = dbus.Interface(gDBus.bus.get_object("org.gna.home.a2jmidid", "/"), "org.gna.home.a2jmidid.control")
  814. gA2JClientName = str(gDBus.a2j.get_jack_client_name())
  815. except:
  816. gDBus.a2j = None
  817. gA2JClientName = None
  818. def JackXRunCallback(self, arg):
  819. if DEBUG: print("JackXRunCallback()")
  820. self.XRunCallback.emit()
  821. return 0
  822. def JackBufferSizeCallback(self, bufferSize, arg):
  823. if DEBUG: print("JackBufferSizeCallback(%i)" % bufferSize)
  824. self.BufferSizeCallback.emit(bufferSize)
  825. return 0
  826. def JackSampleRateCallback(self, sampleRate, arg):
  827. if DEBUG: print("JackSampleRateCallback(%i)" % sampleRate)
  828. self.SampleRateCallback.emit(sampleRate)
  829. return 0
  830. def JackClientRenameCallback(self, oldName, newName, arg):
  831. if DEBUG: print("JackClientRenameCallback(\"%s\", \"%s\")" % (oldName, newName))
  832. self.ClientRenameCallback.emit(str(oldName, encoding="utf-8"), str(newName, encoding="utf-8"))
  833. return 0
  834. def JackPortRegistrationCallback(self, portId, registerYesNo, arg):
  835. if DEBUG: print("JackPortRegistrationCallback(%i, %i)" % (portId, registerYesNo))
  836. self.PortRegistrationCallback.emit(portId, bool(registerYesNo))
  837. return 0
  838. def JackPortConnectCallback(self, portA, portB, connectYesNo, arg):
  839. if DEBUG: print("JackPortConnectCallback(%i, %i, %i)" % (portA, portB, connectYesNo))
  840. self.PortConnectCallback.emit(portA, portB, bool(connectYesNo))
  841. return 0
  842. def JackPortRenameCallback(self, portId, oldName, newName, arg):
  843. if DEBUG: print("JackPortRenameCallback(%i, \"%s\", \"%s\")" % (portId, oldName, newName))
  844. self.PortRenameCallback.emit(portId, str(oldName, encoding="utf-8"), str(newName, encoding="utf-8"))
  845. return 0
  846. def JackSessionCallback(self, event, arg):
  847. if WINDOWS:
  848. filepath = os.path.join(sys.argv[0])
  849. else:
  850. if sys.argv[0].startswith("/"):
  851. filepath = "catia"
  852. else:
  853. filepath = os.path.join(sys.path[0], "catia.py")
  854. event.command_line = str(filepath).encode("utf-8")
  855. jacklib.session_reply(gJack.client, event)
  856. if event.type == jacklib.JackSessionSaveAndQuit:
  857. app.quit()
  858. #jacklib.session_event_free(event)
  859. def JackShutdownCallback(self, arg):
  860. if DEBUG: print("JackShutdownCallback()")
  861. self.ShutdownCallback.emit()
  862. return 0
  863. @pyqtSlot(bool)
  864. def slot_showAlsaMIDI(self, yesNo):
  865. # refresh canvas (remove jack ports)
  866. patchcanvas.clear()
  867. self.initPorts()
  868. @pyqtSlot()
  869. def slot_JackServerStart(self):
  870. ret = False
  871. if gDBus.jack:
  872. try:
  873. ret = bool(gDBus.jack.StartServer())
  874. except:
  875. QMessageBox.warning(self, self.tr("Warning"), self.tr("Failed to start JACK, please check the logs for more information."))
  876. #self.jackStopped()
  877. return ret
  878. @pyqtSlot()
  879. def slot_JackServerStop(self):
  880. ret = False
  881. if gDBus.jack:
  882. ret = bool(gDBus.jack.StopServer())
  883. return ret
  884. @pyqtSlot()
  885. def slot_JackClearXruns(self):
  886. if gJack.client:
  887. self.fXruns = 0
  888. self.ui_setXruns(0)
  889. @pyqtSlot()
  890. def slot_A2JBridgeStart(self):
  891. ret = False
  892. if gDBus.a2j:
  893. ret = bool(gDBus.a2j.start())
  894. return ret
  895. @pyqtSlot()
  896. def slot_A2JBridgeStop(self):
  897. ret = False
  898. if gDBus.a2j:
  899. ret = bool(gDBus.a2j.stop())
  900. return ret
  901. @pyqtSlot()
  902. def slot_A2JBridgeExportHW(self):
  903. if gDBus.a2j:
  904. ask = QMessageBox.question(self, self.tr("A2J Hardware Export"), self.tr("Enable Hardware Export on the A2J Bridge?"), QMessageBox.Yes | QMessageBox.No | QMessageBox.Cancel, QMessageBox.No)
  905. if ask == QMessageBox.Yes:
  906. gDBus.a2j.set_hw_export(True)
  907. elif ask == QMessageBox.No:
  908. gDBus.a2j.set_hw_export(False)
  909. @pyqtSlot()
  910. def slot_XRunCallback(self):
  911. self.fXruns += 1
  912. self.ui_setXruns(self.fXruns)
  913. @pyqtSlot(int)
  914. def slot_BufferSizeCallback(self, bufferSize):
  915. self.ui_setBufferSize(bufferSize)
  916. @pyqtSlot(int)
  917. def slot_SampleRateCallback(self, sampleRate):
  918. self.ui_setSampleRate(sampleRate)
  919. self.ui_setRealTime(bool(int(jacklib.is_realtime(gJack.client))))
  920. self.ui_setXruns(0)
  921. @pyqtSlot(str, str)
  922. def slot_ClientRenameCallback(self, oldName, newName):
  923. pass # TODO
  924. @pyqtSlot(int, bool)
  925. def slot_PortRegistrationCallback(self, portIdJack, registerYesNo):
  926. portPtr = jacklib.port_by_id(gJack.client, portIdJack)
  927. portNameR = str(jacklib.port_name(portPtr), encoding="utf-8")
  928. if registerYesNo:
  929. self.canvas_addJackPort(portPtr, portNameR)
  930. else:
  931. for port in self.fPortList:
  932. if port[iPortNameR] == portNameR:
  933. portIdCanvas = port[iPortId]
  934. break
  935. else:
  936. return
  937. self.canvas_removeJackPort(portIdCanvas)
  938. @pyqtSlot(int, int, bool)
  939. def slot_PortConnectCallback(self, portIdJackA, portIdJackB, connectYesNo):
  940. portPtrA = jacklib.port_by_id(gJack.client, portIdJackA)
  941. portPtrB = jacklib.port_by_id(gJack.client, portIdJackB)
  942. portRealNameA = str(jacklib.port_name(portPtrA), encoding="utf-8")
  943. portRealNameB = str(jacklib.port_name(portPtrB), encoding="utf-8")
  944. if connectYesNo:
  945. self.canvas_connectPortsByName(portRealNameA, portRealNameB)
  946. else:
  947. self.canvas_disconnectPortsByName(portRealNameA, portRealNameB)
  948. @pyqtSlot(int, str, str)
  949. def slot_PortRenameCallback(self, portIdJack, oldName, newName):
  950. portPtr = jacklib.port_by_id(gJack.client, portIdJack)
  951. portShortName = str(jacklib.port_short_name(portPtr), encoding="utf-8")
  952. for port in self.fPortList:
  953. if port[iPortNameR] == oldName:
  954. portIdCanvas = port[iPortId]
  955. port[iPortNameR] = newName
  956. break
  957. else:
  958. return
  959. # Only set new name in canvas if no alias is active for this port
  960. aliases = jacklib.port_get_aliases(portPtr)
  961. if aliases[0] == 1 and self.fSavedSettings["Main/JackPortAlias"] == 1:
  962. pass
  963. elif aliases[0] == 2 and self.fSavedSettings["Main/JackPortAlias"] == 2:
  964. pass
  965. else:
  966. self.canvas_renamePort(portIdCanvas, portShortName)
  967. @pyqtSlot()
  968. def slot_ShutdownCallback(self):
  969. self.jackStopped()
  970. @pyqtSlot()
  971. def slot_handleCrash_a2j(self):
  972. global gA2JClientName
  973. try:
  974. gDBus.a2j = dbus.Interface(gDBus.bus.get_object("org.gna.home.a2jmidid", "/"), "org.gna.home.a2jmidid.control")
  975. gA2JClientName = str(gDBus.a2j.get_jack_client_name())
  976. except:
  977. gDBus.a2j = None
  978. gA2JClientName = None
  979. if gDBus.a2j:
  980. if gDBus.a2j.is_started():
  981. self.a2jStarted()
  982. else:
  983. self.a2jStopped()
  984. else:
  985. self.ui.act_tools_a2j_start.setEnabled(False)
  986. self.ui.act_tools_a2j_stop.setEnabled(False)
  987. self.ui.act_tools_a2j_export_hw.setEnabled(False)
  988. self.ui.menu_A2J_Bridge.setEnabled(False)
  989. @pyqtSlot()
  990. def slot_handleCrash_jack(self):
  991. self.DBusReconnect()
  992. if gDBus.jack:
  993. self.ui.act_jack_configure.setEnabled(True)
  994. self.ui.b_jack_configure.setEnabled(True)
  995. else:
  996. self.ui.act_tools_jack_start.setEnabled(False)
  997. self.ui.act_tools_jack_stop.setEnabled(False)
  998. self.ui.act_jack_configure.setEnabled(False)
  999. self.ui.b_jack_configure.setEnabled(False)
  1000. if gDBus.a2j:
  1001. if gDBus.a2j.is_started():
  1002. self.a2jStarted()
  1003. else:
  1004. self.a2jStopped()
  1005. else:
  1006. self.ui.act_tools_a2j_start.setEnabled(False)
  1007. self.ui.act_tools_a2j_stop.setEnabled(False)
  1008. self.ui.act_tools_a2j_export_hw.setEnabled(False)
  1009. self.ui.menu_A2J_Bridge.setEnabled(False)
  1010. self.jackStopped()
  1011. @pyqtSlot()
  1012. def slot_configureCatia(self):
  1013. dialog = SettingsW(self, "catia", hasGL)
  1014. if dialog.exec_():
  1015. self.loadSettings(False)
  1016. patchcanvas.clear()
  1017. pOptions = patchcanvas.options_t()
  1018. pOptions.theme_name = self.fSavedSettings["Canvas/Theme"]
  1019. pOptions.auto_hide_groups = self.fSavedSettings["Canvas/AutoHideGroups"]
  1020. pOptions.use_bezier_lines = self.fSavedSettings["Canvas/UseBezierLines"]
  1021. pOptions.antialiasing = self.fSavedSettings["Canvas/Antialiasing"]
  1022. pOptions.eyecandy = self.fSavedSettings["Canvas/EyeCandy"]
  1023. pFeatures = patchcanvas.features_t()
  1024. pFeatures.group_info = False
  1025. pFeatures.group_rename = False
  1026. pFeatures.port_info = True
  1027. pFeatures.port_rename = bool(self.fSavedSettings["Main/JackPortAlias"] > 0)
  1028. pFeatures.handle_group_pos = True
  1029. patchcanvas.setOptions(pOptions)
  1030. patchcanvas.setFeatures(pFeatures)
  1031. patchcanvas.init("Catia", self.scene, self.canvasCallback, DEBUG)
  1032. self.initPorts()
  1033. @pyqtSlot()
  1034. def slot_aboutCatia(self):
  1035. QMessageBox.about(self, self.tr("About Catia"), self.tr("<h3>Catia</h3>"
  1036. "<br>Version %s"
  1037. "<br>Catia is a nice JACK Patchbay with A2J Bridge integration.<br>"
  1038. "<br>Copyright (C) 2010-2018 falkTX" % VERSION))
  1039. def saveSettings(self):
  1040. settings = QSettings()
  1041. settings.setValue("Geometry", self.saveGeometry())
  1042. settings.setValue("ShowAlsaMIDI", self.ui.act_settings_show_alsa.isChecked())
  1043. settings.setValue("ShowToolbar", self.ui.frame_toolbar.isVisible())
  1044. settings.setValue("ShowStatusbar", self.ui.frame_statusbar.isVisible())
  1045. settings.setValue("TransportView", self.fCurTransportView)
  1046. def loadSettings(self, geometry):
  1047. settings = QSettings()
  1048. if geometry:
  1049. self.restoreGeometry(settings.value("Geometry", b""))
  1050. showAlsaMidi = settings.value("ShowAlsaMIDI", False, type=bool)
  1051. self.ui.act_settings_show_alsa.setChecked(showAlsaMidi)
  1052. showToolbar = settings.value("ShowToolbar", True, type=bool)
  1053. self.ui.act_settings_show_toolbar.setChecked(showToolbar)
  1054. self.ui.frame_toolbar.setVisible(showToolbar)
  1055. showStatusbar = settings.value("ShowStatusbar", True, type=bool)
  1056. self.ui.act_settings_show_statusbar.setChecked(showStatusbar)
  1057. self.ui.frame_statusbar.setVisible(showStatusbar)
  1058. self.setTransportView(settings.value("TransportView", TRANSPORT_VIEW_HMS, type=int))
  1059. self.fSavedSettings = {
  1060. "Main/RefreshInterval": settings.value("Main/RefreshInterval", 120, type=int),
  1061. "Main/JackPortAlias": settings.value("Main/JackPortAlias", 2, type=int),
  1062. "Canvas/Theme": settings.value("Canvas/Theme", patchcanvas.getDefaultThemeName(), type=str),
  1063. "Canvas/AutoHideGroups": settings.value("Canvas/AutoHideGroups", False, type=bool),
  1064. "Canvas/UseBezierLines": settings.value("Canvas/UseBezierLines", True, type=bool),
  1065. "Canvas/EyeCandy": settings.value("Canvas/EyeCandy", patchcanvas.EYECANDY_SMALL, type=int),
  1066. "Canvas/UseOpenGL": settings.value("Canvas/UseOpenGL", False, type=bool),
  1067. "Canvas/Antialiasing": settings.value("Canvas/Antialiasing", patchcanvas.ANTIALIASING_SMALL, type=int),
  1068. "Canvas/HighQualityAntialiasing": settings.value("Canvas/HighQualityAntialiasing", False, type=bool)
  1069. }
  1070. def timerEvent(self, event):
  1071. if event.timerId() == self.fTimer120:
  1072. if gJack.client:
  1073. self.refreshTransport()
  1074. elif event.timerId() == self.fTimer600:
  1075. if gJack.client:
  1076. self.refreshDSPLoad()
  1077. QMainWindow.timerEvent(self, event)
  1078. def closeEvent(self, event):
  1079. self.saveSettings()
  1080. patchcanvas.clear()
  1081. QMainWindow.closeEvent(self, event)
  1082. # ------------------------------------------------------------------------------------------------------------
  1083. # Main
  1084. if __name__ == '__main__':
  1085. # App initialization
  1086. app = QApplication(sys.argv)
  1087. app.setApplicationName("Catia")
  1088. app.setApplicationVersion(VERSION)
  1089. app.setOrganizationName("Cadence")
  1090. app.setWindowIcon(QIcon(":/scalable/catia.svg"))
  1091. if jacklib is None:
  1092. QMessageBox.critical(None, app.translate("CatiaMainW", "Error"), app.translate("CatiaMainW",
  1093. "JACK is not available in this system, cannot use this application."))
  1094. sys.exit(1)
  1095. if haveDBus:
  1096. gDBus.loop = DBusQtMainLoop(set_as_default=True)
  1097. gDBus.bus = dbus.SessionBus(mainloop=gDBus.loop)
  1098. try:
  1099. gDBus.jack = gDBus.bus.get_object("org.jackaudio.service", "/org/jackaudio/Controller")
  1100. jacksettings.initBus(gDBus.bus)
  1101. except:
  1102. gDBus.jack = None
  1103. try:
  1104. gDBus.a2j = dbus.Interface(gDBus.bus.get_object("org.gna.home.a2jmidid", "/"), "org.gna.home.a2jmidid.control")
  1105. gA2JClientName = str(gDBus.a2j.get_jack_client_name())
  1106. except:
  1107. gDBus.a2j = None
  1108. gA2JClientName = None
  1109. if DEBUG and (gDBus.jack or gDBus.a2j):
  1110. string = "Using DBus for "
  1111. if gDBus.jack:
  1112. string += "JACK"
  1113. if gDBus.a2j:
  1114. string += " and a2jmidid"
  1115. elif gDBus.a2j:
  1116. string += "a2jmidid"
  1117. print(string)
  1118. else:
  1119. gDBus.jack = None
  1120. gDBus.a2j = None
  1121. gA2JClientName = None
  1122. if DEBUG:
  1123. print("Not using DBus")
  1124. # Init GUI
  1125. gui = CatiaMainW()
  1126. # Set-up custom signal handling
  1127. setUpSignals(gui)
  1128. # Show GUI
  1129. gui.show()
  1130. # App-Loop
  1131. ret = app.exec_()
  1132. # Close Jack
  1133. if gJack.client:
  1134. jacklib.deactivate(gJack.client)
  1135. jacklib.client_close(gJack.client)
  1136. # Exit properly
  1137. sys.exit(ret)