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.

carla_database.py 72KB

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