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.

1836 lines
72KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Carla plugin database code
  4. # Copyright (C) 2011-2019 Filipe Coelho <falktx@falktx.com>
  5. #
  6. # This program is free software; you can redistribute it and/or
  7. # modify it under the terms of the GNU General Public License as
  8. # published by the Free Software Foundation; either version 2 of
  9. # the License, or 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 doc/GPL.txt file.
  17. # ----------------------------------------------------------------------------------------------------------------------
  18. # Imports (Global)
  19. from copy import deepcopy
  20. from subprocess import Popen, PIPE
  21. from PyQt5.QtCore import pyqtSignal, pyqtSlot, Qt, QEventLoop, QThread, QSettings
  22. from PyQt5.QtWidgets import QApplication, QDialog, QDialogButtonBox, QTableWidgetItem
  23. # ----------------------------------------------------------------------------------------------------------------------
  24. # Imports (Custom)
  25. import ui_carla_add_jack
  26. import ui_carla_database
  27. import ui_carla_refresh
  28. from carla_shared import *
  29. # ----------------------------------------------------------------------------------------------------------------------
  30. # Try Import LADSPA-RDF
  31. if WINDOWS:
  32. haveLRDF = False
  33. elif not CXFREEZE:
  34. try:
  35. import ladspa_rdf
  36. import json
  37. haveLRDF = True
  38. except:
  39. qWarning("LRDF Support not available (LADSPA-RDF will be disabled)")
  40. haveLRDF = False
  41. else:
  42. qWarning("LRDF Support disabled for static build (LADSPA-RDF will be disabled)")
  43. haveLRDF = False
  44. # ----------------------------------------------------------------------------------------------------------------------
  45. # Set LADSPA-RDF Path
  46. if haveLRDF and readEnvVars:
  47. LADSPA_RDF_PATH_env = os.getenv("LADSPA_RDF_PATH")
  48. if LADSPA_RDF_PATH_env:
  49. try:
  50. ladspa_rdf.set_rdf_path(LADSPA_RDF_PATH_env.split(splitter))
  51. except:
  52. pass
  53. del LADSPA_RDF_PATH_env
  54. # ----------------------------------------------------------------------------------------------------------------------
  55. # Plugin Query (helper functions)
  56. def findBinaries(binPath, OS):
  57. binaries = []
  58. if OS == "WINDOWS":
  59. extensions = (".dll",)
  60. elif OS == "MACOS":
  61. extensions = (".dylib", ".so")
  62. else:
  63. extensions = (".so",)
  64. for root, dirs, files in os.walk(binPath):
  65. for name in [name for name in files if name.lower().endswith(extensions)]:
  66. binaries.append(os.path.join(root, name))
  67. return binaries
  68. def findLV2Bundles(bundlePath):
  69. bundles = []
  70. for root, dirs, files in os.walk(bundlePath, followlinks=True):
  71. if root == bundlePath: continue
  72. if os.path.exists(os.path.join(root, "manifest.ttl")):
  73. bundles.append(root)
  74. return bundles
  75. def findMacVSTBundles(bundlePath):
  76. bundles = []
  77. for root, dirs, files in os.walk(bundlePath, followlinks=True):
  78. #if root == bundlePath: continue # FIXME
  79. for name in [name for name in dirs if name.lower().endswith(".vst")]:
  80. bundles.append(os.path.join(root, name))
  81. return bundles
  82. def findFilenames(filePath, stype):
  83. filenames = []
  84. if stype == "sf2":
  85. extensions = (".sf2",".sf3",)
  86. else:
  87. return []
  88. for root, dirs, files in os.walk(filePath):
  89. for name in [name for name in files if name.lower().endswith(extensions)]:
  90. filenames.append(os.path.join(root, name))
  91. return filenames
  92. # ----------------------------------------------------------------------------------------------------------------------
  93. # Plugin Query
  94. PLUGIN_QUERY_API_VERSION = 8
  95. PyPluginInfo = {
  96. 'API': PLUGIN_QUERY_API_VERSION,
  97. 'valid': False,
  98. 'build': BINARY_NONE,
  99. 'type': PLUGIN_NONE,
  100. 'hints': 0x0,
  101. 'filename': "",
  102. 'name': "",
  103. 'label': "",
  104. 'maker': "",
  105. 'uniqueId': 0,
  106. 'audio.ins': 0,
  107. 'audio.outs': 0,
  108. 'midi.ins': 0,
  109. 'midi.outs': 0,
  110. 'parameters.ins': 0,
  111. 'parameters.outs': 0
  112. }
  113. global gDiscoveryProcess
  114. gDiscoveryProcess = None
  115. def findWinePrefix(filename, recursionLimit = 10):
  116. if recursionLimit == 0 or len(filename) < 5 or "/" not in filename:
  117. return ""
  118. path = filename[:filename.rfind("/")]
  119. if os.path.isdir(path + "/dosdevices"):
  120. return path
  121. return findWinePrefix(path, recursionLimit-1)
  122. def runCarlaDiscovery(itype, stype, filename, tool, wineSettings=None):
  123. if not os.path.exists(tool):
  124. qWarning("runCarlaDiscovery() - tool '%s' does not exist" % tool)
  125. return
  126. command = []
  127. if LINUX or MACOS:
  128. command.append("env")
  129. command.append("LANG=C")
  130. command.append("LD_PRELOAD=")
  131. if wineSettings is not None:
  132. command.append("WINEDEBUG=-all")
  133. if wineSettings['autoPrefix']:
  134. winePrefix = findWinePrefix(filename)
  135. else:
  136. winePrefix = ""
  137. if not winePrefix:
  138. envWinePrefix = os.getenv("WINEPREFIX")
  139. if envWinePrefix:
  140. winePrefix = envWinePrefix
  141. elif wineSettings['fallbackPrefix']:
  142. winePrefix = os.path.expanduser(wineSettings['fallbackPrefix'])
  143. else:
  144. winePrefix = os.path.expanduser("~/.wine")
  145. wineCMD = wineSettings['executable'] if wineSettings['executable'] else "wine"
  146. if tool.endswith("64.exe") and os.path.exists(wineCMD + "64"):
  147. wineCMD += "64"
  148. command.append("WINEPREFIX=" + winePrefix)
  149. command.append(wineCMD)
  150. command.append(tool)
  151. command.append(stype)
  152. command.append(filename)
  153. global gDiscoveryProcess
  154. gDiscoveryProcess = Popen(command, stdout=PIPE)
  155. pinfo = None
  156. plugins = []
  157. fakeLabel = os.path.basename(filename).rsplit(".", 1)[0]
  158. while True:
  159. try:
  160. line = gDiscoveryProcess.stdout.readline().decode("utf-8", errors="ignore")
  161. except:
  162. print("ERROR: discovery readline failed")
  163. break
  164. # line is valid, strip it
  165. if line:
  166. line = line.strip()
  167. # line is invalid, try poll() again
  168. elif gDiscoveryProcess.poll() is None:
  169. continue
  170. # line is invalid and poll() failed, stop here
  171. else:
  172. break
  173. if line == "carla-discovery::init::-----------":
  174. pinfo = deepcopy(PyPluginInfo)
  175. pinfo['type'] = itype
  176. pinfo['filename'] = filename if filename != ":all" else ""
  177. elif line == "carla-discovery::end::------------":
  178. if pinfo is not None:
  179. plugins.append(pinfo)
  180. del pinfo
  181. pinfo = None
  182. elif line == "Segmentation fault":
  183. print("carla-discovery::crash::%s crashed during discovery" % filename)
  184. elif line.startswith("err:module:import_dll Library"):
  185. print(line)
  186. elif line.startswith("carla-discovery::info::"):
  187. print("%s - %s" % (line, filename))
  188. elif line.startswith("carla-discovery::warning::"):
  189. print("%s - %s" % (line, filename))
  190. elif line.startswith("carla-discovery::error::"):
  191. print("%s - %s" % (line, filename))
  192. elif line.startswith("carla-discovery::"):
  193. if pinfo == None:
  194. continue
  195. try:
  196. prop, value = line.replace("carla-discovery::", "").split("::", 1)
  197. except:
  198. continue
  199. if prop == "build":
  200. if value.isdigit(): pinfo['build'] = int(value)
  201. elif prop == "name":
  202. pinfo['name'] = value if value else fakeLabel
  203. elif prop == "label":
  204. pinfo['label'] = value if value else fakeLabel
  205. elif prop == "maker":
  206. pinfo['maker'] = value
  207. elif prop == "uniqueId":
  208. if value.isdigit(): pinfo['uniqueId'] = int(value)
  209. elif prop == "hints":
  210. if value.isdigit(): pinfo['hints'] = int(value)
  211. elif prop == "audio.ins":
  212. if value.isdigit(): pinfo['audio.ins'] = int(value)
  213. elif prop == "audio.outs":
  214. if value.isdigit(): pinfo['audio.outs'] = int(value)
  215. elif prop == "midi.ins":
  216. if value.isdigit(): pinfo['midi.ins'] = int(value)
  217. elif prop == "midi.outs":
  218. if value.isdigit(): pinfo['midi.outs'] = int(value)
  219. elif prop == "parameters.ins":
  220. if value.isdigit(): pinfo['parameters.ins'] = int(value)
  221. elif prop == "parameters.outs":
  222. if value.isdigit(): pinfo['parameters.outs'] = int(value)
  223. elif prop == "uri":
  224. if value:
  225. pinfo['label'] = value
  226. else:
  227. # cannot use empty URIs
  228. del pinfo
  229. pinfo = None
  230. continue
  231. else:
  232. print("%s - %s (unknown property)" % (line, filename))
  233. # FIXME?
  234. tmp = gDiscoveryProcess
  235. gDiscoveryProcess = None
  236. del gDiscoveryProcess, tmp
  237. return plugins
  238. def killDiscovery():
  239. global gDiscoveryProcess
  240. if gDiscoveryProcess is not None:
  241. gDiscoveryProcess.kill()
  242. def checkPluginCached(desc, ptype):
  243. pinfo = deepcopy(PyPluginInfo)
  244. pinfo['build'] = BINARY_NATIVE
  245. pinfo['type'] = ptype
  246. pinfo['hints'] = desc['hints']
  247. pinfo['name'] = desc['name']
  248. pinfo['label'] = desc['label']
  249. pinfo['maker'] = desc['maker']
  250. pinfo['audio.ins'] = desc['audioIns']
  251. pinfo['audio.outs'] = desc['audioOuts']
  252. pinfo['midi.ins'] = desc['midiIns']
  253. pinfo['midi.outs'] = desc['midiOuts']
  254. pinfo['parameters.ins'] = desc['parameterIns']
  255. pinfo['parameters.outs'] = desc['parameterOuts']
  256. if ptype == PLUGIN_SFZ:
  257. pinfo['filename'] = pinfo['label']
  258. pinfo['label'] = pinfo['name']
  259. return pinfo
  260. def checkPluginLADSPA(filename, tool, wineSettings=None):
  261. return runCarlaDiscovery(PLUGIN_LADSPA, "LADSPA", filename, tool, wineSettings)
  262. def checkPluginDSSI(filename, tool, wineSettings=None):
  263. return runCarlaDiscovery(PLUGIN_DSSI, "DSSI", filename, tool, wineSettings)
  264. def checkPluginLV2(filename, tool, wineSettings=None):
  265. return runCarlaDiscovery(PLUGIN_LV2, "LV2", filename, tool, wineSettings)
  266. def checkPluginVST2(filename, tool, wineSettings=None):
  267. return runCarlaDiscovery(PLUGIN_VST2, "VST2", filename, tool, wineSettings)
  268. def checkFileSF2(filename, tool):
  269. return runCarlaDiscovery(PLUGIN_SF2, "SF2", filename, tool)
  270. def checkFileSFZ(filename, tool):
  271. return runCarlaDiscovery(PLUGIN_SFZ, "SFZ", filename, tool)
  272. # ----------------------------------------------------------------------------------------------------------------------
  273. # Separate Thread for Plugin Search
  274. class SearchPluginsThread(QThread):
  275. pluginLook = pyqtSignal(int, str)
  276. def __init__(self, parent, pathBinaries):
  277. QThread.__init__(self, parent)
  278. self.fContinueChecking = False
  279. self.fPathBinaries = pathBinaries
  280. self.fCheckNative = False
  281. self.fCheckPosix32 = False
  282. self.fCheckPosix64 = False
  283. self.fCheckWin32 = False
  284. self.fCheckWin64 = False
  285. self.fCheckLADSPA = False
  286. self.fCheckDSSI = False
  287. self.fCheckLV2 = False
  288. self.fCheckVST2 = False
  289. self.fCheckSF2 = False
  290. self.fCheckSFZ = False
  291. if WINDOWS:
  292. toolNative = "carla-discovery-win64.exe" if kIs64bit else "carla-discovery-win32.exe"
  293. self.fWineSettings = None
  294. else:
  295. toolNative = "carla-discovery-native"
  296. settings = QSettings("falkTX", "Carla2")
  297. self.fWineSettings = {
  298. 'executable' : settings.value(CARLA_KEY_WINE_EXECUTABLE, CARLA_DEFAULT_WINE_EXECUTABLE, type=str),
  299. 'autoPrefix' : settings.value(CARLA_KEY_WINE_AUTO_PREFIX, CARLA_DEFAULT_WINE_AUTO_PREFIX, type=bool),
  300. 'fallbackPrefix': settings.value(CARLA_KEY_WINE_FALLBACK_PREFIX, CARLA_DEFAULT_WINE_FALLBACK_PREFIX, type=str)
  301. }
  302. del settings
  303. self.fToolNative = os.path.join(pathBinaries, toolNative)
  304. if not os.path.exists(self.fToolNative):
  305. self.fToolNative = ""
  306. self.fCurCount = 0
  307. self.fCurPercentValue = 0
  308. self.fLastCheckValue = 0
  309. self.fSomethingChanged = False
  310. # -------------------------------------------------------------
  311. def hasSomethingChanged(self):
  312. return self.fSomethingChanged
  313. def setSearchBinaryTypes(self, native, posix32, posix64, win32, win64):
  314. self.fCheckNative = native
  315. self.fCheckPosix32 = posix32
  316. self.fCheckPosix64 = posix64
  317. self.fCheckWin32 = win32
  318. self.fCheckWin64 = win64
  319. def setSearchPluginTypes(self, ladspa, dssi, lv2, vst2, sf2, sfz):
  320. self.fCheckLADSPA = ladspa
  321. self.fCheckDSSI = dssi
  322. self.fCheckLV2 = lv2
  323. self.fCheckVST2 = vst2
  324. self.fCheckSF2 = sf2
  325. self.fCheckSFZ = sfz
  326. def stop(self):
  327. self.fContinueChecking = False
  328. def run(self):
  329. settingsDB = QSettings("falkTX", "CarlaPlugins4")
  330. self.fContinueChecking = True
  331. self.fCurCount = 0
  332. # looking for plugins via external discovery
  333. pluginCount = 0
  334. if self.fCheckLADSPA: pluginCount += 1
  335. if self.fCheckDSSI: pluginCount += 1
  336. if self.fCheckVST2: pluginCount += 1
  337. # Increase count by the number of externally discoverable plugin types
  338. if self.fCheckNative:
  339. self.fCurCount += pluginCount
  340. if self.fCheckPosix32:
  341. self.fCurCount += pluginCount
  342. if self.fCheckPosix64:
  343. self.fCurCount += pluginCount
  344. if self.fCheckWin32:
  345. self.fCurCount += pluginCount
  346. if self.fCheckWin64:
  347. self.fCurCount += pluginCount
  348. # Special case for cached plugins, only "search" for native plugins (does not need tool)
  349. if self.fCheckLV2:
  350. if self.fCheckNative:
  351. self.fCurCount += 1
  352. else:
  353. self.fCheckLV2 = False
  354. if self.fCheckSFZ:
  355. if self.fCheckNative:
  356. self.fCurCount += 1
  357. else:
  358. self.fCheckSFZ = False
  359. # Special case for Sound Kits, only search native
  360. if self.fCheckNative and self.fToolNative:
  361. if self.fCheckSF2: self.fCurCount += 1
  362. else:
  363. self.fCheckSF2 = False
  364. if self.fCurCount == 0:
  365. return
  366. self.fCurPercentValue = 100.0 / self.fCurCount
  367. self.fLastCheckValue = 0.0
  368. del pluginCount
  369. if HAIKU:
  370. OS = "HAIKU"
  371. elif LINUX:
  372. OS = "LINUX"
  373. elif MACOS:
  374. OS = "MACOS"
  375. elif WINDOWS:
  376. OS = "WINDOWS"
  377. else:
  378. OS = "UNKNOWN"
  379. if not self.fContinueChecking: return
  380. self.fSomethingChanged = True
  381. if self.fCheckLADSPA:
  382. checkValue = 0.0
  383. if haveLRDF:
  384. if self.fCheckNative: checkValue += 0.1
  385. if self.fCheckPosix32: checkValue += 0.1
  386. if self.fCheckPosix64: checkValue += 0.1
  387. if self.fCheckWin32: checkValue += 0.1
  388. if self.fCheckWin64: checkValue += 0.1
  389. rdfPadValue = self.fCurPercentValue * checkValue
  390. if self.fCheckNative:
  391. plugins = self._checkLADSPA(OS, self.fToolNative)
  392. settingsDB.setValue("Plugins/LADSPA_native", plugins)
  393. if not self.fContinueChecking: return
  394. if self.fCheckPosix32:
  395. plugins = self._checkLADSPA(OS, os.path.join(self.fPathBinaries, "carla-discovery-posix32"))
  396. settingsDB.setValue("Plugins/LADSPA_posix32", plugins)
  397. if not self.fContinueChecking: return
  398. if self.fCheckPosix64:
  399. plugins = self._checkLADSPA(OS, os.path.join(self.fPathBinaries, "carla-discovery-posix64"))
  400. settingsDB.setValue("Plugins/LADSPA_posix64", plugins)
  401. if not self.fContinueChecking: return
  402. if self.fCheckWin32:
  403. plugins = self._checkLADSPA("WINDOWS", os.path.join(self.fPathBinaries, "carla-discovery-win32.exe"), not WINDOWS)
  404. settingsDB.setValue("Plugins/LADSPA_win32", plugins)
  405. if not self.fContinueChecking: return
  406. if self.fCheckWin64:
  407. plugins = self._checkLADSPA("WINDOWS", os.path.join(self.fPathBinaries, "carla-discovery-win64.exe"), not WINDOWS)
  408. settingsDB.setValue("Plugins/LADSPA_win64", plugins)
  409. settingsDB.sync()
  410. if not self.fContinueChecking: return
  411. if haveLRDF and checkValue > 0:
  412. startValue = self.fLastCheckValue - rdfPadValue
  413. self._pluginLook(startValue, "LADSPA RDFs...")
  414. try:
  415. ladspaRdfInfo = ladspa_rdf.recheck_all_plugins(self, startValue, self.fCurPercentValue, checkValue)
  416. except:
  417. ladspaRdfInfo = None
  418. if ladspaRdfInfo is not None:
  419. settingsDir = os.path.join(HOME, ".config", "falkTX")
  420. fdLadspa = open(os.path.join(settingsDir, "ladspa_rdf.db"), 'w')
  421. json.dump(ladspaRdfInfo, fdLadspa)
  422. fdLadspa.close()
  423. if not self.fContinueChecking: return
  424. if self.fCheckDSSI:
  425. if self.fCheckNative:
  426. plugins = self._checkDSSI(OS, self.fToolNative)
  427. settingsDB.setValue("Plugins/DSSI_native", plugins)
  428. if not self.fContinueChecking: return
  429. if self.fCheckPosix32:
  430. plugins = self._checkDSSI(OS, os.path.join(self.fPathBinaries, "carla-discovery-posix32"))
  431. settingsDB.setValue("Plugins/DSSI_posix32", plugins)
  432. if not self.fContinueChecking: return
  433. if self.fCheckPosix64:
  434. plugins = self._checkDSSI(OS, os.path.join(self.fPathBinaries, "carla-discovery-posix64"))
  435. settingsDB.setValue("Plugins/DSSI_posix64", plugins)
  436. if not self.fContinueChecking: return
  437. if self.fCheckWin32:
  438. plugins = self._checkDSSI("WINDOWS", os.path.join(self.fPathBinaries, "carla-discovery-win32.exe"), not WINDOWS)
  439. settingsDB.setValue("Plugins/DSSI_win32", plugins)
  440. if not self.fContinueChecking: return
  441. if self.fCheckWin64:
  442. plugins = self._checkDSSI("WINDOWS", os.path.join(self.fPathBinaries, "carla-discovery-win64.exe"), not WINDOWS)
  443. settingsDB.setValue("Plugins/DSSI_win64", plugins)
  444. settingsDB.sync()
  445. if not self.fContinueChecking: return
  446. if self.fCheckLV2:
  447. plugins = self._checkLv2Cached()
  448. settingsDB.setValue("Plugins/LV2", plugins)
  449. settingsDB.sync()
  450. if not self.fContinueChecking: return
  451. if self.fCheckVST2:
  452. if self.fCheckNative:
  453. plugins = self._checkVST2(OS, self.fToolNative)
  454. settingsDB.setValue("Plugins/VST2_native", plugins)
  455. if not self.fContinueChecking: return
  456. if self.fCheckPosix32:
  457. plugins = self._checkVST2(OS, os.path.join(self.fPathBinaries, "carla-discovery-posix32"))
  458. settingsDB.setValue("Plugins/VST2_posix32", plugins)
  459. if not self.fContinueChecking: return
  460. if self.fCheckPosix64:
  461. plugins = self._checkVST2(OS, os.path.join(self.fPathBinaries, "carla-discovery-posix64"))
  462. settingsDB.setValue("Plugins/VST2_posix64", plugins)
  463. if not self.fContinueChecking: return
  464. if self.fCheckWin32:
  465. plugins = self._checkVST2("WINDOWS", os.path.join(self.fPathBinaries, "carla-discovery-win32.exe"), not WINDOWS)
  466. settingsDB.setValue("Plugins/VST2_win32", plugins)
  467. if not self.fContinueChecking: return
  468. if self.fCheckWin64:
  469. plugins = self._checkVST2("WINDOWS", os.path.join(self.fPathBinaries, "carla-discovery-win64.exe"), not WINDOWS)
  470. settingsDB.setValue("Plugins/VST2_win64", plugins)
  471. if not self.fContinueChecking: return
  472. settingsDB.sync()
  473. if not self.fContinueChecking: return
  474. if self.fCheckSF2:
  475. settings = QSettings("falkTX", "Carla2")
  476. SF2_PATH = toList(settings.value(CARLA_KEY_PATHS_SF2, CARLA_DEFAULT_SF2_PATH))
  477. del settings
  478. kits = self._checkKIT(SF2_PATH, "sf2")
  479. settingsDB.setValue("Plugins/SF2", kits)
  480. settingsDB.sync()
  481. if not self.fContinueChecking: return
  482. if self.fCheckSFZ:
  483. kits = self._checkSfzCached()
  484. settingsDB.setValue("Plugins/SFZ", kits)
  485. settingsDB.sync()
  486. def _checkLADSPA(self, OS, tool, isWine=False):
  487. ladspaBinaries = []
  488. ladspaPlugins = []
  489. self._pluginLook(self.fLastCheckValue, "LADSPA plugins...")
  490. settings = QSettings("falkTX", "Carla2")
  491. LADSPA_PATH = toList(settings.value(CARLA_KEY_PATHS_LADSPA, CARLA_DEFAULT_LADSPA_PATH))
  492. del settings
  493. for iPATH in LADSPA_PATH:
  494. binaries = findBinaries(iPATH, OS)
  495. for binary in binaries:
  496. if binary not in ladspaBinaries:
  497. ladspaBinaries.append(binary)
  498. ladspaBinaries.sort()
  499. if not self.fContinueChecking:
  500. return ladspaPlugins
  501. for i in range(len(ladspaBinaries)):
  502. ladspa = ladspaBinaries[i]
  503. percent = ( float(i) / len(ladspaBinaries) ) * self.fCurPercentValue
  504. self._pluginLook((self.fLastCheckValue + percent) * 0.9, ladspa)
  505. plugins = checkPluginLADSPA(ladspa, tool, self.fWineSettings if isWine else None)
  506. if plugins:
  507. ladspaPlugins.append(plugins)
  508. if not self.fContinueChecking:
  509. break
  510. self.fLastCheckValue += self.fCurPercentValue
  511. return ladspaPlugins
  512. def _checkDSSI(self, OS, tool, isWine=False):
  513. dssiBinaries = []
  514. dssiPlugins = []
  515. self._pluginLook(self.fLastCheckValue, "DSSI plugins...")
  516. settings = QSettings("falkTX", "Carla2")
  517. DSSI_PATH = toList(settings.value(CARLA_KEY_PATHS_DSSI, CARLA_DEFAULT_DSSI_PATH))
  518. del settings
  519. for iPATH in DSSI_PATH:
  520. binaries = findBinaries(iPATH, OS)
  521. for binary in binaries:
  522. if binary not in dssiBinaries:
  523. dssiBinaries.append(binary)
  524. dssiBinaries.sort()
  525. if not self.fContinueChecking:
  526. return dssiPlugins
  527. for i in range(len(dssiBinaries)):
  528. dssi = dssiBinaries[i]
  529. percent = ( float(i) / len(dssiBinaries) ) * self.fCurPercentValue
  530. self._pluginLook(self.fLastCheckValue + percent, dssi)
  531. plugins = checkPluginDSSI(dssi, tool, self.fWineSettings if isWine else None)
  532. if plugins:
  533. dssiPlugins.append(plugins)
  534. if not self.fContinueChecking:
  535. break
  536. self.fLastCheckValue += self.fCurPercentValue
  537. return dssiPlugins
  538. def _checkVST2(self, OS, tool, isWine=False):
  539. vst2Binaries = []
  540. vstPlugins = []
  541. if MACOS and not isWine:
  542. self._pluginLook(self.fLastCheckValue, "VST2 bundles...")
  543. else:
  544. self._pluginLook(self.fLastCheckValue, "VST2 plugins...")
  545. settings = QSettings("falkTX", "Carla2")
  546. VST2_PATH = toList(settings.value(CARLA_KEY_PATHS_VST2, CARLA_DEFAULT_VST2_PATH))
  547. del settings
  548. for iPATH in VST2_PATH:
  549. if MACOS and not isWine:
  550. binaries = findMacVSTBundles(iPATH)
  551. else:
  552. binaries = findBinaries(iPATH, OS)
  553. for binary in binaries:
  554. if binary not in vst2Binaries:
  555. vst2Binaries.append(binary)
  556. vst2Binaries.sort()
  557. if not self.fContinueChecking:
  558. return vstPlugins
  559. for i in range(len(vst2Binaries)):
  560. vst2 = vst2Binaries[i]
  561. percent = ( float(i) / len(vst2Binaries) ) * self.fCurPercentValue
  562. self._pluginLook(self.fLastCheckValue + percent, vst2)
  563. plugins = checkPluginVST2(vst2, tool, self.fWineSettings if isWine else None)
  564. if plugins:
  565. vstPlugins.append(plugins)
  566. if not self.fContinueChecking:
  567. break
  568. self.fLastCheckValue += self.fCurPercentValue
  569. return vstPlugins
  570. def _checkKIT(self, kitPATH, kitExtension):
  571. kitFiles = []
  572. kitPlugins = []
  573. for iPATH in kitPATH:
  574. files = findFilenames(iPATH, kitExtension)
  575. for file_ in files:
  576. if file_ not in kitFiles:
  577. kitFiles.append(file_)
  578. kitFiles.sort()
  579. if not self.fContinueChecking:
  580. return kitPlugins
  581. for i in range(len(kitFiles)):
  582. kit = kitFiles[i]
  583. percent = ( float(i) / len(kitFiles) ) * self.fCurPercentValue
  584. self._pluginLook(self.fLastCheckValue + percent, kit)
  585. if kitExtension == "sf2":
  586. plugins = checkFileSF2(kit, self.fToolNative)
  587. else:
  588. plugins = None
  589. if plugins:
  590. kitPlugins.append(plugins)
  591. if not self.fContinueChecking:
  592. break
  593. self.fLastCheckValue += self.fCurPercentValue
  594. return kitPlugins
  595. def _checkLv2Cached(self):
  596. settings = QSettings("falkTX", "Carla2")
  597. PLUG_PATH = splitter.join(toList(settings.value(CARLA_KEY_PATHS_LV2, CARLA_DEFAULT_LV2_PATH)))
  598. del settings
  599. lv2Plugins = []
  600. self._pluginLook(self.fLastCheckValue, "LV2 plugins...")
  601. count = gCarla.utils.get_cached_plugin_count(PLUGIN_LV2, PLUG_PATH)
  602. if not self.fContinueChecking:
  603. return lv2Plugins
  604. for i in range(count):
  605. descInfo = gCarla.utils.get_cached_plugin_info(PLUGIN_LV2, i)
  606. percent = ( float(i) / count ) * self.fCurPercentValue
  607. self._pluginLook(self.fLastCheckValue + percent, descInfo['label'])
  608. if not descInfo['valid']:
  609. continue
  610. lv2Plugins.append(checkPluginCached(descInfo, PLUGIN_LV2))
  611. if not self.fContinueChecking:
  612. break
  613. self.fLastCheckValue += self.fCurPercentValue
  614. return lv2Plugins
  615. def _checkSfzCached(self):
  616. settings = QSettings("falkTX", "Carla2")
  617. PLUG_PATH = splitter.join(toList(settings.value(CARLA_KEY_PATHS_SFZ, CARLA_DEFAULT_SFZ_PATH)))
  618. del settings
  619. sfzKits = []
  620. self._pluginLook(self.fLastCheckValue, "SFZ kits...")
  621. count = gCarla.utils.get_cached_plugin_count(PLUGIN_SFZ, PLUG_PATH)
  622. if not self.fContinueChecking:
  623. return sfzKits
  624. for i in range(count):
  625. descInfo = gCarla.utils.get_cached_plugin_info(PLUGIN_SFZ, i)
  626. percent = ( float(i) / count ) * self.fCurPercentValue
  627. self._pluginLook(self.fLastCheckValue + percent, descInfo['label'])
  628. if not descInfo['valid']:
  629. continue
  630. sfzKits.append(checkPluginCached(descInfo, PLUGIN_SFZ))
  631. if not self.fContinueChecking:
  632. break
  633. self.fLastCheckValue += self.fCurPercentValue
  634. return sfzKits
  635. def _pluginLook(self, percent, plugin):
  636. self.pluginLook.emit(percent, plugin)
  637. # ---------------------------------------------------------------------------------------------------------------------
  638. # Plugin Refresh Dialog
  639. class PluginRefreshW(QDialog):
  640. def __init__(self, parent, host):
  641. QDialog.__init__(self, parent)
  642. self.host = host
  643. self.ui = ui_carla_refresh.Ui_PluginRefreshW()
  644. self.ui.setupUi(self)
  645. if False:
  646. # kdevelop likes this :)
  647. self.host = host = CarlaHostNull()
  648. # -------------------------------------------------------------------------------------------------------------
  649. # Internal stuff
  650. hasNative = os.path.exists(os.path.join(self.host.pathBinaries, "carla-discovery-native"))
  651. hasPosix32 = os.path.exists(os.path.join(self.host.pathBinaries, "carla-discovery-posix32"))
  652. hasPosix64 = os.path.exists(os.path.join(self.host.pathBinaries, "carla-discovery-posix64"))
  653. hasWin32 = os.path.exists(os.path.join(self.host.pathBinaries, "carla-discovery-win32.exe"))
  654. hasWin64 = os.path.exists(os.path.join(self.host.pathBinaries, "carla-discovery-win64.exe"))
  655. self.fThread = SearchPluginsThread(self, host.pathBinaries)
  656. self.fIconYes = getIcon("dialog-ok-apply").pixmap(16, 16)
  657. self.fIconNo = getIcon("dialog-error").pixmap(16, 16)
  658. # -------------------------------------------------------------------------------------------------------------
  659. # Set-up GUI
  660. self.ui.b_skip.setVisible(False)
  661. if HAIKU:
  662. self.ui.ch_posix32.setText("Haiku 32bit")
  663. self.ui.ch_posix64.setText("Haiku 64bit")
  664. elif LINUX:
  665. self.ui.ch_posix32.setText("Linux 32bit")
  666. self.ui.ch_posix64.setText("Linux 64bit")
  667. elif MACOS:
  668. self.ui.ch_posix32.setText("MacOS 32bit")
  669. self.ui.ch_posix64.setText("MacOS 64bit")
  670. if hasPosix32 and not WINDOWS:
  671. self.ui.ico_posix32.setPixmap(self.fIconYes)
  672. else:
  673. self.ui.ico_posix32.setPixmap(self.fIconNo)
  674. self.ui.ch_posix32.setEnabled(False)
  675. if hasPosix64 and not WINDOWS:
  676. self.ui.ico_posix64.setPixmap(self.fIconYes)
  677. else:
  678. self.ui.ico_posix64.setPixmap(self.fIconNo)
  679. self.ui.ch_posix64.setEnabled(False)
  680. if hasWin32:
  681. self.ui.ico_win32.setPixmap(self.fIconYes)
  682. else:
  683. self.ui.ico_win32.setPixmap(self.fIconNo)
  684. self.ui.ch_win32.setEnabled(False)
  685. if hasWin64:
  686. self.ui.ico_win64.setPixmap(self.fIconYes)
  687. else:
  688. self.ui.ico_win64.setPixmap(self.fIconNo)
  689. self.ui.ch_win64.setEnabled(False)
  690. if haveLRDF:
  691. self.ui.ico_rdflib.setPixmap(self.fIconYes)
  692. else:
  693. self.ui.ico_rdflib.setPixmap(self.fIconNo)
  694. if WINDOWS:
  695. if kIs64bit:
  696. hasNative = hasWin64
  697. hasNonNative = hasWin32
  698. self.ui.ch_win64.setEnabled(False)
  699. self.ui.ch_win64.setVisible(False)
  700. self.ui.ico_win64.setVisible(False)
  701. self.ui.label_win64.setVisible(False)
  702. else:
  703. hasNative = hasWin32
  704. hasNonNative = hasWin64
  705. self.ui.ch_win32.setEnabled(False)
  706. self.ui.ch_win32.setVisible(False)
  707. self.ui.ico_win32.setVisible(False)
  708. self.ui.label_win32.setVisible(False)
  709. self.ui.ch_posix32.setEnabled(False)
  710. self.ui.ch_posix32.setVisible(False)
  711. self.ui.ch_posix64.setEnabled(False)
  712. self.ui.ch_posix64.setVisible(False)
  713. self.ui.ico_posix32.hide()
  714. self.ui.ico_posix64.hide()
  715. self.ui.label_posix32.hide()
  716. self.ui.label_posix64.hide()
  717. self.ui.ico_rdflib.hide()
  718. self.ui.label_rdflib.hide()
  719. else:
  720. if kIs64bit:
  721. hasNonNative = bool(hasPosix32 or hasWin32 or hasWin64)
  722. self.ui.ch_posix64.setEnabled(False)
  723. self.ui.ch_posix64.setVisible(False)
  724. self.ui.ico_posix64.setVisible(False)
  725. self.ui.label_posix64.setVisible(False)
  726. else:
  727. hasNonNative = bool(hasPosix64 or hasWin32 or hasWin64)
  728. self.ui.ch_posix32.setEnabled(False)
  729. self.ui.ch_posix32.setVisible(False)
  730. self.ui.ico_posix32.setVisible(False)
  731. self.ui.label_posix32.setVisible(False)
  732. if hasNative:
  733. self.ui.ico_native.setPixmap(self.fIconYes)
  734. else:
  735. self.ui.ico_native.setPixmap(self.fIconNo)
  736. self.ui.ch_native.setEnabled(False)
  737. self.ui.ch_sf2.setEnabled(False)
  738. self.ui.ch_sfz.setEnabled(False)
  739. if not hasNonNative:
  740. self.ui.ch_ladspa.setEnabled(False)
  741. self.ui.ch_dssi.setEnabled(False)
  742. self.ui.ch_vst.setEnabled(False)
  743. self.setWindowFlags(self.windowFlags() & ~Qt.WindowContextHelpButtonHint)
  744. # -------------------------------------------------------------------------------------------------------------
  745. # Load settings
  746. self.loadSettings()
  747. # -------------------------------------------------------------------------------------------------------------
  748. # Hide bridges if disabled
  749. # NOTE: We Assume win32 carla build will not run win64 plugins
  750. if (WINDOWS and not kIs64bit) or not host.showPluginBridges:
  751. self.ui.ch_native.setChecked(True)
  752. self.ui.ch_native.setEnabled(False)
  753. self.ui.ch_native.setVisible(False)
  754. self.ui.ch_posix32.setChecked(False)
  755. self.ui.ch_posix32.setEnabled(False)
  756. self.ui.ch_posix32.setVisible(False)
  757. self.ui.ch_posix64.setChecked(False)
  758. self.ui.ch_posix64.setEnabled(False)
  759. self.ui.ch_posix64.setVisible(False)
  760. self.ui.ch_win32.setChecked(False)
  761. self.ui.ch_win32.setEnabled(False)
  762. self.ui.ch_win32.setVisible(False)
  763. self.ui.ch_win64.setChecked(False)
  764. self.ui.ch_win64.setEnabled(False)
  765. self.ui.ch_win64.setVisible(False)
  766. self.ui.ico_posix32.hide()
  767. self.ui.ico_posix64.hide()
  768. self.ui.ico_win32.hide()
  769. self.ui.ico_win64.hide()
  770. self.ui.label_posix32.hide()
  771. self.ui.label_posix64.hide()
  772. self.ui.label_win32.hide()
  773. self.ui.label_win64.hide()
  774. self.ui.sep_format.hide()
  775. elif not (WINDOWS or host.showWineBridges):
  776. self.ui.ch_win32.setChecked(False)
  777. self.ui.ch_win32.setEnabled(False)
  778. self.ui.ch_win32.setVisible(False)
  779. self.ui.ch_win64.setChecked(False)
  780. self.ui.ch_win64.setEnabled(False)
  781. self.ui.ch_win64.setVisible(False)
  782. self.ui.ico_win32.hide()
  783. self.ui.ico_win64.hide()
  784. self.ui.label_win32.hide()
  785. self.ui.label_win64.hide()
  786. # Disable non-supported features
  787. features = gCarla.utils.get_supported_features()
  788. if "sf2" not in features:
  789. self.ui.ch_sf2.setChecked(False)
  790. self.ui.ch_sf2.setEnabled(False)
  791. # -------------------------------------------------------------------------------------------------------------
  792. # Resize to minimum size, as it's very likely UI stuff was hidden
  793. self.resize(self.minimumSize())
  794. # -------------------------------------------------------------------------------------------------------------
  795. # Set-up connections
  796. self.finished.connect(self.slot_saveSettings)
  797. self.ui.b_start.clicked.connect(self.slot_start)
  798. self.ui.b_skip.clicked.connect(self.slot_skip)
  799. self.ui.ch_native.clicked.connect(self.slot_checkTools)
  800. self.ui.ch_posix32.clicked.connect(self.slot_checkTools)
  801. self.ui.ch_posix64.clicked.connect(self.slot_checkTools)
  802. self.ui.ch_win32.clicked.connect(self.slot_checkTools)
  803. self.ui.ch_win64.clicked.connect(self.slot_checkTools)
  804. self.ui.ch_ladspa.clicked.connect(self.slot_checkTools)
  805. self.ui.ch_dssi.clicked.connect(self.slot_checkTools)
  806. self.ui.ch_lv2.clicked.connect(self.slot_checkTools)
  807. self.ui.ch_vst.clicked.connect(self.slot_checkTools)
  808. self.ui.ch_sf2.clicked.connect(self.slot_checkTools)
  809. self.ui.ch_sfz.clicked.connect(self.slot_checkTools)
  810. self.fThread.pluginLook.connect(self.slot_handlePluginLook)
  811. self.fThread.finished.connect(self.slot_handlePluginThreadFinished)
  812. # -------------------------------------------------------------------------------------------------------------
  813. # Post-connect setup
  814. self.slot_checkTools()
  815. # -----------------------------------------------------------------------------------------------------------------
  816. def loadSettings(self):
  817. settings = QSettings("falkTX", "CarlaRefresh2")
  818. check = settings.value("PluginDatabase/SearchLADSPA", True, type=bool) and self.ui.ch_ladspa.isEnabled()
  819. self.ui.ch_ladspa.setChecked(check)
  820. check = settings.value("PluginDatabase/SearchDSSI", True, type=bool) and self.ui.ch_dssi.isEnabled()
  821. self.ui.ch_dssi.setChecked(check)
  822. check = settings.value("PluginDatabase/SearchLV2", True, type=bool) and self.ui.ch_lv2.isEnabled()
  823. self.ui.ch_lv2.setChecked(check)
  824. check = settings.value("PluginDatabase/SearchVST2", True, type=bool) and self.ui.ch_vst.isEnabled()
  825. self.ui.ch_vst.setChecked(check)
  826. check = settings.value("PluginDatabase/SearchSF2", False, type=bool) and self.ui.ch_sf2.isEnabled()
  827. self.ui.ch_sf2.setChecked(check)
  828. check = settings.value("PluginDatabase/SearchSFZ", False, type=bool) and self.ui.ch_sfz.isEnabled()
  829. self.ui.ch_sfz.setChecked(check)
  830. check = settings.value("PluginDatabase/SearchNative", True, type=bool) and self.ui.ch_native.isEnabled()
  831. self.ui.ch_native.setChecked(check)
  832. check = settings.value("PluginDatabase/SearchPOSIX32", False, type=bool) and self.ui.ch_posix32.isEnabled()
  833. self.ui.ch_posix32.setChecked(check)
  834. check = settings.value("PluginDatabase/SearchPOSIX64", False, type=bool) and self.ui.ch_posix64.isEnabled()
  835. self.ui.ch_posix64.setChecked(check)
  836. check = settings.value("PluginDatabase/SearchWin32", False, type=bool) and self.ui.ch_win32.isEnabled()
  837. self.ui.ch_win32.setChecked(check)
  838. check = settings.value("PluginDatabase/SearchWin64", False, type=bool) and self.ui.ch_win64.isEnabled()
  839. self.ui.ch_win64.setChecked(check)
  840. self.ui.ch_do_checks.setChecked(settings.value("PluginDatabase/DoChecks", False, type=bool))
  841. # ------------------------------------------------------------------------------------------------------------------
  842. @pyqtSlot()
  843. def slot_saveSettings(self):
  844. settings = QSettings("falkTX", "CarlaRefresh2")
  845. settings.setValue("PluginDatabase/SearchLADSPA", self.ui.ch_ladspa.isChecked())
  846. settings.setValue("PluginDatabase/SearchDSSI", self.ui.ch_dssi.isChecked())
  847. settings.setValue("PluginDatabase/SearchLV2", self.ui.ch_lv2.isChecked())
  848. settings.setValue("PluginDatabase/SearchVST2", self.ui.ch_vst.isChecked())
  849. settings.setValue("PluginDatabase/SearchSF2", self.ui.ch_sf2.isChecked())
  850. settings.setValue("PluginDatabase/SearchSFZ", self.ui.ch_sfz.isChecked())
  851. settings.setValue("PluginDatabase/SearchNative", self.ui.ch_native.isChecked())
  852. settings.setValue("PluginDatabase/SearchPOSIX32", self.ui.ch_posix32.isChecked())
  853. settings.setValue("PluginDatabase/SearchPOSIX64", self.ui.ch_posix64.isChecked())
  854. settings.setValue("PluginDatabase/SearchWin32", self.ui.ch_win32.isChecked())
  855. settings.setValue("PluginDatabase/SearchWin64", self.ui.ch_win64.isChecked())
  856. settings.setValue("PluginDatabase/DoChecks", self.ui.ch_do_checks.isChecked())
  857. # ------------------------------------------------------------------------------------------------------------------
  858. @pyqtSlot()
  859. def slot_start(self):
  860. self.ui.progressBar.setMinimum(0)
  861. self.ui.progressBar.setMaximum(100)
  862. self.ui.progressBar.setValue(0)
  863. self.ui.b_start.setEnabled(False)
  864. self.ui.b_skip.setVisible(True)
  865. self.ui.b_close.setVisible(False)
  866. self.ui.group_types.setEnabled(False)
  867. self.ui.group_options.setEnabled(False)
  868. if self.ui.ch_do_checks.isChecked():
  869. gCarla.utils.unsetenv("CARLA_DISCOVERY_NO_PROCESSING_CHECKS")
  870. else:
  871. gCarla.utils.setenv("CARLA_DISCOVERY_NO_PROCESSING_CHECKS", "true")
  872. native, posix32, posix64, win32, win64 = (self.ui.ch_native.isChecked(),
  873. self.ui.ch_posix32.isChecked(), self.ui.ch_posix64.isChecked(),
  874. self.ui.ch_win32.isChecked(), self.ui.ch_win64.isChecked())
  875. ladspa, dssi, lv2, vst, sf2, sfz = (self.ui.ch_ladspa.isChecked(), self.ui.ch_dssi.isChecked(),
  876. self.ui.ch_lv2.isChecked(), self.ui.ch_vst.isChecked(),
  877. self.ui.ch_sf2.isChecked(), self.ui.ch_sfz.isChecked())
  878. self.fThread.setSearchBinaryTypes(native, posix32, posix64, win32, win64)
  879. self.fThread.setSearchPluginTypes(ladspa, dssi, lv2, vst, sf2, sfz)
  880. self.fThread.start()
  881. # ------------------------------------------------------------------------------------------------------------------
  882. @pyqtSlot()
  883. def slot_skip(self):
  884. killDiscovery()
  885. # ------------------------------------------------------------------------------------------------------------------
  886. @pyqtSlot()
  887. def slot_checkTools(self):
  888. enabled1 = bool(self.ui.ch_native.isChecked() or
  889. self.ui.ch_posix32.isChecked() or self.ui.ch_posix64.isChecked() or
  890. self.ui.ch_win32.isChecked() or self.ui.ch_win64.isChecked())
  891. enabled2 = bool(self.ui.ch_ladspa.isChecked() or self.ui.ch_dssi.isChecked() or
  892. self.ui.ch_lv2.isChecked() or self.ui.ch_vst.isChecked() or
  893. self.ui.ch_sf2.isChecked() or self.ui.ch_sfz.isChecked())
  894. self.ui.b_start.setEnabled(enabled1 and enabled2)
  895. # ------------------------------------------------------------------------------------------------------------------
  896. @pyqtSlot(int, str)
  897. def slot_handlePluginLook(self, percent, plugin):
  898. self.ui.progressBar.setFormat("%s" % plugin)
  899. self.ui.progressBar.setValue(percent)
  900. # ------------------------------------------------------------------------------------------------------------------
  901. @pyqtSlot()
  902. def slot_handlePluginThreadFinished(self):
  903. self.ui.progressBar.setMinimum(0)
  904. self.ui.progressBar.setMaximum(1)
  905. self.ui.progressBar.setValue(1)
  906. self.ui.progressBar.setFormat(self.tr("Done"))
  907. self.ui.b_start.setEnabled(True)
  908. self.ui.b_skip.setVisible(False)
  909. self.ui.b_close.setVisible(True)
  910. self.ui.group_types.setEnabled(True)
  911. self.ui.group_options.setEnabled(True)
  912. # ------------------------------------------------------------------------------------------------------------------
  913. def closeEvent(self, event):
  914. if self.fThread.isRunning():
  915. self.fThread.stop()
  916. killDiscovery()
  917. #self.fThread.terminate()
  918. self.fThread.wait()
  919. if self.fThread.hasSomethingChanged():
  920. self.accept()
  921. else:
  922. self.reject()
  923. QDialog.closeEvent(self, event)
  924. # ------------------------------------------------------------------------------------------------------------------
  925. def done(self, r):
  926. QDialog.done(self, r)
  927. self.close()
  928. # ----------------------------------------------------------------------------------------------------------------------
  929. # Plugin Database Dialog
  930. class PluginDatabaseW(QDialog):
  931. def __init__(self, parent, host):
  932. QDialog.__init__(self, parent)
  933. self.host = host
  934. self.ui = ui_carla_database.Ui_PluginDatabaseW()
  935. self.ui.setupUi(self)
  936. if False:
  937. # kdevelop likes this :)
  938. host = CarlaHostNull()
  939. self.host = host
  940. # ----------------------------------------------------------------------------------------------------
  941. # Internal stuff
  942. self.fLastTableIndex = 0
  943. self.fRetPlugin = None
  944. self.fRealParent = parent
  945. # ----------------------------------------------------------------------------------------------------
  946. # Set-up GUI
  947. self.ui.b_add.setEnabled(False)
  948. if BINARY_NATIVE in (BINARY_POSIX32, BINARY_WIN32):
  949. self.ui.ch_bridged.setText(self.tr("Bridged (64bit)"))
  950. else:
  951. self.ui.ch_bridged.setText(self.tr("Bridged (32bit)"))
  952. if not (LINUX or MACOS):
  953. self.ui.ch_bridged_wine.setChecked(False)
  954. self.ui.ch_bridged_wine.setEnabled(False)
  955. self.setWindowFlags(self.windowFlags() & ~Qt.WindowContextHelpButtonHint)
  956. # ----------------------------------------------------------------------------------------------------
  957. # Load settings
  958. self.loadSettings()
  959. # ----------------------------------------------------------------------------------------------------
  960. # Disable bridges if not enabled in settings
  961. # NOTE: We Assume win32 carla build will not run win64 plugins
  962. if (WINDOWS and not kIs64bit) or not host.showPluginBridges:
  963. self.ui.ch_native.setChecked(True)
  964. self.ui.ch_native.setEnabled(False)
  965. self.ui.ch_native.setVisible(False)
  966. self.ui.ch_bridged.setChecked(False)
  967. self.ui.ch_bridged.setEnabled(False)
  968. self.ui.ch_bridged.setVisible(False)
  969. self.ui.ch_bridged_wine.setChecked(False)
  970. self.ui.ch_bridged_wine.setEnabled(False)
  971. self.ui.ch_bridged_wine.setVisible(False)
  972. self.ui.l_arch.setVisible(False)
  973. elif not host.showWineBridges:
  974. self.ui.ch_bridged_wine.setChecked(False)
  975. self.ui.ch_bridged_wine.setEnabled(False)
  976. self.ui.ch_bridged_wine.setVisible(False)
  977. # ----------------------------------------------------------------------------------------------------
  978. # Set-up connections
  979. self.finished.connect(self.slot_saveSettings)
  980. self.ui.b_add.clicked.connect(self.slot_addPlugin)
  981. self.ui.b_cancel.clicked.connect(self.reject)
  982. self.ui.b_refresh.clicked.connect(self.slot_refreshPlugins)
  983. self.ui.tb_filters.clicked.connect(self.slot_maybeShowFilters)
  984. self.ui.lineEdit.textChanged.connect(self.slot_checkFilters)
  985. self.ui.tableWidget.currentCellChanged.connect(self.slot_checkPlugin)
  986. self.ui.tableWidget.cellDoubleClicked.connect(self.slot_addPlugin)
  987. self.ui.ch_effects.clicked.connect(self.slot_checkFilters)
  988. self.ui.ch_instruments.clicked.connect(self.slot_checkFilters)
  989. self.ui.ch_midi.clicked.connect(self.slot_checkFilters)
  990. self.ui.ch_other.clicked.connect(self.slot_checkFilters)
  991. self.ui.ch_internal.clicked.connect(self.slot_checkFilters)
  992. self.ui.ch_ladspa.clicked.connect(self.slot_checkFilters)
  993. self.ui.ch_dssi.clicked.connect(self.slot_checkFilters)
  994. self.ui.ch_lv2.clicked.connect(self.slot_checkFilters)
  995. self.ui.ch_vst.clicked.connect(self.slot_checkFilters)
  996. self.ui.ch_kits.clicked.connect(self.slot_checkFilters)
  997. self.ui.ch_native.clicked.connect(self.slot_checkFilters)
  998. self.ui.ch_bridged.clicked.connect(self.slot_checkFilters)
  999. self.ui.ch_bridged_wine.clicked.connect(self.slot_checkFilters)
  1000. self.ui.ch_rtsafe.clicked.connect(self.slot_checkFilters)
  1001. self.ui.ch_gui.clicked.connect(self.slot_checkFilters)
  1002. self.ui.ch_stereo.clicked.connect(self.slot_checkFilters)
  1003. # ----------------------------------------------------------------------------------------------------
  1004. # Post-connect setup
  1005. self._reAddPlugins()
  1006. # --------------------------------------------------------------------------------------------------------
  1007. @pyqtSlot()
  1008. def slot_addPlugin(self):
  1009. if self.ui.tableWidget.currentRow() >= 0:
  1010. self.fRetPlugin = self.ui.tableWidget.item(self.ui.tableWidget.currentRow(), 0).data(Qt.UserRole)
  1011. self.accept()
  1012. else:
  1013. self.reject()
  1014. @pyqtSlot(int)
  1015. def slot_checkPlugin(self, row):
  1016. self.ui.b_add.setEnabled(row >= 0)
  1017. @pyqtSlot()
  1018. def slot_checkFilters(self):
  1019. self._checkFilters()
  1020. @pyqtSlot()
  1021. def slot_maybeShowFilters(self):
  1022. self._showFilters(not self.ui.frame.isVisible())
  1023. @pyqtSlot()
  1024. def slot_refreshPlugins(self):
  1025. if PluginRefreshW(self, self.host).exec_():
  1026. self._reAddPlugins()
  1027. if self.fRealParent:
  1028. self.fRealParent.setLoadRDFsNeeded()
  1029. # --------------------------------------------------------------------------------------------------------
  1030. @pyqtSlot()
  1031. def slot_saveSettings(self):
  1032. settings = QSettings("falkTX", "CarlaDatabase2")
  1033. settings.setValue("PluginDatabase/Geometry", self.saveGeometry())
  1034. settings.setValue("PluginDatabase/TableGeometry_5", self.ui.tableWidget.horizontalHeader().saveState())
  1035. settings.setValue("PluginDatabase/ShowFilters", (self.ui.tb_filters.arrowType() == Qt.UpArrow))
  1036. settings.setValue("PluginDatabase/ShowEffects", self.ui.ch_effects.isChecked())
  1037. settings.setValue("PluginDatabase/ShowInstruments", self.ui.ch_instruments.isChecked())
  1038. settings.setValue("PluginDatabase/ShowMIDI", self.ui.ch_midi.isChecked())
  1039. settings.setValue("PluginDatabase/ShowOther", self.ui.ch_other.isChecked())
  1040. settings.setValue("PluginDatabase/ShowInternal", self.ui.ch_internal.isChecked())
  1041. settings.setValue("PluginDatabase/ShowLADSPA", self.ui.ch_ladspa.isChecked())
  1042. settings.setValue("PluginDatabase/ShowDSSI", self.ui.ch_dssi.isChecked())
  1043. settings.setValue("PluginDatabase/ShowLV2", self.ui.ch_lv2.isChecked())
  1044. settings.setValue("PluginDatabase/ShowVST2", self.ui.ch_vst.isChecked())
  1045. settings.setValue("PluginDatabase/ShowKits", self.ui.ch_kits.isChecked())
  1046. settings.setValue("PluginDatabase/ShowNative", self.ui.ch_native.isChecked())
  1047. settings.setValue("PluginDatabase/ShowBridged", self.ui.ch_bridged.isChecked())
  1048. settings.setValue("PluginDatabase/ShowBridgedWine", self.ui.ch_bridged_wine.isChecked())
  1049. settings.setValue("PluginDatabase/ShowRtSafe", self.ui.ch_rtsafe.isChecked())
  1050. settings.setValue("PluginDatabase/ShowHasGUI", self.ui.ch_gui.isChecked())
  1051. settings.setValue("PluginDatabase/ShowStereoOnly", self.ui.ch_stereo.isChecked())
  1052. settings.setValue("PluginDatabase/SearchText", self.ui.lineEdit.text())
  1053. # --------------------------------------------------------------------------------------------------------
  1054. def loadSettings(self):
  1055. settings = QSettings("falkTX", "CarlaDatabase2")
  1056. self.restoreGeometry(settings.value("PluginDatabase/Geometry", b""))
  1057. self.ui.ch_effects.setChecked(settings.value("PluginDatabase/ShowEffects", True, type=bool))
  1058. self.ui.ch_instruments.setChecked(settings.value("PluginDatabase/ShowInstruments", True, type=bool))
  1059. self.ui.ch_midi.setChecked(settings.value("PluginDatabase/ShowMIDI", True, type=bool))
  1060. self.ui.ch_other.setChecked(settings.value("PluginDatabase/ShowOther", True, type=bool))
  1061. self.ui.ch_internal.setChecked(settings.value("PluginDatabase/ShowInternal", True, type=bool))
  1062. self.ui.ch_ladspa.setChecked(settings.value("PluginDatabase/ShowLADSPA", True, type=bool))
  1063. self.ui.ch_dssi.setChecked(settings.value("PluginDatabase/ShowDSSI", True, type=bool))
  1064. self.ui.ch_lv2.setChecked(settings.value("PluginDatabase/ShowLV2", True, type=bool))
  1065. self.ui.ch_vst.setChecked(settings.value("PluginDatabase/ShowVST2", True, type=bool))
  1066. self.ui.ch_kits.setChecked(settings.value("PluginDatabase/ShowKits", True, type=bool))
  1067. self.ui.ch_native.setChecked(settings.value("PluginDatabase/ShowNative", True, type=bool))
  1068. self.ui.ch_bridged.setChecked(settings.value("PluginDatabase/ShowBridged", True, type=bool))
  1069. self.ui.ch_bridged_wine.setChecked(settings.value("PluginDatabase/ShowBridgedWine", True, type=bool))
  1070. self.ui.ch_rtsafe.setChecked(settings.value("PluginDatabase/ShowRtSafe", False, type=bool))
  1071. self.ui.ch_gui.setChecked(settings.value("PluginDatabase/ShowHasGUI", False, type=bool))
  1072. self.ui.ch_stereo.setChecked(settings.value("PluginDatabase/ShowStereoOnly", False, type=bool))
  1073. self.ui.lineEdit.setText(settings.value("PluginDatabase/SearchText", "", type=str))
  1074. tableGeometry = settings.value("PluginDatabase/TableGeometry_5")
  1075. if tableGeometry:
  1076. self.ui.tableWidget.horizontalHeader().restoreState(tableGeometry)
  1077. else:
  1078. self.ui.tableWidget.sortByColumn(0, Qt.AscendingOrder)
  1079. self._showFilters(settings.value("PluginDatabase/ShowFilters", False, type=bool))
  1080. # --------------------------------------------------------------------------------------------------------
  1081. def _checkFilters(self):
  1082. text = self.ui.lineEdit.text().lower()
  1083. hideEffects = not self.ui.ch_effects.isChecked()
  1084. hideInstruments = not self.ui.ch_instruments.isChecked()
  1085. hideMidi = not self.ui.ch_midi.isChecked()
  1086. hideOther = not self.ui.ch_other.isChecked()
  1087. hideInternal = not self.ui.ch_internal.isChecked()
  1088. hideLadspa = not self.ui.ch_ladspa.isChecked()
  1089. hideDssi = not self.ui.ch_dssi.isChecked()
  1090. hideLV2 = not self.ui.ch_lv2.isChecked()
  1091. hideVST2 = not self.ui.ch_vst.isChecked()
  1092. hideKits = not self.ui.ch_kits.isChecked()
  1093. hideNative = not self.ui.ch_native.isChecked()
  1094. hideBridged = not self.ui.ch_bridged.isChecked()
  1095. hideBridgedWine = not self.ui.ch_bridged_wine.isChecked()
  1096. hideNonRtSafe = self.ui.ch_rtsafe.isChecked()
  1097. hideNonGui = self.ui.ch_gui.isChecked()
  1098. hideNonStereo = self.ui.ch_stereo.isChecked()
  1099. if HAIKU or LINUX or MACOS:
  1100. nativeBins = [BINARY_POSIX32, BINARY_POSIX64]
  1101. wineBins = [BINARY_WIN32, BINARY_WIN64]
  1102. elif WINDOWS:
  1103. nativeBins = [BINARY_WIN32, BINARY_WIN64]
  1104. wineBins = []
  1105. else:
  1106. nativeBins = []
  1107. wineBins = []
  1108. rowCount = self.ui.tableWidget.rowCount()
  1109. for i in range(self.fLastTableIndex):
  1110. plugin = self.ui.tableWidget.item(i, 0).data(Qt.UserRole)
  1111. aIns = plugin['audio.ins']
  1112. aOuts = plugin['audio.outs']
  1113. mIns = plugin['midi.ins']
  1114. mOuts = plugin['midi.outs']
  1115. ptype = self.ui.tableWidget.item(i, 12).text()
  1116. isSynth = bool(plugin['hints'] & PLUGIN_IS_SYNTH)
  1117. isEffect = bool(aIns > 0 < aOuts and not isSynth)
  1118. isMidi = bool(aIns == 0 and aOuts == 0 and mIns > 0 < mOuts)
  1119. isKit = bool(ptype in ("SF2", "SFZ"))
  1120. isOther = bool(not (isEffect or isSynth or isMidi or isKit))
  1121. isNative = bool(plugin['build'] == BINARY_NATIVE)
  1122. isRtSafe = bool(plugin['hints'] & PLUGIN_IS_RTSAFE)
  1123. isStereo = bool(aIns == 2 and aOuts == 2) or (isSynth and aOuts == 2)
  1124. hasGui = bool(plugin['hints'] & PLUGIN_HAS_CUSTOM_UI)
  1125. isBridged = bool(not isNative and plugin['build'] in nativeBins)
  1126. isBridgedWine = bool(not isNative and plugin['build'] in wineBins)
  1127. if hideEffects and isEffect:
  1128. self.ui.tableWidget.hideRow(i)
  1129. elif hideInstruments and isSynth:
  1130. self.ui.tableWidget.hideRow(i)
  1131. elif hideMidi and isMidi:
  1132. self.ui.tableWidget.hideRow(i)
  1133. elif hideOther and isOther:
  1134. self.ui.tableWidget.hideRow(i)
  1135. elif hideKits and isKit:
  1136. self.ui.tableWidget.hideRow(i)
  1137. elif hideInternal and ptype == self.tr("Internal"):
  1138. self.ui.tableWidget.hideRow(i)
  1139. elif hideLadspa and ptype == "LADSPA":
  1140. self.ui.tableWidget.hideRow(i)
  1141. elif hideDssi and ptype == "DSSI":
  1142. self.ui.tableWidget.hideRow(i)
  1143. elif hideLV2 and ptype == "LV2":
  1144. self.ui.tableWidget.hideRow(i)
  1145. elif hideVST2 and ptype == "VST2":
  1146. self.ui.tableWidget.hideRow(i)
  1147. elif hideNative and isNative:
  1148. self.ui.tableWidget.hideRow(i)
  1149. elif hideBridged and isBridged:
  1150. self.ui.tableWidget.hideRow(i)
  1151. elif hideBridgedWine and isBridgedWine:
  1152. self.ui.tableWidget.hideRow(i)
  1153. elif hideNonRtSafe and not isRtSafe:
  1154. self.ui.tableWidget.hideRow(i)
  1155. elif hideNonGui and not hasGui:
  1156. self.ui.tableWidget.hideRow(i)
  1157. elif hideNonStereo and not isStereo:
  1158. self.ui.tableWidget.hideRow(i)
  1159. elif (text and not (
  1160. text in self.ui.tableWidget.item(i, 0).text().lower() or
  1161. text in self.ui.tableWidget.item(i, 1).text().lower() or
  1162. text in self.ui.tableWidget.item(i, 2).text().lower() or
  1163. text in self.ui.tableWidget.item(i, 3).text().lower() or
  1164. text in self.ui.tableWidget.item(i, 13).text().lower())):
  1165. self.ui.tableWidget.hideRow(i)
  1166. else:
  1167. self.ui.tableWidget.showRow(i)
  1168. # --------------------------------------------------------------------------------------------------------
  1169. def _showFilters(self, yesNo):
  1170. self.ui.tb_filters.setArrowType(Qt.UpArrow if yesNo else Qt.DownArrow)
  1171. self.ui.frame.setVisible(yesNo)
  1172. # --------------------------------------------------------------------------------------------------------
  1173. def _addPluginToTable(self, plugin, ptype):
  1174. if plugin['API'] != PLUGIN_QUERY_API_VERSION and ptype == self.tr("Internal"):
  1175. return
  1176. if ptype in (self.tr("Internal"), "LV2", "SF2", "SFZ"):
  1177. plugin['build'] = BINARY_NATIVE
  1178. index = self.fLastTableIndex
  1179. if plugin['build'] == BINARY_NATIVE:
  1180. bridgeText = self.tr("No")
  1181. else:
  1182. if WINDOWS:
  1183. if plugin['build'] == BINARY_WIN32:
  1184. typeText = "32bit"
  1185. elif plugin['build'] == BINARY_WIN64:
  1186. typeText = "64bit"
  1187. else:
  1188. typeText = self.tr("Unknown")
  1189. else:
  1190. if plugin['build'] == BINARY_POSIX32:
  1191. typeText = "32bit"
  1192. elif plugin['build'] == BINARY_POSIX64:
  1193. typeText = "64bit"
  1194. elif plugin['build'] == BINARY_WIN32:
  1195. typeText = "Windows 32bit"
  1196. elif plugin['build'] == BINARY_WIN64:
  1197. typeText = "Windows 64bit"
  1198. else:
  1199. typeText = self.tr("Unknown")
  1200. bridgeText = self.tr("Yes (%s)" % typeText)
  1201. self.ui.tableWidget.setItem(index, 0, QTableWidgetItem(str(plugin['name'])))
  1202. self.ui.tableWidget.setItem(index, 1, QTableWidgetItem(str(plugin['label'])))
  1203. self.ui.tableWidget.setItem(index, 2, QTableWidgetItem(str(plugin['maker'])))
  1204. self.ui.tableWidget.setItem(index, 3, QTableWidgetItem(str(plugin['uniqueId'])))
  1205. self.ui.tableWidget.setItem(index, 4, QTableWidgetItem(str(plugin['audio.ins'])))
  1206. self.ui.tableWidget.setItem(index, 5, QTableWidgetItem(str(plugin['audio.outs'])))
  1207. self.ui.tableWidget.setItem(index, 6, QTableWidgetItem(str(plugin['parameters.ins'])))
  1208. self.ui.tableWidget.setItem(index, 7, QTableWidgetItem(str(plugin['parameters.outs'])))
  1209. self.ui.tableWidget.setItem(index, 9, QTableWidgetItem(self.tr("Yes") if (plugin['hints'] & PLUGIN_HAS_CUSTOM_UI) else self.tr("No")))
  1210. self.ui.tableWidget.setItem(index, 10, QTableWidgetItem(self.tr("Yes") if (plugin['hints'] & PLUGIN_IS_SYNTH) else self.tr("No")))
  1211. self.ui.tableWidget.setItem(index, 11, QTableWidgetItem(bridgeText))
  1212. self.ui.tableWidget.setItem(index, 12, QTableWidgetItem(ptype))
  1213. self.ui.tableWidget.setItem(index, 13, QTableWidgetItem(str(plugin['filename'])))
  1214. self.ui.tableWidget.item(index, 0).setData(Qt.UserRole, plugin)
  1215. self.fLastTableIndex += 1
  1216. # --------------------------------------------------------------------------------------------------------
  1217. def _reAddInternalHelper(self, settingsDB, ptype, path):
  1218. if ptype == PLUGIN_INTERNAL:
  1219. ptypeStr = "Internal"
  1220. ptypeStrTr = self.tr("Internal")
  1221. elif ptype == PLUGIN_LV2:
  1222. ptypeStr = "LV2"
  1223. ptypeStrTr = ptypeStr
  1224. #elif ptype == PLUGIN_SFZ:
  1225. #ptypeStr = "SFZ"
  1226. #ptypeStrTr = ptypeStr
  1227. else:
  1228. return 0
  1229. plugins = toList(settingsDB.value("Plugins/" + ptypeStr, []))
  1230. pluginCount = settingsDB.value("PluginCount/" + ptypeStr, 0, type=int)
  1231. pluginCountNew = gCarla.utils.get_cached_plugin_count(ptype, path)
  1232. if pluginCountNew != pluginCount or (len(plugins) > 0 and plugins[0]['API'] != PLUGIN_QUERY_API_VERSION):
  1233. plugins = []
  1234. pluginCount = pluginCountNew
  1235. QApplication.processEvents(QEventLoop.ExcludeUserInputEvents, 50)
  1236. for i in range(pluginCountNew):
  1237. descInfo = gCarla.utils.get_cached_plugin_info(ptype, i)
  1238. if not descInfo['valid']:
  1239. continue
  1240. info = checkPluginCached(descInfo, ptype)
  1241. plugins.append(info)
  1242. if i % 50 == 0:
  1243. QApplication.processEvents(QEventLoop.ExcludeUserInputEvents, 50)
  1244. settingsDB.setValue("Plugins/" + ptypeStr, plugins)
  1245. settingsDB.setValue("PluginCount/" + ptypeStr, pluginCount)
  1246. # prepare rows in advance
  1247. self.ui.tableWidget.setRowCount(self.fLastTableIndex + len(plugins))
  1248. for plugin in plugins:
  1249. self._addPluginToTable(plugin, ptypeStrTr)
  1250. return pluginCount
  1251. def _reAddPlugins(self):
  1252. settingsDB = QSettings("falkTX", "CarlaPlugins4")
  1253. for x in range(self.ui.tableWidget.rowCount()):
  1254. self.ui.tableWidget.removeRow(0)
  1255. self.fLastTableIndex = 0
  1256. self.ui.tableWidget.setSortingEnabled(False)
  1257. settings = QSettings("falkTX", "Carla2")
  1258. LV2_PATH = splitter.join(toList(settings.value(CARLA_KEY_PATHS_LV2, CARLA_DEFAULT_LV2_PATH)))
  1259. del settings
  1260. # ----------------------------------------------------------------------------------------------------
  1261. # plugins handled through backend
  1262. internalCount = self._reAddInternalHelper(settingsDB, PLUGIN_INTERNAL, "")
  1263. lv2Count = self._reAddInternalHelper(settingsDB, PLUGIN_LV2, LV2_PATH)
  1264. # ----------------------------------------------------------------------------------------------------
  1265. # LADSPA
  1266. ladspaPlugins = []
  1267. ladspaPlugins += toList(settingsDB.value("Plugins/LADSPA_native", []))
  1268. ladspaPlugins += toList(settingsDB.value("Plugins/LADSPA_posix32", []))
  1269. ladspaPlugins += toList(settingsDB.value("Plugins/LADSPA_posix64", []))
  1270. ladspaPlugins += toList(settingsDB.value("Plugins/LADSPA_win32", []))
  1271. ladspaPlugins += toList(settingsDB.value("Plugins/LADSPA_win64", []))
  1272. # ----------------------------------------------------------------------------------------------------
  1273. # DSSI
  1274. dssiPlugins = []
  1275. dssiPlugins += toList(settingsDB.value("Plugins/DSSI_native", []))
  1276. dssiPlugins += toList(settingsDB.value("Plugins/DSSI_posix32", []))
  1277. dssiPlugins += toList(settingsDB.value("Plugins/DSSI_posix64", []))
  1278. dssiPlugins += toList(settingsDB.value("Plugins/DSSI_win32", []))
  1279. dssiPlugins += toList(settingsDB.value("Plugins/DSSI_win64", []))
  1280. # ----------------------------------------------------------------------------------------------------
  1281. # VST2
  1282. vst2Plugins = []
  1283. vst2Plugins += toList(settingsDB.value("Plugins/VST2_native", []))
  1284. vst2Plugins += toList(settingsDB.value("Plugins/VST2_posix32", []))
  1285. vst2Plugins += toList(settingsDB.value("Plugins/VST2_posix64", []))
  1286. vst2Plugins += toList(settingsDB.value("Plugins/VST2_win32", []))
  1287. vst2Plugins += toList(settingsDB.value("Plugins/VST2_win64", []))
  1288. # ----------------------------------------------------------------------------------------------------
  1289. # Kits
  1290. sf2s = toList(settingsDB.value("Plugins/SF2", []))
  1291. sfzs = toList(settingsDB.value("Plugins/SFZ", []))
  1292. # ----------------------------------------------------------------------------------------------------
  1293. # count plugins first, so we can create rows in advance
  1294. ladspaCount = 0
  1295. dssiCount = 0
  1296. vstCount = 0
  1297. sf2Count = 0
  1298. sfzCount = 0
  1299. for plugins in ladspaPlugins:
  1300. ladspaCount += len(plugins)
  1301. for plugins in dssiPlugins:
  1302. dssiCount += len(plugins)
  1303. for plugins in vst2Plugins:
  1304. vstCount += len(plugins)
  1305. for plugins in sf2s:
  1306. sf2Count += len(plugins)
  1307. sfzCount += len(sfzs)
  1308. self.ui.tableWidget.setRowCount(self.fLastTableIndex+ladspaCount+dssiCount+vstCount+sf2Count+sfzCount)
  1309. self.ui.label.setText(self.tr("Have %i Internal, %i LADSPA, %i DSSI, %i LV2 and %i VST plugins, plus %i Sound Kits" % (
  1310. internalCount, ladspaCount, dssiCount, lv2Count, vstCount, sf2Count+sfzCount)))
  1311. # ----------------------------------------------------------------------------------------------------
  1312. # now add all plugins to the table
  1313. for plugins in ladspaPlugins:
  1314. for plugin in plugins:
  1315. self._addPluginToTable(plugin, "LADSPA")
  1316. for plugins in dssiPlugins:
  1317. for plugin in plugins:
  1318. self._addPluginToTable(plugin, "DSSI")
  1319. for plugins in vst2Plugins:
  1320. for plugin in plugins:
  1321. self._addPluginToTable(plugin, "VST2")
  1322. for sf2 in sf2s:
  1323. for sf2_i in sf2:
  1324. self._addPluginToTable(sf2_i, "SF2")
  1325. for sfz in sfzs:
  1326. self._addPluginToTable(sfz, "SFZ")
  1327. # ----------------------------------------------------------------------------------------------------
  1328. self.ui.tableWidget.setSortingEnabled(True)
  1329. self._checkFilters()
  1330. # --------------------------------------------------------------------------------------------------------
  1331. def done(self, r):
  1332. QDialog.done(self, r)
  1333. self.close()
  1334. # ----------------------------------------------------------------------------------------------------------------------
  1335. # Jack Application Dialog
  1336. class JackApplicationW(QDialog):
  1337. SESSION_MGR_NONE = 0
  1338. SESSION_MGR_AUTO = 1
  1339. SESSION_MGR_JACK = 2
  1340. SESSION_MGR_LADISH = 3
  1341. SESSION_MGR_NSM = 4
  1342. FLAG_CONTROL_WINDOW = 0x01
  1343. FLAG_CAPTURE_FIRST_WINDOW = 0x02
  1344. FLAG_BUFFERS_ADDITION_MODE = 0x10
  1345. def __init__(self, parent, host):
  1346. QDialog.__init__(self, parent)
  1347. self.host = host
  1348. self.ui = ui_carla_add_jack.Ui_Dialog()
  1349. self.ui.setupUi(self)
  1350. if False:
  1351. # kdevelop likes this :)
  1352. self.host = host = CarlaHostNull()
  1353. self.setWindowFlags(self.windowFlags() & ~Qt.WindowContextHelpButtonHint)
  1354. # --------------------------------------------------------------------------------------------------------------
  1355. # Load settings
  1356. self.loadSettings()
  1357. # --------------------------------------------------------------------------------------------------------------
  1358. # Set-up connections
  1359. self.finished.connect(self.slot_saveSettings)
  1360. self.ui.le_command.textChanged.connect(self.slot_commandChanged)
  1361. # ------------------------------------------------------------------------------------------------------------------
  1362. def getCommandAndFlags(self):
  1363. name = self.ui.le_name.text()
  1364. command = self.ui.le_command.text()
  1365. smgr = self.SESSION_MGR_NONE
  1366. flags = 0x0
  1367. if not name:
  1368. name = os.path.basename(command.split(" ",1)[0])
  1369. # TODO finalize flag definitions
  1370. uiSessionMgrIndex = self.ui.cb_session_mgr.currentIndex()
  1371. if uiSessionMgrIndex == 1:
  1372. smgr = self.SESSION_MGR_AUTO
  1373. elif uiSessionMgrIndex == 2:
  1374. smgr = self.SESSION_MGR_LADISH
  1375. #elif uiSessionMgrIndex == 2:
  1376. #smgr = self.SESSION_MGR_NSM
  1377. if self.ui.cb_manage_window.isChecked():
  1378. flags |= self.FLAG_CONTROL_WINDOW
  1379. if self.ui.cb_capture_first_window.isChecked():
  1380. flags |= self.FLAG_CAPTURE_FIRST_WINDOW
  1381. if self.ui.cb_buffers_addition_mode.isChecked():
  1382. flags |= self.FLAG_BUFFERS_ADDITION_MODE
  1383. baseIntVal = ord('0')
  1384. labelSetup = "%s%s%s%s%s%s" % (chr(baseIntVal+self.ui.sb_audio_ins.value()),
  1385. chr(baseIntVal+self.ui.sb_audio_outs.value()),
  1386. chr(baseIntVal+self.ui.sb_midi_ins.value()),
  1387. chr(baseIntVal+self.ui.sb_midi_outs.value()),
  1388. chr(baseIntVal+smgr),
  1389. chr(baseIntVal+flags))
  1390. return (command, name, labelSetup)
  1391. def loadSettings(self):
  1392. settings = QSettings("falkTX", "CarlaAddJackApp")
  1393. command = settings.value("Command", "", type=str)
  1394. self.ui.le_command.setText(command)
  1395. self.ui.buttonBox.button(QDialogButtonBox.Ok).setEnabled(len(command) > 0)
  1396. self.ui.le_name.setText(settings.value("Name", "", type=str))
  1397. self.ui.sb_audio_ins.setValue(settings.value("NumAudioIns", 2, type=int))
  1398. self.ui.sb_audio_outs.setValue(settings.value("NumAudioOuts", 2, type=int))
  1399. self.ui.sb_midi_ins.setValue(settings.value("NumMidiIns", 0, type=int))
  1400. self.ui.sb_midi_outs.setValue(settings.value("NumMidiOuts", 0, type=int))
  1401. self.ui.cb_manage_window.setChecked(settings.value("ManageWindow", True, type=bool))
  1402. # ------------------------------------------------------------------------------------------------------------------
  1403. @pyqtSlot(str)
  1404. def slot_commandChanged(self, text):
  1405. self.ui.buttonBox.button(QDialogButtonBox.Ok).setEnabled(len(text) > 0)
  1406. @pyqtSlot()
  1407. def slot_saveSettings(self):
  1408. settings = QSettings("falkTX", "CarlaAddJackApp")
  1409. settings.setValue("Command", self.ui.le_command.text())
  1410. settings.setValue("Name", self.ui.le_name.text())
  1411. settings.setValue("NumAudioIns", self.ui.sb_audio_ins.value())
  1412. settings.setValue("NumAudioOuts", self.ui.sb_audio_outs.value())
  1413. settings.setValue("NumMidiIns", self.ui.sb_midi_ins.value())
  1414. settings.setValue("NumMidiOuts", self.ui.sb_midi_outs.value())
  1415. settings.setValue("ManageWindow", self.ui.cb_manage_window.isChecked())
  1416. # ------------------------------------------------------------------------------------------------------------------
  1417. def done(self, r):
  1418. QDialog.done(self, r)
  1419. self.close()
  1420. # ----------------------------------------------------------------------------------------------------------------------
  1421. # Main
  1422. if __name__ == '__main__':
  1423. from carla_app import CarlaApplication
  1424. from carla_host import initHost, loadHostSettings
  1425. initName, libPrefix = handleInitialCommandLineArguments(__file__ if "__file__" in dir() else None)
  1426. app = CarlaApplication("Carla2-Database", libPrefix)
  1427. host = initHost("Carla2-Database", libPrefix, False, False, False)
  1428. loadHostSettings(host)
  1429. gui = PluginDatabaseW(None, host)
  1430. gui.show()
  1431. app.exit_exec()
  1432. # ------------------------------------------------------------------------------------------------------------