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.

2109 lines
84KB

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