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.

3082 lines
117KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Common Carla code
  4. # Copyright (C) 2011-2013 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 GPL.txt file
  17. # ------------------------------------------------------------------------------------------------------------
  18. # Imports (Global)
  19. import os
  20. import json
  21. import platform
  22. import sys
  23. from codecs import open as codecopen
  24. from copy import deepcopy
  25. #from decimal import Decimal
  26. from subprocess import Popen, PIPE
  27. from PyQt4.QtCore import pyqtSlot, qWarning, Qt, QByteArray, QSettings, QThread, QTimer, SIGNAL, SLOT
  28. from PyQt4.QtGui import QColor, QDialog, QIcon, QFontMetrics, QFrame, QMessageBox
  29. from PyQt4.QtGui import QPainter, QPainterPath, QTableWidgetItem, QVBoxLayout, QWidget
  30. #from PyQt4.QtGui import QCursor, QGraphicsScene, QInputDialog, QLinearGradient, QMenu,
  31. #from PyQt4.QtXml import QDomDocument
  32. # ------------------------------------------------------------------------------------------------------------
  33. # Imports (Custom)
  34. import ui_carla_about
  35. import ui_carla_database
  36. import ui_carla_edit
  37. import ui_carla_parameter
  38. import ui_carla_plugin
  39. import ui_carla_refresh
  40. # ------------------------------------------------------------------------------------------------------------
  41. # Try Import LADSPA-RDF
  42. try:
  43. import ladspa_rdf
  44. haveLRDF = True
  45. except:
  46. print("LRDF Support not available (LADSPA-RDF will be disabled)")
  47. haveLRDF = False
  48. # ------------------------------------------------------------------------------------------------------------
  49. # Try Import Signal
  50. try:
  51. from signal import signal, SIGINT, SIGTERM, SIGUSR1, SIGUSR2
  52. haveSignal = True
  53. except:
  54. haveSignal = False
  55. # ------------------------------------------------------------------------------------------------------------
  56. # Set Platform
  57. if sys.platform == "darwin":
  58. from PyQt4.QtGui import qt_mac_set_menubar_icons
  59. qt_mac_set_menubar_icons(False)
  60. HAIKU = False
  61. LINUX = False
  62. MACOS = True
  63. WINDOWS = False
  64. elif "haiku" in sys.platform:
  65. HAIKU = True
  66. LINUX = False
  67. MACOS = False
  68. WINDOWS = False
  69. elif "linux" in sys.platform:
  70. HAIKU = False
  71. LINUX = True
  72. MACOS = False
  73. WINDOWS = False
  74. elif sys.platform in ("win32", "win64", "cygwin"):
  75. WINDIR = os.getenv("WINDIR")
  76. HAIKU = False
  77. LINUX = False
  78. MACOS = False
  79. WINDOWS = True
  80. else:
  81. HAIKU = False
  82. LINUX = False
  83. MACOS = False
  84. WINDOWS = False
  85. # ------------------------------------------------------------------------------------------------------------
  86. # Set Version
  87. VERSION = "0.5.0"
  88. # ------------------------------------------------------------------------------------------------------------
  89. # Set TMP
  90. TMP = os.getenv("TMP")
  91. if TMP is None:
  92. if WINDOWS:
  93. qWarning("TMP variable not set")
  94. TMP = os.path.join(WINDIR, "temp")
  95. else:
  96. TMP = "/tmp"
  97. # ------------------------------------------------------------------------------------------------------------
  98. # Set HOME
  99. HOME = os.getenv("HOME")
  100. if HOME is None:
  101. HOME = os.path.expanduser("~")
  102. if LINUX or MACOS:
  103. qWarning("HOME variable not set")
  104. if not os.path.exists(HOME):
  105. qWarning("HOME does not exist")
  106. HOME = TMP
  107. # ------------------------------------------------------------------------------------------------------------
  108. # Set PATH
  109. PATH = os.getenv("PATH")
  110. if PATH is None:
  111. qWarning("PATH variable not set")
  112. if MACOS:
  113. PATH = ("/opt/local/bin", "/usr/local/bin", "/usr/bin", "/bin")
  114. elif WINDOWS:
  115. PATH = (os.path.join(WINDIR, "system32"), WINDIR)
  116. else:
  117. PATH = ("/usr/local/bin", "/usr/bin", "/bin")
  118. else:
  119. PATH = PATH.split(os.pathsep)
  120. # ------------------------------------------------------------------------------------------------------------
  121. # 64bit check
  122. kIs64bit = bool(platform.architecture()[0] == "64bit" and sys.maxsize > 2**32)
  123. # ------------------------------------------------------------------------------------------------
  124. # Backend defines
  125. MAX_DEFAULT_PLUGINS = 99
  126. MAX_RACK_PLUGINS = 16
  127. MAX_PATCHBAY_PLUGINS = 999
  128. MAX_DEFAULT_PARAMETERS = 200
  129. # Plugin Hints
  130. PLUGIN_IS_BRIDGE = 0x001
  131. PLUGIN_IS_RTSAFE = 0x002
  132. PLUGIN_IS_SYNTH = 0x004
  133. PLUGIN_HAS_GUI = 0x010
  134. PLUGIN_USES_CHUNKS = 0x020
  135. PLUGIN_USES_SINGLE_THREAD = 0x040
  136. PLUGIN_CAN_DRYWET = 0x100
  137. PLUGIN_CAN_VOLUME = 0x200
  138. PLUGIN_CAN_BALANCE = 0x400
  139. PLUGIN_CAN_FORCE_STEREO = 0x800
  140. # Plugin Options
  141. PLUGIN_OPTION_FIXED_BUFFER = 0x001
  142. PLUGIN_OPTION_FORCE_STEREO = 0x002
  143. PLUGIN_OPTION_SELF_AUTOMATION = 0x004
  144. PLUGIN_OPTION_USE_CHUNKS = 0x008
  145. PLUGIN_OPTION_SEND_ALL_SOUND_OFF = 0x010
  146. PLUGIN_OPTION_SEND_NOTE_OFF_VELO = 0x020
  147. PLUGIN_OPTION_SEND_NOTE_AFTERTOUCH = 0x040
  148. PLUGIN_OPTION_SEND_PITCHBEND = 0x080
  149. PLUGIN_OPTION_VST_SUPPLY_IDLE = 0x100
  150. PLUGIN_OPTION_VST_UPDATE_DISPLAY = 0x200
  151. # Parameter Hints
  152. PARAMETER_IS_BOOLEAN = 0x01
  153. PARAMETER_IS_INTEGER = 0x02
  154. PARAMETER_IS_LOGARITHMIC = 0x04
  155. PARAMETER_IS_ENABLED = 0x08
  156. PARAMETER_IS_AUTOMABLE = 0x10
  157. PARAMETER_USES_SAMPLERATE = 0x20
  158. PARAMETER_USES_SCALEPOINTS = 0x40
  159. PARAMETER_USES_CUSTOM_TEXT = 0x80
  160. # FIXME
  161. # Custom Data types
  162. #CUSTOM_DATA_INVALID = None
  163. #CUSTOM_DATA_CHUNK = "http://kxstudio.sf.net/ns/carla/chunk"
  164. #CUSTOM_DATA_STRING = "http://kxstudio.sf.net/ns/carla/string"
  165. # Binary Type
  166. BINARY_NONE = 0
  167. BINARY_POSIX32 = 1
  168. BINARY_POSIX64 = 2
  169. BINARY_WIN32 = 3
  170. BINARY_WIN64 = 4
  171. BINARY_OTHER = 5
  172. # Plugin Type
  173. PLUGIN_NONE = 0
  174. PLUGIN_INTERNAL = 1
  175. PLUGIN_LADSPA = 2
  176. PLUGIN_DSSI = 3
  177. PLUGIN_LV2 = 4
  178. PLUGIN_VST = 5
  179. PLUGIN_GIG = 6
  180. PLUGIN_SF2 = 7
  181. PLUGIN_SFZ = 8
  182. # Plugin Category
  183. PLUGIN_CATEGORY_NONE = 0
  184. PLUGIN_CATEGORY_SYNTH = 1
  185. PLUGIN_CATEGORY_DELAY = 2 # also Reverb
  186. PLUGIN_CATEGORY_EQ = 3
  187. PLUGIN_CATEGORY_FILTER = 4
  188. PLUGIN_CATEGORY_DYNAMICS = 5 # Amplifier, Compressor, Gate
  189. PLUGIN_CATEGORY_MODULATOR = 6 # Chorus, Flanger, Phaser
  190. PLUGIN_CATEGORY_UTILITY = 7 # Analyzer, Converter, Mixer
  191. PLUGIN_CATEGORY_OTHER = 8 # used to check if a plugin has a category
  192. # Parameter Type
  193. PARAMETER_UNKNOWN = 0
  194. PARAMETER_INPUT = 1
  195. PARAMETER_OUTPUT = 2
  196. PARAMETER_LATENCY = 3
  197. PARAMETER_SAMPLE_RATE = 4
  198. PARAMETER_LV2_FREEWHEEL = 5
  199. PARAMETER_LV2_TIME = 6
  200. # Internal Parameters Index
  201. PARAMETER_NULL = -1
  202. PARAMETER_ACTIVE = -2
  203. PARAMETER_DRYWET = -3
  204. PARAMETER_VOLUME = -4
  205. PARAMETER_BALANCE_LEFT = -5
  206. PARAMETER_BALANCE_RIGHT = -6
  207. PARAMETER_PANNING = -7
  208. PARAMETER_MAX = -8
  209. # Options Type
  210. OPTION_PROCESS_NAME = 0
  211. OPTION_PROCESS_MODE = 1
  212. OPTION_FORCE_STEREO = 2
  213. OPTION_PREFER_PLUGIN_BRIDGES = 3
  214. OPTION_PREFER_UI_BRIDGES = 4
  215. OPTION_USE_DSSI_VST_CHUNKS = 5
  216. OPTION_MAX_PARAMETERS = 6
  217. OPTION_OSC_UI_TIMEOUT = 7
  218. OPTION_PREFERRED_BUFFER_SIZE = 8
  219. OPTION_PREFERRED_SAMPLE_RATE = 9
  220. OPTION_PATH_BRIDGE_NATIVE = 10
  221. OPTION_PATH_BRIDGE_POSIX32 = 11
  222. OPTION_PATH_BRIDGE_POSIX64 = 12
  223. OPTION_PATH_BRIDGE_WIN32 = 13
  224. OPTION_PATH_BRIDGE_WIN64 = 14
  225. OPTION_PATH_BRIDGE_LV2_GTK2 = 15
  226. OPTION_PATH_BRIDGE_LV2_GTK3 = 16
  227. OPTION_PATH_BRIDGE_LV2_QT4 = 17
  228. OPTION_PATH_BRIDGE_LV2_QT5 = 18
  229. OPTION_PATH_BRIDGE_LV2_COCOA = 19
  230. OPTION_PATH_BRIDGE_LV2_WINDOWS = 20
  231. OPTION_PATH_BRIDGE_LV2_X11 = 21
  232. OPTION_PATH_BRIDGE_VST_COCOA = 22
  233. OPTION_PATH_BRIDGE_VST_HWND = 23
  234. OPTION_PATH_BRIDGE_VST_X11 = 24
  235. # Callback Type
  236. CALLBACK_DEBUG = 0
  237. CALLBACK_PARAMETER_VALUE_CHANGED = 1
  238. CALLBACK_PARAMETER_MIDI_CHANNEL_CHANGED = 2
  239. CALLBACK_PARAMETER_MIDI_CC_CHANGED = 3
  240. CALLBACK_PROGRAM_CHANGED = 4
  241. CALLBACK_MIDI_PROGRAM_CHANGED = 5
  242. CALLBACK_NOTE_ON = 6
  243. CALLBACK_NOTE_OFF = 7
  244. CALLBACK_SHOW_GUI = 8
  245. CALLBACK_UPDATE = 9
  246. CALLBACK_RELOAD_INFO = 10
  247. CALLBACK_RELOAD_PARAMETERS = 11
  248. CALLBACK_RELOAD_PROGRAMS = 12
  249. CALLBACK_RELOAD_ALL = 13
  250. CALLBACK_NSM_ANNOUNCE = 14
  251. CALLBACK_NSM_OPEN1 = 15
  252. CALLBACK_NSM_OPEN2 = 16
  253. CALLBACK_NSM_SAVE = 17
  254. CALLBACK_ERROR = 18
  255. CALLBACK_QUIT = 19
  256. # Process Mode Type
  257. PROCESS_MODE_SINGLE_CLIENT = 0
  258. PROCESS_MODE_MULTIPLE_CLIENTS = 1
  259. PROCESS_MODE_CONTINUOUS_RACK = 2
  260. PROCESS_MODE_PATCHBAY = 3
  261. # Set BINARY_NATIVE
  262. if HAIKU or LINUX or MACOS:
  263. BINARY_NATIVE = BINARY_POSIX64 if kIs64bit else BINARY_POSIX32
  264. elif WINDOWS:
  265. BINARY_NATIVE = BINARY_WIN64 if kIs64bit else BINARY_WIN32
  266. else:
  267. BINARY_NATIVE = BINARY_OTHER
  268. # ------------------------------------------------------------------------------------------------------------
  269. # Carla Host object
  270. class CarlaHostObject(object):
  271. __slots__ = [
  272. 'host',
  273. 'gui',
  274. 'isControl',
  275. 'processMode',
  276. 'maxParameters'
  277. ]
  278. Carla = CarlaHostObject()
  279. Carla.host = None
  280. Carla.gui = None
  281. Carla.isControl = False
  282. Carla.processMode = PROCESS_MODE_CONTINUOUS_RACK
  283. Carla.maxParameters = MAX_RACK_PLUGINS
  284. # ------------------------------------------------------------------------------------------------------------
  285. # Carla GUI stuff
  286. ICON_STATE_NULL = 0
  287. ICON_STATE_WAIT = 1
  288. ICON_STATE_OFF = 2
  289. ICON_STATE_ON = 3
  290. PALETTE_COLOR_NONE = 0
  291. PALETTE_COLOR_WHITE = 1
  292. PALETTE_COLOR_RED = 2
  293. PALETTE_COLOR_GREEN = 3
  294. PALETTE_COLOR_BLUE = 4
  295. PALETTE_COLOR_YELLOW = 5
  296. PALETTE_COLOR_ORANGE = 6
  297. PALETTE_COLOR_BROWN = 7
  298. PALETTE_COLOR_PINK = 8
  299. CarlaStateParameter = {
  300. 'index': 0,
  301. 'name': "",
  302. 'symbol': "",
  303. 'value': 0.0,
  304. 'midiChannel': 1,
  305. 'midiCC': -1
  306. }
  307. CarlaStateCustomData = {
  308. 'type': "",
  309. 'key': "",
  310. 'value': ""
  311. }
  312. CarlaSaveState = {
  313. 'type': "",
  314. 'name': "",
  315. 'label': "",
  316. 'binary': "",
  317. 'uniqueId': 0,
  318. 'active': False,
  319. 'dryWet': 1.0,
  320. 'volume': 1.0,
  321. 'balanceLeft': -1.0,
  322. 'balanceRight': 1.0,
  323. 'pannning': 0.0,
  324. 'parameterList': [],
  325. 'currentProgramIndex': -1,
  326. 'currentProgramName': "",
  327. 'currentMidiBank': -1,
  328. 'currentMidiProgram': -1,
  329. 'customDataList': [],
  330. 'chunk': None
  331. }
  332. # ------------------------------------------------------------------------------------------------------------
  333. # Static MIDI CC list
  334. MIDI_CC_LIST = (
  335. #"0x00 Bank Select",
  336. "0x01 Modulation",
  337. "0x02 Breath",
  338. "0x03 (Undefined)",
  339. "0x04 Foot",
  340. "0x05 Portamento",
  341. #"0x06 (Data Entry MSB)",
  342. "0x07 Volume",
  343. "0x08 Balance",
  344. "0x09 (Undefined)",
  345. "0x0A Pan",
  346. "0x0B Expression",
  347. "0x0C FX Control 1",
  348. "0x0D FX Control 2",
  349. "0x0E (Undefined)",
  350. "0x0F (Undefined)",
  351. "0x10 General Purpose 1",
  352. "0x11 General Purpose 2",
  353. "0x12 General Purpose 3",
  354. "0x13 General Purpose 4",
  355. "0x14 (Undefined)",
  356. "0x15 (Undefined)",
  357. "0x16 (Undefined)",
  358. "0x17 (Undefined)",
  359. "0x18 (Undefined)",
  360. "0x19 (Undefined)",
  361. "0x1A (Undefined)",
  362. "0x1B (Undefined)",
  363. "0x1C (Undefined)",
  364. "0x1D (Undefined)",
  365. "0x1E (Undefined)",
  366. "0x1F (Undefined)",
  367. #"0x20 *Bank Select",
  368. #"0x21 *Modulation",
  369. #"0x22 *Breath",
  370. #"0x23 *(Undefined)",
  371. #"0x24 *Foot",
  372. #"0x25 *Portamento",
  373. #"0x26 *(Data Entry MSB)",
  374. #"0x27 *Volume",
  375. #"0x28 *Balance",
  376. #"0x29 *(Undefined)",
  377. #"0x2A *Pan",
  378. #"0x2B *Expression",
  379. #"0x2C *FX *Control 1",
  380. #"0x2D *FX *Control 2",
  381. #"0x2E *(Undefined)",
  382. #"0x2F *(Undefined)",
  383. #"0x30 *General Purpose 1",
  384. #"0x31 *General Purpose 2",
  385. #"0x32 *General Purpose 3",
  386. #"0x33 *General Purpose 4",
  387. #"0x34 *(Undefined)",
  388. #"0x35 *(Undefined)",
  389. #"0x36 *(Undefined)",
  390. #"0x37 *(Undefined)",
  391. #"0x38 *(Undefined)",
  392. #"0x39 *(Undefined)",
  393. #"0x3A *(Undefined)",
  394. #"0x3B *(Undefined)",
  395. #"0x3C *(Undefined)",
  396. #"0x3D *(Undefined)",
  397. #"0x3E *(Undefined)",
  398. #"0x3F *(Undefined)",
  399. #"0x40 Damper On/Off", # <63 off, >64 on
  400. #"0x41 Portamento On/Off", # <63 off, >64 on
  401. #"0x42 Sostenuto On/Off", # <63 off, >64 on
  402. #"0x43 Soft Pedal On/Off", # <63 off, >64 on
  403. #"0x44 Legato Footswitch", # <63 Normal, >64 Legato
  404. #"0x45 Hold 2", # <63 off, >64 on
  405. "0x46 Control 1 [Variation]",
  406. "0x47 Control 2 [Timbre]",
  407. "0x48 Control 3 [Release]",
  408. "0x49 Control 4 [Attack]",
  409. "0x4A Control 5 [Brightness]",
  410. "0x4B Control 6 [Decay]",
  411. "0x4C Control 7 [Vib Rate]",
  412. "0x4D Control 8 [Vib Depth]",
  413. "0x4E Control 9 [Vib Delay]",
  414. "0x4F Control 10 [Undefined]",
  415. "0x50 General Purpose 5",
  416. "0x51 General Purpose 6",
  417. "0x52 General Purpose 7",
  418. "0x53 General Purpose 8",
  419. "0x54 Portamento Control",
  420. "0x5B FX 1 Depth [Reverb]",
  421. "0x5C FX 2 Depth [Tremolo]",
  422. "0x5D FX 3 Depth [Chorus]",
  423. "0x5E FX 4 Depth [Detune]",
  424. "0x5F FX 5 Depth [Phaser]"
  425. )
  426. # ------------------------------------------------------------------------------------------------------------
  427. # Default Plugin Folders
  428. if WINDOWS:
  429. splitter = ";"
  430. APPDATA = os.getenv("APPDATA")
  431. PROGRAMFILES = os.getenv("PROGRAMFILES")
  432. PROGRAMFILESx86 = os.getenv("PROGRAMFILES(x86)")
  433. COMMONPROGRAMFILES = os.getenv("COMMONPROGRAMFILES")
  434. # Small integrity tests
  435. if not APPDATA:
  436. print("APPDATA variable not set, cannot continue")
  437. sys.exit(1)
  438. if not PROGRAMFILES:
  439. print("PROGRAMFILES variable not set, cannot continue")
  440. sys.exit(1)
  441. if not COMMONPROGRAMFILES:
  442. print("COMMONPROGRAMFILES variable not set, cannot continue")
  443. sys.exit(1)
  444. DEFAULT_LADSPA_PATH = ";".join((os.path.join(APPDATA, "LADSPA"),
  445. os.path.join(PROGRAMFILES, "LADSPA")))
  446. DEFAULT_DSSI_PATH = ";".join((os.path.join(APPDATA, "DSSI"),
  447. os.path.join(PROGRAMFILES, "DSSI")))
  448. DEFAULT_LV2_PATH = ";".join((os.path.join(APPDATA, "LV2"),
  449. os.path.join(COMMONPROGRAMFILES, "LV2")))
  450. DEFAULT_VST_PATH = ";".join((os.path.join(PROGRAMFILES, "VstPlugins"),
  451. os.path.join(PROGRAMFILES, "Steinberg", "VstPlugins")))
  452. DEFAULT_GIG_PATH = ";".join((os.path.join(APPDATA, "GIG"),))
  453. DEFAULT_SF2_PATH = ";".join((os.path.join(APPDATA, "SF2"),))
  454. DEFAULT_SFZ_PATH = ";".join((os.path.join(APPDATA, "SFZ"),))
  455. if PROGRAMFILESx86:
  456. DEFAULT_LADSPA_PATH += ";"+os.path.join(PROGRAMFILESx86, "LADSPA")
  457. DEFAULT_DSSI_PATH += ";"+os.path.join(PROGRAMFILESx86, "DSSI")
  458. DEFAULT_VST_PATH += ";"+os.path.join(PROGRAMFILESx86, "VstPlugins")
  459. DEFAULT_VST_PATH += ";"+os.path.join(PROGRAMFILESx86, "Steinberg", "VstPlugins")
  460. elif HAIKU:
  461. splitter = ":"
  462. # TODO
  463. DEFAULT_LADSPA_PATH = ""
  464. DEFAULT_DSSI_PATH = ""
  465. DEFAULT_LV2_PATH = ""
  466. DEFAULT_VST_PATH = ""
  467. DEFAULT_GIG_PATH = ""
  468. DEFAULT_SF2_PATH = ""
  469. DEFAULT_SFZ_PATH = ""
  470. elif MACOS:
  471. splitter = ":"
  472. DEFAULT_LADSPA_PATH = ":".join((os.path.join(HOME, "Library", "Audio", "Plug-Ins", "LADSPA"),
  473. os.path.join("/", "Library", "Audio", "Plug-Ins", "LADSPA")))
  474. DEFAULT_DSSI_PATH = ":".join((os.path.join(HOME, "Library", "Audio", "Plug-Ins", "DSSI"),
  475. os.path.join("/", "Library", "Audio", "Plug-Ins", "DSSI")))
  476. DEFAULT_LV2_PATH = ":".join((os.path.join(HOME, "Library", "Audio", "Plug-Ins", "LV2"),
  477. os.path.join("/", "Library", "Audio", "Plug-Ins", "LV2")))
  478. DEFAULT_VST_PATH = ":".join((os.path.join(HOME, "Library", "Audio", "Plug-Ins", "VST"),
  479. os.path.join("/", "Library", "Audio", "Plug-Ins", "VST")))
  480. # TODO
  481. DEFAULT_GIG_PATH = ""
  482. DEFAULT_SF2_PATH = ""
  483. DEFAULT_SFZ_PATH = ""
  484. else:
  485. splitter = ":"
  486. DEFAULT_LADSPA_PATH = ":".join((os.path.join(HOME, ".ladspa"),
  487. os.path.join("/", "usr", "lib", "ladspa"),
  488. os.path.join("/", "usr", "local", "lib", "ladspa")))
  489. DEFAULT_DSSI_PATH = ":".join((os.path.join(HOME, ".dssi"),
  490. os.path.join("/", "usr", "lib", "dssi"),
  491. os.path.join("/", "usr", "local", "lib", "dssi")))
  492. DEFAULT_LV2_PATH = ":".join((os.path.join(HOME, ".lv2"),
  493. os.path.join("/", "usr", "lib", "lv2"),
  494. os.path.join("/", "usr", "local", "lib", "lv2")))
  495. DEFAULT_VST_PATH = ":".join((os.path.join(HOME, ".vst"),
  496. os.path.join("/", "usr", "lib", "vst"),
  497. os.path.join("/", "usr", "local", "lib", "vst")))
  498. DEFAULT_GIG_PATH = ":".join((os.path.join(HOME, ".sounds"),
  499. os.path.join("/", "usr", "share", "sounds", "gig")))
  500. DEFAULT_SF2_PATH = ":".join((os.path.join(HOME, ".sounds"),
  501. os.path.join("/", "usr", "share", "sounds", "sf2")))
  502. DEFAULT_SFZ_PATH = ":".join((os.path.join(HOME, ".sounds"),
  503. os.path.join("/", "usr", "share", "sounds", "sfz")))
  504. # ------------------------------------------------------------------------------------------------------------
  505. # Default Plugin Folders (set)
  506. global LADSPA_PATH, DSSI_PATH, LV2_PATH, VST_PATH, GIG_PATH, SF2_PATH, SFZ_PATH
  507. LADSPA_PATH = os.getenv("LADSPA_PATH", DEFAULT_LADSPA_PATH).split(splitter)
  508. DSSI_PATH = os.getenv("DSSI_PATH", DEFAULT_DSSI_PATH).split(splitter)
  509. LV2_PATH = os.getenv("LV2_PATH", DEFAULT_LV2_PATH).split(splitter)
  510. VST_PATH = os.getenv("VST_PATH", DEFAULT_VST_PATH).split(splitter)
  511. GIG_PATH = os.getenv("GIG_PATH", DEFAULT_GIG_PATH).split(splitter)
  512. SF2_PATH = os.getenv("SF2_PATH", DEFAULT_SF2_PATH).split(splitter)
  513. SFZ_PATH = os.getenv("SFZ_PATH", DEFAULT_SFZ_PATH).split(splitter)
  514. if haveLRDF:
  515. LADSPA_RDF_PATH_env = os.getenv("LADSPA_RDF_PATH")
  516. if LADSPA_RDF_PATH_env:
  517. ladspa_rdf.set_rdf_path(LADSPA_RDF_PATH_env.split(splitter))
  518. del LADSPA_RDF_PATH_env
  519. # ------------------------------------------------------------------------------------------------------------
  520. # Search for Carla library and tools
  521. global carla_library_path
  522. carla_library_path = ""
  523. carla_discovery_native = ""
  524. carla_discovery_posix32 = ""
  525. carla_discovery_posix64 = ""
  526. carla_discovery_win32 = ""
  527. carla_discovery_win64 = ""
  528. carla_bridge_native = ""
  529. carla_bridge_posix32 = ""
  530. carla_bridge_posix64 = ""
  531. carla_bridge_win32 = ""
  532. carla_bridge_win64 = ""
  533. carla_bridge_lv2_gtk2 = ""
  534. carla_bridge_lv2_gtk3 = ""
  535. carla_bridge_lv2_qt4 = ""
  536. carla_bridge_lv2_qt5 = ""
  537. carla_bridge_lv2_cocoa = ""
  538. carla_bridge_lv2_windows = ""
  539. carla_bridge_lv2_x11 = ""
  540. carla_bridge_vst_cocoa = ""
  541. carla_bridge_vst_hwnd = ""
  542. carla_bridge_vst_x11 = ""
  543. if WINDOWS:
  544. carla_libname = "libcarla_standalone.dll"
  545. elif MACOS:
  546. carla_libname = "libcarla_standalone.dylib"
  547. else:
  548. carla_libname = "libcarla_standalone.so"
  549. CWD = sys.path[0]
  550. # make it work with cxfreeze
  551. if CWD.endswith("%scarla" % os.sep):
  552. CWD = CWD.rsplit("%scarla" % os.sep, 1)[0]
  553. # find carla_library_path
  554. if os.path.exists(os.path.join(CWD, "backend", carla_libname)):
  555. carla_library_path = os.path.join(CWD, "backend", carla_libname)
  556. else:
  557. if WINDOWS:
  558. CARLA_PATH = (os.path.join(PROGRAMFILES, "Carla"),)
  559. elif MACOS:
  560. CARLA_PATH = ("/opt/local/lib", "/usr/local/lib/", "/usr/lib")
  561. else:
  562. CARLA_PATH = ("/usr/local/lib/", "/usr/lib")
  563. for path in CARLA_PATH:
  564. if os.path.exists(os.path.join(path, "carla", carla_libname)):
  565. carla_library_path = os.path.join(path, "carla", carla_libname)
  566. break
  567. # find any tool
  568. def findTool(tdir, tname):
  569. if os.path.exists(os.path.join(CWD, tdir, tname)):
  570. return os.path.join(CWD, tdir, tname)
  571. for p in PATH:
  572. if os.path.exists(os.path.join(p, tname)):
  573. return os.path.join(p, tname)
  574. return ""
  575. # find wine/windows tools
  576. carla_discovery_win32 = findTool("discovery", "carla-discovery-win32.exe")
  577. carla_discovery_win64 = findTool("discovery", "carla-discovery-win64.exe")
  578. carla_bridge_win32 = findTool("bridge", "carla-bridge-win32.exe")
  579. carla_bridge_win64 = findTool("bridge", "carla-bridge-win64.exe")
  580. # find native and posix tools
  581. if not WINDOWS:
  582. carla_discovery_native = findTool("discovery", "carla-discovery-native")
  583. carla_discovery_posix32 = findTool("discovery", "carla-discovery-posix32")
  584. carla_discovery_posix64 = findTool("discovery", "carla-discovery-posix64")
  585. carla_bridge_native = findTool("bridge", "carla-bridge-native")
  586. carla_bridge_posix32 = findTool("bridge", "carla-bridge-posix32")
  587. carla_bridge_posix64 = findTool("bridge", "carla-bridge-posix64")
  588. # find windows only tools
  589. if WINDOWS:
  590. carla_bridge_lv2_windows = findTool("bridge", "carla-bridge-lv2-windows.exe")
  591. carla_bridge_vst_hwnd = findTool("bridge", "carla-bridge-vst-hwnd.exe")
  592. # find mac os only tools
  593. elif MACOS:
  594. carla_bridge_lv2_cocoa = findTool("bridge", "carla-bridge-lv2-cocoa")
  595. carla_bridge_vst_cocoa = findTool("bridge", "carla-bridge-vst-cocoa")
  596. # find generic tools
  597. else:
  598. carla_bridge_lv2_gtk2 = findTool("bridge", "carla-bridge-lv2-gtk2")
  599. carla_bridge_lv2_gtk3 = findTool("bridge", "carla-bridge-lv2-gtk3")
  600. carla_bridge_lv2_qt4 = findTool("bridge", "carla-bridge-lv2-qt4")
  601. carla_bridge_lv2_qt5 = findTool("bridge", "carla-bridge-lv2-qt5")
  602. # find linux only tools
  603. if LINUX:
  604. carla_bridge_lv2_x11 = os.path.join("bridge", "carla-bridge-lv2-x11")
  605. carla_bridge_vst_x11 = os.path.join("bridge", "carla-bridge-vst-x11")
  606. # ------------------------------------------------------------------------------------------------------------
  607. # Convert a ctypes c_char_p into a python string
  608. def cString(value):
  609. if not value:
  610. return ""
  611. if isinstance(value, str):
  612. return value
  613. return value.decode("utf-8", errors="ignore")
  614. # ------------------------------------------------------------------------------------------------------------
  615. # Check if a value is a number (float support)
  616. def isNumber(value):
  617. try:
  618. float(value)
  619. return True
  620. except:
  621. return False
  622. # ------------------------------------------------------------------------------------------------------------
  623. # Convert a value to a list
  624. def toList(value):
  625. if value is None:
  626. return []
  627. elif not isinstance(value, list):
  628. return [value]
  629. else:
  630. return value
  631. # ------------------------------------------------------------------------------------------------------------
  632. # Unicode open
  633. def uopen(filename, mode="r"):
  634. return codecopen(filename, encoding="utf-8", mode=mode)
  635. # ------------------------------------------------------------------------------------------------------------
  636. # Get Icon from user theme, using our own as backup (Oxygen)
  637. def getIcon(icon, size=16):
  638. return QIcon.fromTheme(icon, QIcon(":/%ix%i/%s.png" % (size, size, icon)))
  639. # ------------------------------------------------------------------------------------------------------------
  640. # Custom MessageBox
  641. def CustomMessageBox(self_, icon, title, text, extraText="", buttons=QMessageBox.Yes|QMessageBox.No, defButton=QMessageBox.No):
  642. msgBox = QMessageBox(self_)
  643. msgBox.setIcon(icon)
  644. msgBox.setWindowTitle(title)
  645. msgBox.setText(text)
  646. msgBox.setInformativeText(extraText)
  647. msgBox.setStandardButtons(buttons)
  648. msgBox.setDefaultButton(defButton)
  649. return msgBox.exec_()
  650. # ------------------------------------------------------------------------------------------------------------
  651. # Plugin Query (helper functions)
  652. def findBinaries(bPATH, OS):
  653. binaries = []
  654. if OS == "WINDOWS":
  655. extensions = (".dll",)
  656. elif OS == "MACOS":
  657. extensions = (".dylib", ".so")
  658. else:
  659. extensions = (".so", ".sO", ".SO", ".So")
  660. for root, dirs, files in os.walk(bPATH):
  661. for name in [name for name in files if name.endswith(extensions)]:
  662. binaries.append(os.path.join(root, name))
  663. return binaries
  664. # FIXME - may use any extension, just needs to have manifest.ttl
  665. def findLV2Bundles(bPATH):
  666. bundles = []
  667. extensions = (".lv2", ".lV2", ".LV2", ".Lv2") if not WINDOWS else (".lv2",)
  668. for root, dirs, files in os.walk(bPATH):
  669. for dir_ in [dir_ for dir_ in dirs if dir_.endswith(extensions)]:
  670. bundles.append(os.path.join(root, dir_))
  671. return bundles
  672. def findSoundKits(bPATH, stype):
  673. soundfonts = []
  674. if stype == "gig":
  675. extensions = (".gig", ".giG", ".gIG", ".GIG", ".GIg", ".Gig") if not WINDOWS else (".gig",)
  676. elif stype == "sf2":
  677. extensions = (".sf2", ".sF2", ".SF2", ".Sf2") if not WINDOWS else (".sf2",)
  678. elif stype == "sfz":
  679. extensions = (".sfz", ".sfZ", ".sFZ", ".SFZ", ".SFz", ".Sfz") if not WINDOWS else (".sfz",)
  680. else:
  681. return []
  682. for root, dirs, files in os.walk(bPATH):
  683. for name in [name for name in files if name.endswith(extensions)]:
  684. soundfonts.append(os.path.join(root, name))
  685. return soundfonts
  686. def findDSSIGUI(filename, name, label):
  687. pluginDir = filename.rsplit(".", 1)[0]
  688. shortName = os.path.basename(pluginDir)
  689. guiFilename = ""
  690. checkName = name.replace(" ", "_")
  691. checkLabel = label
  692. checkSName = shortName
  693. if checkName[-1] != "_": checkName += "_"
  694. if checkLabel[-1] != "_": checkLabel += "_"
  695. if checkSName[-1] != "_": checkSName += "_"
  696. for root, dirs, files in os.walk(pluginDir):
  697. guiFiles = files
  698. break
  699. else:
  700. guiFiles = []
  701. for guiFile in guiFiles:
  702. if guiFile.startswith(checkName) or guiFile.startswith(checkLabel) or guiFile.startswith(checkSName):
  703. guiFilename = os.path.join(pluginDir, guiFile)
  704. break
  705. return guiFilename
  706. # ------------------------------------------------------------------------------------------------------------
  707. # Plugin Query
  708. PLUGIN_QUERY_API_VERSION = 1
  709. PyPluginInfo = {
  710. 'API': PLUGIN_QUERY_API_VERSION,
  711. 'build': 0, # BINARY_NONE
  712. 'type': 0, # PLUGIN_NONE
  713. 'hints': 0x0,
  714. 'binary': "",
  715. 'name': "",
  716. 'label': "",
  717. 'maker': "",
  718. 'copyright': "",
  719. 'uniqueId': 0,
  720. 'audio.ins': 0,
  721. 'audio.outs': 0,
  722. 'audio.total': 0,
  723. 'midi.ins': 0,
  724. 'midi.outs': 0,
  725. 'midi.total': 0,
  726. 'parameters.ins': 0,
  727. 'parameters.outs': 0,
  728. 'parameters.total': 0,
  729. 'programs.total': 0
  730. }
  731. def runCarlaDiscovery(itype, stype, filename, tool, isWine=False):
  732. fakeLabel = os.path.basename(filename).rsplit(".", 1)[0]
  733. plugins = []
  734. command = []
  735. if LINUX or MACOS:
  736. command.append("env")
  737. command.append("LANG=C")
  738. if isWine:
  739. command.append("WINEDEBUG=-all")
  740. command.append(tool)
  741. command.append(stype)
  742. command.append(filename)
  743. Ps = Popen(command, stdout=PIPE)
  744. Ps.wait()
  745. output = Ps.stdout.read().decode("utf-8", errors="ignore").split("\n")
  746. pinfo = None
  747. for line in output:
  748. line = line.strip()
  749. if line == "carla-discovery::init::-----------":
  750. pinfo = deepcopy(PyPluginInfo)
  751. pinfo['type'] = itype
  752. pinfo['binary'] = filename
  753. elif line == "carla-discovery::end::------------":
  754. if pinfo != None:
  755. plugins.append(pinfo)
  756. pinfo = None
  757. elif line == "Segmentation fault":
  758. print("carla-discovery::crash::%s crashed during discovery" % filename)
  759. elif line.startswith("err:module:import_dll Library"):
  760. print(line)
  761. elif line.startswith("carla-discovery::error::"):
  762. print("%s - %s" % (line, filename))
  763. elif line.startswith("carla-discovery::"):
  764. if pinfo == None:
  765. continue
  766. prop, value = line.replace("carla-discovery::", "").split("::", 1)
  767. if prop == "name":
  768. pinfo['name'] = value if value else fakeLabel
  769. elif prop == "label":
  770. pinfo['label'] = value if value else fakeLabel
  771. elif prop == "maker":
  772. pinfo['maker'] = value
  773. elif prop == "copyright":
  774. pinfo['copyright'] = value
  775. elif prop == "uniqueId":
  776. if value.isdigit(): pinfo['uniqueId'] = int(value)
  777. elif prop == "hints":
  778. if value.isdigit(): pinfo['hints'] = int(value)
  779. elif prop == "audio.ins":
  780. if value.isdigit(): pinfo['audio.ins'] = int(value)
  781. elif prop == "audio.outs":
  782. if value.isdigit(): pinfo['audio.outs'] = int(value)
  783. elif prop == "audio.total":
  784. if value.isdigit(): pinfo['audio.total'] = int(value)
  785. elif prop == "midi.ins":
  786. if value.isdigit(): pinfo['midi.ins'] = int(value)
  787. elif prop == "midi.outs":
  788. if value.isdigit(): pinfo['midi.outs'] = int(value)
  789. elif prop == "midi.total":
  790. if value.isdigit(): pinfo['midi.total'] = int(value)
  791. elif prop == "parameters.ins":
  792. if value.isdigit(): pinfo['parameters.ins'] = int(value)
  793. elif prop == "parameters.outs":
  794. if value.isdigit(): pinfo['parameters.outs'] = int(value)
  795. elif prop == "parameters.total":
  796. if value.isdigit(): pinfo['parameters.total'] = int(value)
  797. elif prop == "programs.total":
  798. if value.isdigit(): pinfo['programs.total'] = int(value)
  799. elif prop == "build":
  800. if value.isdigit(): pinfo['build'] = int(value)
  801. # Additional checks
  802. for pinfo in plugins:
  803. if itype == PLUGIN_DSSI:
  804. if findDSSIGUI(pinfo['binary'], pinfo['name'], pinfo['label']):
  805. pinfo['hints'] |= PLUGIN_HAS_GUI
  806. return plugins
  807. def checkPluginInternal(desc):
  808. plugins = []
  809. pinfo = deepcopy(PyPluginInfo)
  810. pinfo['build'] = BINARY_NATIVE
  811. pinfo['type'] = PLUGIN_INTERNAL
  812. pinfo['hints'] = int(desc['hints'])
  813. pinfo['name'] = cString(desc['name'])
  814. pinfo['label'] = cString(desc['label'])
  815. pinfo['maker'] = cString(desc['maker'])
  816. pinfo['copyright'] = cString(desc['copyright'])
  817. pinfo['audio.ins'] = int(desc['audioIns'])
  818. pinfo['audio.outs'] = int(desc['audioOuts'])
  819. pinfo['audio.total'] = pinfo['audio.ins'] + pinfo['audio.outs']
  820. pinfo['midi.ins'] = int(desc['midiIns'])
  821. pinfo['midi.outs'] = int(desc['midiOuts'])
  822. pinfo['midi.total'] = pinfo['midi.ins'] + pinfo['midi.outs']
  823. pinfo['parameters.ins'] = int(desc['parameterIns'])
  824. pinfo['parameters.outs'] = int(desc['parameterOuts'])
  825. pinfo['parameters.total'] = pinfo['parameters.ins'] + pinfo['parameters.outs']
  826. plugins.append(pinfo)
  827. return plugins
  828. def checkPluginLADSPA(filename, tool, isWine=False):
  829. return runCarlaDiscovery(PLUGIN_LADSPA, "LADSPA", filename, tool, isWine)
  830. def checkPluginDSSI(filename, tool, isWine=False):
  831. return runCarlaDiscovery(PLUGIN_DSSI, "DSSI", filename, tool, isWine)
  832. def checkPluginLV2(filename, tool, isWine=False):
  833. return runCarlaDiscovery(PLUGIN_LV2, "LV2", filename, tool, isWine)
  834. def checkPluginVST(filename, tool, isWine=False):
  835. return runCarlaDiscovery(PLUGIN_VST, "VST", filename, tool, isWine)
  836. def checkPluginGIG(filename, tool):
  837. return runCarlaDiscovery(PLUGIN_GIG, "GIG", filename, tool)
  838. def checkPluginSF2(filename, tool):
  839. return runCarlaDiscovery(PLUGIN_SF2, "SF2", filename, tool)
  840. def checkPluginSFZ(filename, tool):
  841. return runCarlaDiscovery(PLUGIN_SFZ, "SFZ", filename, tool)
  842. # ------------------------------------------------------------------------------------------------------------
  843. # Carla XML helpers
  844. def getSaveStateDictFromXML(xmlNode):
  845. saveState = deepcopy(CarlaSaveState)
  846. node = xmlNode.firstChild()
  847. while not node.isNull():
  848. # ------------------------------------------------------
  849. # Info
  850. if node.toElement().tagName() == "Info":
  851. xmlInfo = node.toElement().firstChild()
  852. while not xmlInfo.isNull():
  853. tag = xmlInfo.toElement().tagName()
  854. text = xmlInfo.toElement().text().strip()
  855. if tag == "Type":
  856. saveState["type"] = text
  857. elif tag == "Name":
  858. saveState["name"] = xmlSafeString(text, False)
  859. elif tag in ("Label", "URI"):
  860. saveState["label"] = xmlSafeString(text, False)
  861. elif tag == "Binary":
  862. saveState["binary"] = xmlSafeString(text, False)
  863. elif tag == "UniqueID":
  864. if text.isdigit(): saveState["uniqueId"] = int(text)
  865. xmlInfo = xmlInfo.nextSibling()
  866. # ------------------------------------------------------
  867. # Data
  868. elif node.toElement().tagName() == "Data":
  869. xmlData = node.toElement().firstChild()
  870. while not xmlData.isNull():
  871. tag = xmlData.toElement().tagName()
  872. text = xmlData.toElement().text().strip()
  873. # ----------------------------------------------
  874. # Internal Data
  875. if tag == "Active":
  876. saveState['active'] = bool(text == "Yes")
  877. elif tag == "DryWet":
  878. if isNumber(text): saveState["dryWet"] = float(text)
  879. elif tag == "Volume":
  880. if isNumber(text): saveState["volume"] = float(text)
  881. elif tag == "Balance-Left":
  882. if isNumber(text): saveState["balanceLeft"] = float(text)
  883. elif tag == "Balance-Right":
  884. if isNumber(text): saveState["balanceRight"] = float(text)
  885. elif tag == "Panning":
  886. if isNumber(text): saveState["pannning"] = float(text)
  887. # ----------------------------------------------
  888. # Program (current)
  889. elif tag == "CurrentProgramIndex":
  890. if text.isdigit(): saveState["currentProgramIndex"] = int(text)
  891. elif tag == "CurrentProgramName":
  892. saveState["currentProgramName"] = xmlSafeString(text, False)
  893. # ----------------------------------------------
  894. # Midi Program (current)
  895. elif tag == "CurrentMidiBank":
  896. if text.isdigit(): saveState["currentMidiBank"] = int(text)
  897. elif tag == "CurrentMidiProgram":
  898. if text.isdigit(): saveState["currentMidiProgram"] = int(text)
  899. # ----------------------------------------------
  900. # Parameters
  901. elif tag == "Parameter":
  902. stateParameter = deepcopy(CarlaStateParameter)
  903. xmlSubData = xmlData.toElement().firstChild()
  904. while not xmlSubData.isNull():
  905. pTag = xmlSubData.toElement().tagName()
  906. pText = xmlSubData.toElement().text().strip()
  907. if pTag == "Index":
  908. if pText.isdigit(): stateParameter["index"] = int(pText)
  909. elif pTag == "Name":
  910. stateParameter["name"] = xmlSafeString(pText, False)
  911. elif pTag == "Symbol":
  912. stateParameter["symbol"] = xmlSafeString(pText, False)
  913. elif pTag == "Value":
  914. if isNumber(pText): stateParameter["value"] = float(pText)
  915. elif pTag == "MidiChannel":
  916. if pText.isdigit(): stateParameter["midiChannel"] = int(pText)
  917. elif pTag == "MidiCC":
  918. if pText.isdigit(): stateParameter["midiCC"] = int(pText)
  919. xmlSubData = xmlSubData.nextSibling()
  920. saveState["parameterList"].append(stateParameter)
  921. # ----------------------------------------------
  922. # Custom Data
  923. elif tag == "CustomData":
  924. stateCustomData = deepcopy(CarlaStateCustomData)
  925. xmlSubData = xmlData.toElement().firstChild()
  926. while not xmlSubData.isNull():
  927. cTag = xmlSubData.toElement().tagName()
  928. cText = xmlSubData.toElement().text().strip()
  929. if cTag == "Type":
  930. stateCustomData["type"] = xmlSafeString(cText, False)
  931. elif cTag == "Key":
  932. stateCustomData["key"] = xmlSafeString(cText, False)
  933. elif cTag == "Value":
  934. stateCustomData["value"] = xmlSafeString(cText, False)
  935. xmlSubData = xmlSubData.nextSibling()
  936. saveState["customDataList"].append(stateCustomData)
  937. # ----------------------------------------------
  938. # Chunk
  939. elif tag == "Chunk":
  940. saveState["chunk"] = xmlSafeString(text, False)
  941. # ----------------------------------------------
  942. xmlData = xmlData.nextSibling()
  943. # ------------------------------------------------------
  944. node = node.nextSibling()
  945. return saveState
  946. def xmlSafeString(string, toXml):
  947. if toXml:
  948. return string.replace("&", "&amp;").replace("<","&lt;").replace(">","&gt;").replace("'","&apos;").replace("\"","&quot;")
  949. else:
  950. return string.replace("&amp;", "&").replace("&lt;","<").replace("&gt;",">").replace("&apos;","'").replace("&quot;","\"")
  951. # ------------------------------------------------------------------------------------------------------------
  952. # Carla About dialog
  953. class CarlaAboutW(QDialog):
  954. def __init__(self, parent):
  955. QDialog.__init__(self, parent)
  956. self.ui = ui_carla_about.Ui_CarlaAboutW()
  957. self.ui.setupUi(self)
  958. if Carla.isControl:
  959. extraInfo = " - <b>%s</b>" % self.tr("OSC Bridge Version")
  960. else:
  961. extraInfo = ""
  962. self.ui.l_about.setText(self.tr(""
  963. "<br>Version %s"
  964. "<br>Carla is a Multi-Plugin Host for JACK%s.<br>"
  965. "<br>Copyright (C) 2011-2013 falkTX<br>"
  966. "" % (VERSION, extraInfo)))
  967. if Carla.isControl:
  968. self.ui.l_extended.hide()
  969. self.ui.tabWidget.removeTab(1)
  970. self.ui.tabWidget.removeTab(1)
  971. else:
  972. self.ui.l_extended.setText(cString(Carla.host.get_extended_license_text()))
  973. self.ui.le_osc_url.setText(cString(Carla.host.get_host_osc_url()) if Carla.host.is_engine_running() else self.tr("(Engine not running)"))
  974. self.ui.l_osc_cmds.setText(
  975. " /set_active <i-value>\n"
  976. " /set_drywet <f-value>\n"
  977. " /set_volume <f-value>\n"
  978. " /set_balance_left <f-value>\n"
  979. " /set_balance_right <f-value>\n"
  980. " /set_panning <f-value>\n"
  981. " /set_parameter_value <i-index> <f-value>\n"
  982. " /set_parameter_midi_cc <i-index> <i-cc>\n"
  983. " /set_parameter_midi_channel <i-index> <i-channel>\n"
  984. " /set_program <i-index>\n"
  985. " /set_midi_program <i-index>\n"
  986. " /note_on <i-note> <i-velo>\n"
  987. " /note_off <i-note>\n"
  988. )
  989. self.ui.l_example.setText("/Carla/2/set_parameter_value 5 1.0")
  990. self.ui.l_example_help.setText("<i>(as in this example, \"2\" is the plugin number and \"5\" the parameter)</i>")
  991. self.ui.l_ladspa.setText(self.tr("Everything! (Including LRDF)"))
  992. self.ui.l_dssi.setText(self.tr("Everything! (Including CustomData/Chunks)"))
  993. self.ui.l_lv2.setText(self.tr("About 95&#37; complete (using custom extensions).<br/>"
  994. "Implemented Feature/Extensions:"
  995. "<ul>"
  996. "<li>http://lv2plug.in/ns/ext/atom</li>"
  997. "<li>http://lv2plug.in/ns/ext/buf-size</li>"
  998. "<li>http://lv2plug.in/ns/ext/data-access</li>"
  999. #"<li>http://lv2plug.in/ns/ext/dynmanifest</li>"
  1000. "<li>http://lv2plug.in/ns/ext/event</li>"
  1001. "<li>http://lv2plug.in/ns/ext/instance-access</li>"
  1002. "<li>http://lv2plug.in/ns/ext/log</li>"
  1003. "<li>http://lv2plug.in/ns/ext/midi</li>"
  1004. "<li>http://lv2plug.in/ns/ext/options</li>"
  1005. #"<li>http://lv2plug.in/ns/ext/parameters</li>"
  1006. "<li>http://lv2plug.in/ns/ext/patch</li>"
  1007. #"<li>http://lv2plug.in/ns/ext/port-groups</li>"
  1008. "<li>http://lv2plug.in/ns/ext/port-props</li>"
  1009. #"<li>http://lv2plug.in/ns/ext/presets</li>"
  1010. "<li>http://lv2plug.in/ns/ext/state</li>"
  1011. "<li>http://lv2plug.in/ns/ext/time</li>"
  1012. "<li>http://lv2plug.in/ns/ext/uri-map</li>"
  1013. "<li>http://lv2plug.in/ns/ext/urid</li>"
  1014. "<li>http://lv2plug.in/ns/ext/worker</li>"
  1015. "<li>http://lv2plug.in/ns/extensions/ui</li>"
  1016. "<li>http://lv2plug.in/ns/extensions/units</li>"
  1017. "<li>http://kxstudio.sf.net/ns/lv2ext/external-ui</li>"
  1018. "<li>http://kxstudio.sf.net/ns/lv2ext/programs</li>"
  1019. "<li>http://kxstudio.sf.net/ns/lv2ext/rtmempool</li>"
  1020. "<li>http://ll-plugins.nongnu.org/lv2/ext/midimap</li>"
  1021. "<li>http://ll-plugins.nongnu.org/lv2/ext/miditype</li>"
  1022. "</ul>"))
  1023. self.ui.l_vst.setText(self.tr("<p>About 85&#37; complete (missing vst bank/presets and some minor stuff)</p>"))
  1024. def done(self, r):
  1025. QDialog.done(self, r)
  1026. self.close()
  1027. # ------------------------------------------------------------------------------------------------------------
  1028. # Plugin Parameter
  1029. class PluginParameter(QWidget):
  1030. def __init__(self, parent, pInfo, pluginId, tabIndex):
  1031. QWidget.__init__(self, parent)
  1032. self.ui = ui_carla_parameter.Ui_PluginParameter()
  1033. self.ui.setupUi(self)
  1034. pType = pInfo['type']
  1035. pHints = pInfo['hints']
  1036. self.fMidiCC = -1
  1037. self.fMidiChannel = 1
  1038. self.fParameterId = pInfo['index']
  1039. self.fPluginId = pluginId
  1040. self.fTabIndex = tabIndex
  1041. self.ui.label.setText(pInfo['name'])
  1042. for MIDI_CC in MIDI_CC_LIST:
  1043. self.ui.combo.addItem(MIDI_CC)
  1044. if pType == PARAMETER_INPUT:
  1045. self.ui.widget.setMinimum(pInfo['minimum'])
  1046. self.ui.widget.setMaximum(pInfo['maximum'])
  1047. self.ui.widget.setDefault(pInfo['default'])
  1048. self.ui.widget.setValue(pInfo['current'], False)
  1049. self.ui.widget.setLabel(pInfo['unit'])
  1050. self.ui.widget.setStep(pInfo['step'])
  1051. self.ui.widget.setStepSmall(pInfo['stepSmall'])
  1052. self.ui.widget.setStepLarge(pInfo['stepLarge'])
  1053. self.ui.widget.setScalePoints(pInfo['scalePoints'], bool(pHints & PARAMETER_USES_SCALEPOINTS))
  1054. if not pHints & PARAMETER_IS_ENABLED:
  1055. self.ui.widget.setReadOnly(True)
  1056. self.ui.combo.setEnabled(False)
  1057. self.ui.sb_channel.setEnabled(False)
  1058. elif not pHints & PARAMETER_IS_AUTOMABLE:
  1059. self.ui.combo.setEnabled(False)
  1060. self.ui.sb_channel.setEnabled(False)
  1061. elif pType == PARAMETER_OUTPUT:
  1062. self.ui.widget.setMinimum(pInfo['minimum'])
  1063. self.ui.widget.setMaximum(pInfo['maximum'])
  1064. self.ui.widget.setValue(pInfo['current'], False)
  1065. self.ui.widget.setLabel(pInfo['unit'])
  1066. self.ui.widget.setReadOnly(True)
  1067. if not pHints & PARAMETER_IS_AUTOMABLE:
  1068. self.ui.combo.setEnabled(False)
  1069. self.ui.sb_channel.setEnabled(False)
  1070. else:
  1071. self.ui.widget.setVisible(False)
  1072. self.ui.combo.setVisible(False)
  1073. self.ui.sb_channel.setVisible(False)
  1074. if pHints & PARAMETER_USES_CUSTOM_TEXT:
  1075. self.ui.widget.setTextCallback(self._textCallBack)
  1076. self.setMidiCC(pInfo['midiCC'])
  1077. self.setMidiChannel(pInfo['midiChannel'])
  1078. self.connect(self.ui.widget, SIGNAL("valueChanged(double)"), SLOT("slot_valueChanged(double)"))
  1079. self.connect(self.ui.sb_channel, SIGNAL("valueChanged(int)"), SLOT("slot_midiChannelChanged(int)"))
  1080. self.connect(self.ui.combo, SIGNAL("currentIndexChanged(int)"), SLOT("slot_midiCcChanged(int)"))
  1081. #if force_parameters_style:
  1082. #self.widget.force_plastique_style()
  1083. self.ui.widget.updateAll()
  1084. def setDefault(self, value):
  1085. self.ui.widget.setDefault(value)
  1086. def setValue(self, value, send=True):
  1087. self.ui.widget.setValue(value, send)
  1088. def setMidiCC(self, cc):
  1089. self.fMidiCC = cc
  1090. self._setMidiCcInComboBox(cc)
  1091. def setMidiChannel(self, channel):
  1092. self.fMidiChannel = channel
  1093. self.ui.sb_channel.setValue(channel)
  1094. def setLabelWidth(self, width):
  1095. self.ui.label.setMinimumWidth(width)
  1096. self.ui.label.setMaximumWidth(width)
  1097. def tabIndex(self):
  1098. return self.fTabIndex
  1099. @pyqtSlot(float)
  1100. def slot_valueChanged(self, value):
  1101. self.emit(SIGNAL("valueChanged(int, double)"), self.fParameterId, value)
  1102. @pyqtSlot(int)
  1103. def slot_midiCcChanged(self, ccIndex):
  1104. if ccIndex <= 0:
  1105. cc = -1
  1106. else:
  1107. ccStr = MIDI_CC_LIST[ccIndex - 1].split(" ")[0]
  1108. cc = int(ccStr, 16)
  1109. if self.fMidiCC != cc:
  1110. self.emit(SIGNAL("midiCcChanged(int, int)"), self.fParameterId, cc)
  1111. self.fMidiCC = cc
  1112. @pyqtSlot(int)
  1113. def slot_midiChannelChanged(self, channel):
  1114. if self.fMidiChannel != channel:
  1115. self.emit(SIGNAL("midiChannelChanged(int, int)"), self.fParameterId, channel)
  1116. self.fMidiChannel = channel
  1117. def _setMidiCcInComboBox(self, cc):
  1118. for i in range(len(MIDI_CC_LIST)):
  1119. ccStr = MIDI_CC_LIST[i].split(" ")[0]
  1120. if int(ccStr, 16) == cc:
  1121. ccIndex = i+1
  1122. break
  1123. else:
  1124. ccIndex = 0
  1125. self.ui.combo.setCurrentIndex(ccIndex)
  1126. def _textCallBack(self):
  1127. return cString(Carla.host.get_parameter_text(self.fPluginId, self.fParameterId))
  1128. # ------------------------------------------------------------------------------------------------------------
  1129. # Plugin Editor (Built-in)
  1130. class PluginEdit(QDialog):
  1131. def __init__(self, parent, pluginId):
  1132. QDialog.__init__(self, Carla.gui)
  1133. self.ui = ui_carla_edit.Ui_PluginEdit()
  1134. self.ui.setupUi(self)
  1135. self.fGeometry = QByteArray()
  1136. self.fPluginId = pluginId
  1137. self.fPuginInfo = None
  1138. self.fRealParent = parent
  1139. self.fCurrentProgram = -1
  1140. self.fCurrentMidiProgram = -1
  1141. self.fCurrentStateFilename = None
  1142. self.fParameterCount = 0
  1143. self.fParameterList = [] # (type, id, widget)
  1144. self.fParameterIdsToUpdate = [] # id
  1145. self.fTabIconOff = QIcon(":/bitmaps/led_off.png")
  1146. self.fTabIconOn = QIcon(":/bitmaps/led_yellow.png")
  1147. self.fTabIconCount = 0
  1148. self.fTabIconTimers = []
  1149. self.ui.keyboard.setMode(self.ui.keyboard.HORIZONTAL)
  1150. self.ui.keyboard.setOctaves(6)
  1151. self.ui.scrollArea.ensureVisible(self.ui.keyboard.width() * 1 / 5, 0)
  1152. self.ui.scrollArea.setVisible(False)
  1153. # TODO - not implemented yet
  1154. self.ui.b_reload_program.setEnabled(False)
  1155. self.ui.b_reload_midi_program.setEnabled(False)
  1156. # Not available for carla-control
  1157. if Carla.isControl:
  1158. self.ui.b_load_state.setEnabled(False)
  1159. self.ui.b_save_state.setEnabled(False)
  1160. else:
  1161. self.connect(self.ui.b_save_state, SIGNAL("clicked()"), SLOT("slot_saveState()"))
  1162. self.connect(self.ui.b_load_state, SIGNAL("clicked()"), SLOT("slot_loadState()"))
  1163. self.connect(self.ui.keyboard, SIGNAL("noteOn(int)"), SLOT("slot_noteOn(int)"))
  1164. self.connect(self.ui.keyboard, SIGNAL("noteOff(int)"), SLOT("slot_noteOff(int)"))
  1165. self.connect(self.ui.cb_programs, SIGNAL("currentIndexChanged(int)"), SLOT("slot_programIndexChanged(int)"))
  1166. self.connect(self.ui.cb_midi_programs, SIGNAL("currentIndexChanged(int)"), SLOT("slot_midiProgramIndexChanged(int)"))
  1167. self.connect(self, SIGNAL("finished(int)"), SLOT("slot_finished()"))
  1168. self.reloadAll()
  1169. def reloadAll(self):
  1170. self.fPluginInfo = Carla.host.get_plugin_info(self.fPluginId)
  1171. self.fPluginInfo["binary"] = cString(self.fPluginInfo["binary"])
  1172. self.fPluginInfo["name"] = cString(self.fPluginInfo["name"])
  1173. self.fPluginInfo["label"] = cString(self.fPluginInfo["label"])
  1174. self.fPluginInfo["maker"] = cString(self.fPluginInfo["maker"])
  1175. self.fPluginInfo["copyright"] = cString(self.fPluginInfo["copyright"])
  1176. self.reloadInfo()
  1177. self.reloadParameters()
  1178. self.reloadPrograms()
  1179. def reloadInfo(self):
  1180. pluginName = cString(Carla.host.get_real_plugin_name(self.fPluginId))
  1181. pluginType = self.fPluginInfo['type']
  1182. pluginHints = self.fPluginInfo['hints']
  1183. # Automatically change to MidiProgram tab
  1184. if pluginType != PLUGIN_VST and not self.ui.le_name.text():
  1185. self.ui.tab_programs.setCurrentIndex(1)
  1186. # Set Meta-Data
  1187. if pluginType == PLUGIN_INTERNAL:
  1188. self.ui.le_type.setText(self.tr("Internal"))
  1189. elif pluginType == PLUGIN_LADSPA:
  1190. self.ui.le_type.setText("LADSPA")
  1191. elif pluginType == PLUGIN_DSSI:
  1192. self.ui.le_type.setText("DSSI")
  1193. elif pluginType == PLUGIN_LV2:
  1194. self.ui.le_type.setText("LV2")
  1195. elif pluginType == PLUGIN_VST:
  1196. self.ui.le_type.setText("VST")
  1197. elif pluginType == PLUGIN_GIG:
  1198. self.ui.le_type.setText("GIG")
  1199. elif pluginType == PLUGIN_SF2:
  1200. self.ui.le_type.setText("SF2")
  1201. elif pluginType == PLUGIN_SFZ:
  1202. self.ui.le_type.setText("SFZ")
  1203. else:
  1204. self.ui.le_type.setText(self.tr("Unknown"))
  1205. self.ui.le_name.setText(pluginName)
  1206. self.ui.le_name.setToolTip(pluginName)
  1207. self.ui.le_label.setText(self.fPluginInfo['label'])
  1208. self.ui.le_label.setToolTip(self.fPluginInfo['label'])
  1209. self.ui.le_maker.setText(self.fPluginInfo['maker'])
  1210. self.ui.le_maker.setToolTip(self.fPluginInfo['maker'])
  1211. self.ui.le_copyright.setText(self.fPluginInfo['copyright'])
  1212. self.ui.le_copyright.setToolTip(self.fPluginInfo['copyright'])
  1213. self.ui.le_unique_id.setText(str(self.fPluginInfo['uniqueId']))
  1214. self.ui.le_unique_id.setToolTip(str(self.fPluginInfo['uniqueId']))
  1215. self.ui.label_plugin.setText("\n%s\n" % self.fPluginInfo['name'])
  1216. self.setWindowTitle(self.fPluginInfo['name'])
  1217. # Set Processing Data
  1218. audioCountInfo = Carla.host.get_audio_port_count_info(self.fPluginId)
  1219. midiCountInfo = Carla.host.get_midi_port_count_info(self.fPluginId)
  1220. paramCountInfo = Carla.host.get_parameter_count_info(self.fPluginId)
  1221. self.ui.le_ains.setText(str(audioCountInfo['ins']))
  1222. self.ui.le_aouts.setText(str(audioCountInfo['outs']))
  1223. self.ui.le_params.setText(str(paramCountInfo['ins']))
  1224. self.ui.le_couts.setText(str(paramCountInfo['outs']))
  1225. self.ui.le_is_synth.setText(self.tr("Yes") if (pluginHints & PLUGIN_IS_SYNTH) else self.tr("No"))
  1226. self.ui.le_has_gui.setText(self.tr("Yes") if (pluginHints & PLUGIN_HAS_GUI) else self.tr("No"))
  1227. # Show/hide keyboard
  1228. self.ui.scrollArea.setVisible((pluginHints & PLUGIN_IS_SYNTH) != 0 or (midiCountInfo['ins'] > 0 < midiCountInfo['outs']))
  1229. # Force-Update parent for new hints (knobs)
  1230. if self.fRealParent:
  1231. self.fRealParent.recheckPluginHints(pluginHints)
  1232. def reloadParameters(self):
  1233. parameterCount = Carla.host.get_parameter_count(self.fPluginId)
  1234. # Reset
  1235. self.fParameterCount = 0
  1236. self.fParameterList = []
  1237. self.fParameterIdsToUpdate = []
  1238. self.fTabIconCount = 0
  1239. self.fTabIconTimers = []
  1240. # Remove all previous parameters
  1241. for i in range(self.ui.tabWidget.count()-1):
  1242. self.ui.tabWidget.widget(1).deleteLater()
  1243. self.ui.tabWidget.removeTab(1)
  1244. if parameterCount <= 0:
  1245. pass
  1246. elif parameterCount <= Carla.maxParameters:
  1247. paramInputListFull = []
  1248. paramOutputListFull = []
  1249. paramInputList = [] # ([params], width)
  1250. paramInputWidth = 0
  1251. paramOutputList = [] # ([params], width)
  1252. paramOutputWidth = 0
  1253. for i in range(parameterCount):
  1254. paramInfo = Carla.host.get_parameter_info(self.fPluginId, i)
  1255. paramData = Carla.host.get_parameter_data(self.fPluginId, i)
  1256. paramRanges = Carla.host.get_parameter_ranges(self.fPluginId, i)
  1257. if paramData['type'] not in (PARAMETER_INPUT, PARAMETER_OUTPUT):
  1258. continue
  1259. parameter = {
  1260. 'type': paramData['type'],
  1261. 'hints': paramData['hints'],
  1262. 'name': cString(paramInfo['name']),
  1263. 'unit': cString(paramInfo['unit']),
  1264. 'scalePoints': [],
  1265. 'index': paramData['index'],
  1266. 'default': paramRanges['def'],
  1267. 'minimum': paramRanges['min'],
  1268. 'maximum': paramRanges['max'],
  1269. 'step': paramRanges['step'],
  1270. 'stepSmall': paramRanges['stepSmall'],
  1271. 'stepLarge': paramRanges['stepLarge'],
  1272. 'midiCC': paramData['midiCC'],
  1273. 'midiChannel': paramData['midiChannel'],
  1274. 'current': Carla.host.get_current_parameter_value(self.fPluginId, i)
  1275. }
  1276. for j in range(paramInfo['scalePointCount']):
  1277. scalePointInfo = Carla.host.get_parameter_scalepoint_info(self.fPluginId, i, j)
  1278. parameter['scalepoints'].append(
  1279. {
  1280. 'value': scalePointInfo['value'],
  1281. 'label': cString(scalePointInfo['label'])
  1282. })
  1283. paramInputList.append(parameter)
  1284. # -----------------------------------------------------------------
  1285. # Get width values, in packs of 10
  1286. if parameter['type'] == PARAMETER_INPUT:
  1287. paramInputWidthTMP = QFontMetrics(self.font()).width(parameter['name'])
  1288. if paramInputWidthTMP > paramInputWidth:
  1289. paramInputWidth = paramInputWidthTMP
  1290. if len(paramInputList) == 10:
  1291. paramInputListFull.append((paramInputList, paramInputWidth))
  1292. paramInputList = []
  1293. paramInputWidth = 0
  1294. else:
  1295. paramOutputWidthTMP = QFontMetrics(self.font()).width(parameter['name'])
  1296. if paramOutputWidthTMP > paramOutputWidth:
  1297. paramOutputWidth = paramOutputWidthTMP
  1298. if len(paramOutputList) == 10:
  1299. paramOutputListFull.append((paramOutputList, paramOutputWidth))
  1300. paramOutputList = []
  1301. paramOutputWidth = 0
  1302. # for i in range(parameterCount)
  1303. else:
  1304. # Final page width values
  1305. if 0 < len(paramInputList) < 10:
  1306. paramInputListFull.append((paramInputList, paramInputWidth))
  1307. if 0 < len(paramOutputList) < 10:
  1308. paramOutputListFull.append((paramOutputList, paramOutputWidth))
  1309. # -----------------------------------------------------------------
  1310. # Create parameter widgets
  1311. self._createParameterWidgets(PARAMETER_INPUT, paramInputListFull, self.tr("Parameters"))
  1312. self._createParameterWidgets(PARAMETER_OUTPUT, paramOutputListFull, self.tr("Outputs"))
  1313. else: # > Carla.maxParameters
  1314. fakeName = self.tr("This plugin has too many parameters to display here!")
  1315. paramFakeListFull = []
  1316. paramFakeList = []
  1317. paramFakeWidth = QFontMetrics(self.font()).width(fakeName)
  1318. parameter = {
  1319. 'type': PARAMETER_UNKNOWN,
  1320. 'hints': 0,
  1321. 'name': fakeName,
  1322. 'unit': "",
  1323. 'scalepoints': [],
  1324. 'index': 0,
  1325. 'default': 0,
  1326. 'minimum': 0,
  1327. 'maximum': 0,
  1328. 'step': 0,
  1329. 'stepSmall': 0,
  1330. 'stepLarge': 0,
  1331. 'midiCC': -1,
  1332. 'midiChannel': 0,
  1333. 'current': 0.0
  1334. }
  1335. paramFakeList.append(parameter)
  1336. paramFakeListFull.append((paramFakeList, paramFakeWidth))
  1337. self._createParameterWidgets(PARAMETER_UNKNOWN, paramFakeListFull, self.tr("Information"))
  1338. def reloadPrograms(self):
  1339. # Programs
  1340. self.ui.cb_programs.blockSignals(True)
  1341. self.ui.cb_programs.clear()
  1342. programCount = Carla.host.get_program_count(self.fPluginId)
  1343. if programCount > 0:
  1344. self.ui.cb_programs.setEnabled(True)
  1345. for i in range(programCount):
  1346. pName = cString(Carla.host.get_program_name(self.fPluginId, i))
  1347. self.ui.cb_programs.addItem(pName)
  1348. self.fCurrentProgram = Carla.host.get_current_program_index(self.fPluginId)
  1349. self.ui.cb_programs.setCurrentIndex(self.fCurrentProgram)
  1350. else:
  1351. self.fCurrentProgram = -1
  1352. self.ui.cb_programs.setEnabled(False)
  1353. self.ui.cb_programs.blockSignals(False)
  1354. # MIDI Programs
  1355. self.ui.cb_midi_programs.blockSignals(True)
  1356. self.ui.cb_midi_programs.clear()
  1357. midiProgramCount = Carla.host.get_midi_program_count(self.fPluginId)
  1358. if midiProgramCount > 0:
  1359. self.ui.cb_midi_programs.setEnabled(True)
  1360. for i in range(midiProgramCount):
  1361. mpData = Carla.host.get_midi_program_data(self.fPluginId, i)
  1362. mpBank = int(mpData['bank'])
  1363. mpProg = int(mpData['program'])
  1364. mpLabel = cString(mpData['label'])
  1365. self.ui.cb_midi_programs.addItem("%03i:%03i - %s" % (mpBank, mpProg, mpLabel))
  1366. self.fCurrentMidiProgram = Carla.host.get_current_midi_program_index(self.fPluginId)
  1367. self.ui.cb_midi_programs.setCurrentIndex(self.fCurrentMidiProgram)
  1368. else:
  1369. self.fCurrentMidiProgram = -1
  1370. self.ui.cb_midi_programs.setEnabled(False)
  1371. self.ui.cb_midi_programs.blockSignals(False)
  1372. def setVisible(self, yesNo):
  1373. if yesNo:
  1374. if not self.fGeometry.isNull():
  1375. self.restoreGeometry(self.fGeometry)
  1376. else:
  1377. self.fGeometry = self.saveGeometry()
  1378. QDialog.setVisible(self, yesNo)
  1379. @pyqtSlot(int, float)
  1380. def slot_parameterValueChanged(self, parameterId, value):
  1381. Carla.host.set_parameter_value(self.fPluginId, parameterId, value)
  1382. @pyqtSlot(int, int)
  1383. def slot_parameterMidiChannelChanged(self, parameterId, channel):
  1384. Carla.host.set_parameter_midi_channel(self.fPluginId, parameterId, channel-1)
  1385. @pyqtSlot(int, int)
  1386. def slot_parameterMidiCcChanged(self, parameterId, cc):
  1387. Carla.host.set_parameter_midi_cc(self.fPluginId, parameterId, cc)
  1388. @pyqtSlot(int)
  1389. def slot_programIndexChanged(self, index):
  1390. if self.fCurrentProgram != index:
  1391. self.fCurrentProgram = index
  1392. Carla.host.set_program(self.fPluginId, index)
  1393. @pyqtSlot(int)
  1394. def slot_midiProgramIndexChanged(self, index):
  1395. if self.fCurrentMidiProgram != index:
  1396. self.fCurrentMidiProgram = index
  1397. Carla.host.set_midi_program(self.fPluginId, index)
  1398. @pyqtSlot(int)
  1399. def slot_noteOn(self, note):
  1400. Carla.host.send_midi_note(self.fPluginId, 0, note, 100)
  1401. @pyqtSlot(int)
  1402. def slot_noteOff(self, note):
  1403. Carla.host.send_midi_note(self.fPluginId, 0, note, 0)
  1404. @pyqtSlot()
  1405. def slot_notesOn(self):
  1406. if self.fRealParent:
  1407. self.fRealParent.led_midi.setChecked(True)
  1408. @pyqtSlot()
  1409. def slot_notesOff(self):
  1410. if self.fRealParent:
  1411. self.fRealParent.led_midi.setChecked(False)
  1412. @pyqtSlot()
  1413. def slot_finished(self):
  1414. if self.fRealParent:
  1415. self.fRealParent.editClosed()
  1416. def _createParameterWidgets(self, paramType, paramListFull, tabPageName):
  1417. i = 1
  1418. for paramList, width in paramListFull:
  1419. if len(paramList) == 0:
  1420. break
  1421. tabIndex = self.ui.tabWidget.count()
  1422. tabPageContainer = QWidget(self.ui.tabWidget)
  1423. tabPageLayout = QVBoxLayout(tabPageContainer)
  1424. tabPageContainer.setLayout(tabPageLayout)
  1425. for paramInfo in paramList:
  1426. paramWidget = PluginParameter(tabPageContainer, paramInfo, self.fPluginId, tabIndex)
  1427. paramWidget.setLabelWidth(width)
  1428. tabPageLayout.addWidget(paramWidget)
  1429. self.fParameterList.append((paramType, paramInfo['index'], paramWidget))
  1430. if paramType == PARAMETER_INPUT:
  1431. self.connect(paramWidget, SIGNAL("valueChanged(int, double)"), SLOT("slot_parameterValueChanged(int, double)"))
  1432. self.connect(paramWidget, SIGNAL("midiChannelChanged(int, int)"), SLOT("slot_parameterMidiChannelChanged(int, int)"))
  1433. self.connect(paramWidget, SIGNAL("midiCcChanged(int, int)"), SLOT("slot_parameterMidiCcChanged(int, int)"))
  1434. tabPageLayout.addStretch()
  1435. self.ui.tabWidget.addTab(tabPageContainer, "%s (%i)" % (tabPageName, i))
  1436. i += 1
  1437. if paramType == PARAMETER_INPUT:
  1438. self.ui.tabWidget.setTabIcon(tabIndex, self.fTabIconOff)
  1439. self.fTabIconTimers.append(ICON_STATE_NULL)
  1440. def done(self, r):
  1441. QDialog.done(self, r)
  1442. self.close()
  1443. # ------------------------------------------------------------------------------------------------------------
  1444. # Plugin Widget
  1445. class PluginWidget(QFrame):
  1446. def __init__(self, parent, pluginId):
  1447. QFrame.__init__(self, parent)
  1448. self.ui = ui_carla_plugin.Ui_PluginWidget()
  1449. self.ui.setupUi(self)
  1450. self.fPluginId = pluginId
  1451. self.fPluginInfo = Carla.host.get_plugin_info(self.fPluginId)
  1452. self.fPluginInfo["binary"] = cString(self.fPluginInfo["binary"])
  1453. self.fPluginInfo["name"] = cString(self.fPluginInfo["name"])
  1454. self.fPluginInfo["label"] = cString(self.fPluginInfo["label"])
  1455. self.fPluginInfo["maker"] = cString(self.fPluginInfo["maker"])
  1456. self.fPluginInfo["copyright"] = cString(self.fPluginInfo["copyright"])
  1457. self.fParameterIconTimer = ICON_STATE_NULL
  1458. self.fLastGreenLedState = False
  1459. self.fLastBlueLedState = False
  1460. if Carla.processMode == PROCESS_MODE_CONTINUOUS_RACK:
  1461. self.fPeaksInputCount = 2
  1462. self.fPeaksOutputCount = 2
  1463. #self.ui.stackedWidget.setCurrentIndex(0)
  1464. else:
  1465. audioCountInfo = Carla.host.get_audio_port_count_info(self.fPluginId)
  1466. self.fPeaksInputCount = int(audioCountInfo['ins'])
  1467. self.fPeaksOutputCount = int(audioCountInfo['outs'])
  1468. if self.fPeaksInputCount > 2:
  1469. self.fPeaksInputCount = 2
  1470. if self.fPeaksOutputCount > 2:
  1471. self.fPeaksOutputCount = 2
  1472. #if audioCountInfo['total'] == 0:
  1473. #self.ui.stackedWidget.setCurrentIndex(1)
  1474. #else:
  1475. #self.ui.stackedWidget.setCurrentIndex(0)
  1476. # Background
  1477. self.fColorTop = QColor(60, 60, 60)
  1478. self.fColorBottom = QColor(47, 47, 47)
  1479. # Colorify
  1480. #if self.m_pluginInfo['category'] == PLUGIN_CATEGORY_SYNTH:
  1481. #self.setWidgetColor(PALETTE_COLOR_WHITE)
  1482. #elif self.m_pluginInfo['category'] == PLUGIN_CATEGORY_DELAY:
  1483. #self.setWidgetColor(PALETTE_COLOR_ORANGE)
  1484. #elif self.m_pluginInfo['category'] == PLUGIN_CATEGORY_EQ:
  1485. #self.setWidgetColor(PALETTE_COLOR_GREEN)
  1486. #elif self.m_pluginInfo['category'] == PLUGIN_CATEGORY_FILTER:
  1487. #self.setWidgetColor(PALETTE_COLOR_BLUE)
  1488. #elif self.m_pluginInfo['category'] == PLUGIN_CATEGORY_DYNAMICS:
  1489. #self.setWidgetColor(PALETTE_COLOR_PINK)
  1490. #elif self.m_pluginInfo['category'] == PLUGIN_CATEGORY_MODULATOR:
  1491. #self.setWidgetColor(PALETTE_COLOR_RED)
  1492. #elif self.m_pluginInfo['category'] == PLUGIN_CATEGORY_UTILITY:
  1493. #self.setWidgetColor(PALETTE_COLOR_YELLOW)
  1494. #elif self.m_pluginInfo['category'] == PLUGIN_CATEGORY_OTHER:
  1495. #self.setWidgetColor(PALETTE_COLOR_BROWN)
  1496. #else:
  1497. #self.setWidgetColor(PALETTE_COLOR_NONE)
  1498. self.ui.led_enable.setColor(self.ui.led_enable.BIG_RED)
  1499. self.ui.led_enable.setChecked(False)
  1500. self.ui.led_control.setColor(self.ui.led_control.YELLOW)
  1501. self.ui.led_control.setEnabled(False)
  1502. self.ui.led_midi.setColor(self.ui.led_midi.RED)
  1503. self.ui.led_midi.setEnabled(False)
  1504. self.ui.led_audio_in.setColor(self.ui.led_audio_in.GREEN)
  1505. self.ui.led_audio_in.setEnabled(False)
  1506. self.ui.led_audio_out.setColor(self.ui.led_audio_out.BLUE)
  1507. self.ui.led_audio_out.setEnabled(False)
  1508. self.ui.dial_drywet.setPixmap(3)
  1509. self.ui.dial_vol.setPixmap(3)
  1510. self.ui.dial_b_left.setPixmap(4)
  1511. self.ui.dial_b_right.setPixmap(4)
  1512. self.ui.dial_drywet.setCustomPaint(self.ui.dial_drywet.CUSTOM_PAINT_CARLA_WET)
  1513. self.ui.dial_vol.setCustomPaint(self.ui.dial_vol.CUSTOM_PAINT_CARLA_VOL)
  1514. self.ui.dial_b_left.setCustomPaint(self.ui.dial_b_left.CUSTOM_PAINT_CARLA_L)
  1515. self.ui.dial_b_right.setCustomPaint(self.ui.dial_b_right.CUSTOM_PAINT_CARLA_R)
  1516. self.ui.peak_in.setColor(self.ui.peak_in.GREEN)
  1517. self.ui.peak_in.setOrientation(self.ui.peak_in.HORIZONTAL)
  1518. self.ui.peak_out.setColor(self.ui.peak_in.BLUE)
  1519. self.ui.peak_out.setOrientation(self.ui.peak_out.HORIZONTAL)
  1520. self.ui.peak_in.setChannels(self.fPeaksInputCount)
  1521. self.ui.peak_out.setChannels(self.fPeaksOutputCount)
  1522. self.ui.label_name.setText(self.fPluginInfo['name'])
  1523. self.ui.edit_dialog = PluginEdit(self, self.fPluginId)
  1524. self.ui.edit_dialog.hide()
  1525. self.connect(self.ui.led_enable, SIGNAL("clicked(bool)"), SLOT("slot_setActive(bool)"))
  1526. self.connect(self.ui.dial_drywet, SIGNAL("sliderMoved(int)"), SLOT("slot_setDryWet(int)"))
  1527. self.connect(self.ui.dial_vol, SIGNAL("sliderMoved(int)"), SLOT("slot_setVolume(int)"))
  1528. self.connect(self.ui.dial_b_left, SIGNAL("sliderMoved(int)"), SLOT("slot_setBalanceLeft(int)"))
  1529. self.connect(self.ui.dial_b_right, SIGNAL("sliderMoved(int)"), SLOT("slot_setBalanceRight(int)"))
  1530. #self.connect(self.ui.b_gui, SIGNAL("clicked(bool)"), SLOT("slot_guiClicked(bool)"))
  1531. self.connect(self.ui.b_edit, SIGNAL("clicked(bool)"), SLOT("slot_editClicked(bool)"))
  1532. #self.connect(self.ui.b_remove, SIGNAL("clicked()"), SLOT("slot_removeClicked()"))
  1533. #self.connect(self.ui.dial_drywet, SIGNAL("customContextMenuRequested(QPoint)"), SLOT("slot_showCustomDialMenu()"))
  1534. #self.connect(self.ui.dial_vol, SIGNAL("customContextMenuRequested(QPoint)"), SLOT("slot_showCustomDialMenu()"))
  1535. #self.connect(self.ui.dial_b_left, SIGNAL("customContextMenuRequested(QPoint)"), SLOT("slot_showCustomDialMenu()"))
  1536. #self.connect(self.ui.dial_b_right, SIGNAL("customContextMenuRequested(QPoint)"), SLOT("slot_showCustomDialMenu()"))
  1537. # FIXME
  1538. self.ui.frame_controls.setVisible(False)
  1539. self.ui.pushButton.setVisible(False)
  1540. self.ui.pushButton_2.setVisible(False)
  1541. self.ui.pushButton_3.setVisible(False)
  1542. self.setMaximumHeight(48)
  1543. def setActive(self, active, sendGui=False, sendCallback=True):
  1544. if sendGui: self.ui.led_enable.setChecked(active)
  1545. if sendCallback: Carla.host.set_active(self.fPluginId, active)
  1546. if active:
  1547. self.ui.edit_dialog.ui.keyboard.allNotesOff()
  1548. def setDryWet(self, value, sendGui=False, sendCallback=True):
  1549. if sendGui: self.ui.dial_drywet.setValue(value)
  1550. if sendCallback: Carla.host.set_drywet(self.fPluginId, float(value)/1000)
  1551. message = self.tr("Output dry/wet (%i%%)" % int(value / 10))
  1552. self.ui.dial_drywet.setStatusTip(message)
  1553. #Carla.gui.statusBar().showMessage(message)
  1554. def setVolume(self, value, sendGui=False, sendCallback=True):
  1555. if sendGui: self.ui.dial_vol.setValue(value)
  1556. if sendCallback: Carla.host.set_volume(self.fPluginId, float(value)/1000)
  1557. message = self.tr("Output volume (%i%%)" % int(value / 10))
  1558. self.ui.dial_vol.setStatusTip(message)
  1559. #Carla.gui.statusBar().showMessage(message)
  1560. def setBalanceLeft(self, value, sendGui=False, sendCallback=True):
  1561. if sendGui: self.ui.dial_b_left.setValue(value)
  1562. if sendCallback: Carla.host.set_balance_left(self.fPluginId, float(value)/1000)
  1563. if value < 0:
  1564. message = self.tr("Left Panning (%i%% Left)" % int(-value / 10))
  1565. elif value > 0:
  1566. message = self.tr("Left Panning (%i%% Right)" % int(value / 10))
  1567. else:
  1568. message = self.tr("Left Panning (Center)")
  1569. self.ui.dial_b_left.setStatusTip(message)
  1570. #Carla.gui.statusBar().showMessage(message)
  1571. def setBalanceRight(self, value, sendGui=False, sendCallback=True):
  1572. if sendGui: self.ui.dial_b_right.setValue(value)
  1573. if sendCallback: Carla.host.set_balance_right(self.fPluginId, float(value)/1000)
  1574. if value < 0:
  1575. message = self.tr("Right Panning (%i%% Left)" % int(-value / 10))
  1576. elif value > 0:
  1577. message = self.tr("Right Panning (%i%% Right)" % int(value / 10))
  1578. else:
  1579. message = self.tr("Right Panning (Center)")
  1580. self.ui.dial_b_right.setStatusTip(message)
  1581. #Carla.gui.statusBar().showMessage(message)
  1582. def setPanning(self, value, sendGui=False, sendCallback=True):
  1583. #if sendGui: self.ui.dial_vol.setValue(value)
  1584. if sendCallback: Carla.host.set_panning(self.fPluginId, float(value)/1000)
  1585. #message = self.tr("Output panning (%i%%)" % int(value / 10))
  1586. #self.ui.dial_vol.setStatusTip(message)
  1587. #Carla.gui.statusBar().showMessage(message)
  1588. def editClosed(self):
  1589. self.ui.b_edit.setChecked(False)
  1590. def recheckPluginHints(self, hints):
  1591. self.fPluginInfo['hints'] = hints
  1592. self.ui.dial_drywet.setEnabled(hints & PLUGIN_CAN_DRYWET)
  1593. self.ui.dial_vol.setEnabled(hints & PLUGIN_CAN_VOLUME)
  1594. self.ui.dial_b_left.setEnabled(hints & PLUGIN_CAN_BALANCE)
  1595. self.ui.dial_b_right.setEnabled(hints & PLUGIN_CAN_BALANCE)
  1596. self.ui.b_gui.setEnabled(hints & PLUGIN_HAS_GUI)
  1597. def paintEvent(self, event):
  1598. painter = QPainter(self)
  1599. areaX = self.ui.area_right.x()
  1600. # background
  1601. #painter.setPen(self.m_colorTop)
  1602. #painter.setBrush(self.m_colorTop)
  1603. #painter.drawRect(0, 0, areaX+40, self.height())
  1604. # bottom line
  1605. painter.setPen(self.fColorBottom)
  1606. painter.setBrush(self.fColorBottom)
  1607. painter.drawRect(0, self.height()-5, areaX, 5)
  1608. # top line
  1609. painter.drawLine(0, 0, areaX+40, 0)
  1610. # name -> leds arc
  1611. path = QPainterPath()
  1612. path.moveTo(areaX-80, self.height())
  1613. path.cubicTo(areaX+40, self.height()-5, areaX-40, 30, areaX+20, 0)
  1614. path.lineTo(areaX+20, self.height())
  1615. painter.drawPath(path)
  1616. # fill the rest
  1617. painter.drawRect(areaX+20, 0, self.width(), self.height())
  1618. #painter.drawLine(0, 3, self.width(), 3)
  1619. #painter.drawLine(0, self.height() - 4, self.width(), self.height() - 4)
  1620. #painter.setPen(self.m_color2)
  1621. #painter.drawLine(0, 2, self.width(), 2)
  1622. #painter.drawLine(0, self.height() - 3, self.width(), self.height() - 3)
  1623. #painter.setPen(self.m_color3)
  1624. #painter.drawLine(0, 1, self.width(), 1)
  1625. #painter.drawLine(0, self.height() - 2, self.width(), self.height() - 2)
  1626. #painter.setPen(self.m_color4)
  1627. #painter.drawLine(0, 0, self.width(), 0)
  1628. #painter.drawLine(0, self.height() - 1, self.width(), self.height() - 1)
  1629. QFrame.paintEvent(self, event)
  1630. @pyqtSlot(bool)
  1631. def slot_setActive(self, yesNo):
  1632. self.setActive(yesNo, False, True)
  1633. @pyqtSlot(int)
  1634. def slot_setDryWet(self, value):
  1635. self.setDryWet(value, False, True)
  1636. @pyqtSlot(int)
  1637. def slot_setVolume(self, value):
  1638. self.setVolume(value, False, True)
  1639. @pyqtSlot(int)
  1640. def slot_setBalanceLeft(self, value):
  1641. self.setBalanceLeft(value, False, True)
  1642. @pyqtSlot(int)
  1643. def slot_setBalanceRight(self, value):
  1644. self.setBalanceRight(value, False, True)
  1645. @pyqtSlot(int)
  1646. def slot_setPanning(self, value):
  1647. self.setPanning(value, False, True)
  1648. @pyqtSlot(bool)
  1649. def slot_editClicked(self, show):
  1650. self.ui.edit_dialog.setVisible(show)
  1651. # ------------------------------------------------------------------------------------------------------------
  1652. # Separate Thread for Plugin Search
  1653. class SearchPluginsThread(QThread):
  1654. def __init__(self, parent):
  1655. QThread.__init__(self, parent)
  1656. self.fCheckNative = False
  1657. self.fCheckPosix32 = False
  1658. self.fCheckPosix64 = False
  1659. self.fCheckWin32 = False
  1660. self.fCheckWin64 = False
  1661. self.fCheckLADSPA = False
  1662. self.fCheckDSSI = False
  1663. self.fCheckLV2 = False
  1664. self.fCheckVST = False
  1665. self.fCheckGIG = False
  1666. self.fCheckSF2 = False
  1667. self.fCheckSFZ = False
  1668. self.fToolNative = carla_discovery_native
  1669. self.fCurCount = 0
  1670. self.fCurPercentValue = 0
  1671. self.fLastCheckValue = 0
  1672. self.fSomethingChanged = False
  1673. def somethingChanged(self):
  1674. return self.fSomethingChanged
  1675. def skipPlugin(self):
  1676. # TODO - windows and mac support
  1677. apps = " carla-discovery"
  1678. apps += " carla-discovery-native"
  1679. apps += " carla-discovery-posix32"
  1680. apps += " carla-discovery-posix64"
  1681. apps += " carla-discovery-win32.exe"
  1682. apps += " carla-discovery-win64.exe"
  1683. if LINUX:
  1684. os.system("killall -KILL %s" % apps)
  1685. def setSearchBinaryTypes(self, native, posix32, posix64, win32, win64):
  1686. self.fCheckNative = native
  1687. self.fCheckPosix32 = posix32
  1688. self.fCheckPosix64 = posix64
  1689. self.fCheckWin32 = win32
  1690. self.fCheckWin64 = win64
  1691. def setSearchPluginTypes(self, ladspa, dssi, lv2, vst, gig, sf2, sfz):
  1692. self.fCheckLADSPA = ladspa
  1693. self.fCheckDSSI = dssi
  1694. self.fCheckLV2 = lv2
  1695. self.fCheckVST = vst
  1696. self.fCheckGIG = gig
  1697. self.fCheckSF2 = sf2
  1698. self.fCheckSFZ = sfz
  1699. def setSearchToolNative(self, tool):
  1700. self.fToolNative = tool
  1701. def run(self):
  1702. self.fCurCount = 0
  1703. pluginCount = 0
  1704. settingsDB = QSettings("falkTX", "CarlaPlugins")
  1705. if self.fCheckLADSPA: pluginCount += 1
  1706. if self.fCheckDSSI: pluginCount += 1
  1707. if self.fCheckLV2: pluginCount += 1
  1708. if self.fCheckVST: pluginCount += 1
  1709. if self.fCheckNative:
  1710. self.fCurCount += pluginCount
  1711. if self.fCheckPosix32:
  1712. self.fCurCount += pluginCount
  1713. if self.fCheckPosix64:
  1714. self.fCurCount += pluginCount
  1715. if self.fCheckWin32:
  1716. self.fCurCount += pluginCount
  1717. if self.fCheckWin64:
  1718. self.fCurCount += pluginCount
  1719. if self.fToolNative:
  1720. if self.fCheckGIG: self.fCurCount += 1
  1721. if self.fCheckSF2: self.fCurCount += 1
  1722. if self.fCheckSFZ: self.fCurCount += 1
  1723. else:
  1724. self.fCheckGIG = False
  1725. self.fCheckSF2 = False
  1726. self.fCheckSFZ = False
  1727. if self.fCurCount == 0:
  1728. return
  1729. self.fCurPercentValue = 100 / self.fCurCount
  1730. self.fLastCheckValue = 0
  1731. if HAIKU:
  1732. OS = "HAIKU"
  1733. elif LINUX:
  1734. OS = "LINUX"
  1735. elif MACOS:
  1736. OS = "MACOS"
  1737. elif WINDOWS:
  1738. OS = "WINDOWS"
  1739. else:
  1740. OS = "UNKNOWN"
  1741. if self.fCheckLADSPA:
  1742. checkValue = 0
  1743. if haveLRDF:
  1744. if self.fCheckNative: checkValue += 0.1
  1745. if self.fCheckPosix32: checkValue += 0.1
  1746. if self.fCheckPosix64: checkValue += 0.1
  1747. if self.fCheckWin32: checkValue += 0.1
  1748. if self.fCheckWin64: checkValue += 0.1
  1749. rdfPadValue = self.fCurPercentValue * checkValue
  1750. if self.fCheckNative:
  1751. self._checkLADSPA(OS, carla_discovery_native)
  1752. settingsDB.setValue("Plugins/LADSPA_native", self.fLadspaPlugins)
  1753. settingsDB.sync()
  1754. if self.fCheckPosix32:
  1755. self._checkLADSPA(OS, carla_discovery_posix32)
  1756. settingsDB.setValue("Plugins/LADSPA_posix32", self.fLadspaPlugins)
  1757. settingsDB.sync()
  1758. if self.fCheckPosix64:
  1759. self._checkLADSPA(OS, carla_discovery_posix64)
  1760. settingsDB.setValue("Plugins/LADSPA_posix64", self.fLadspaPlugins)
  1761. settingsDB.sync()
  1762. if self.fCheckWin32:
  1763. self._checkLADSPA("WINDOWS", carla_discovery_win32, not WINDOWS)
  1764. settingsDB.setValue("Plugins/LADSPA_win32", self.fLadspaPlugins)
  1765. settingsDB.sync()
  1766. if self.fCheckWin64:
  1767. self._checkLADSPA("WINDOWS", carla_discovery_win64, not WINDOWS)
  1768. settingsDB.setValue("Plugins/LADSPA_win64", self.fLadspaPlugins)
  1769. settingsDB.sync()
  1770. if haveLRDF and checkValue > 0:
  1771. startValue = self.fLastCheckValue - rdfPadValue
  1772. self._pluginLook(startValue, "LADSPA RDFs...")
  1773. ladspaRdfInfo = ladspa_rdf.recheck_all_plugins(self, startValue, self.fCurPercentValue, checkValue)
  1774. SettingsDir = os.path.join(HOME, ".config", "falkTX")
  1775. fLadspa = open(os.path.join(SettingsDir, "ladspa_rdf.db"), 'w')
  1776. json.dump(ladspaRdfInfo, fLadspa)
  1777. fLadspa.close()
  1778. if self.fCheckDSSI:
  1779. if self.fCheckNative:
  1780. self._checkDSSI(OS, carla_discovery_native)
  1781. settingsDB.setValue("Plugins/DSSI_native", self.fDssiPlugins)
  1782. settingsDB.sync()
  1783. if self.fCheckPosix32:
  1784. self._checkDSSI(OS, carla_discovery_posix32)
  1785. settingsDB.setValue("Plugins/DSSI_posix32", self.fDssiPlugins)
  1786. settingsDB.sync()
  1787. if self.fCheckPosix64:
  1788. self._checkDSSI(OS, carla_discovery_posix64)
  1789. settingsDB.setValue("Plugins/DSSI_posix64", self.fDssiPlugins)
  1790. settingsDB.sync()
  1791. if self.fCheckWin32:
  1792. self._checkDSSI("WINDOWS", carla_discovery_win32, not WINDOWS)
  1793. settingsDB.setValue("Plugins/DSSI_win32", self.fDssiPlugins)
  1794. settingsDB.sync()
  1795. if self.fCheckWin64:
  1796. self._checkDSSI("WINDOWS", carla_discovery_win64, not WINDOWS)
  1797. settingsDB.setValue("Plugins/DSSI_win64", self.fDssiPlugins)
  1798. settingsDB.sync()
  1799. if self.fCheckLV2:
  1800. if self.fCheckNative:
  1801. self._checkLV2(carla_discovery_native)
  1802. settingsDB.setValue("Plugins/LV2_native", self.fLv2Plugins)
  1803. settingsDB.sync()
  1804. if self.fCheckPosix32:
  1805. self._checkLV2(carla_discovery_posix32)
  1806. settingsDB.setValue("Plugins/LV2_posix32", self.fLv2Plugins)
  1807. settingsDB.sync()
  1808. if self.fCheckPosix64:
  1809. self._checkLV2(carla_discovery_posix64)
  1810. settingsDB.setValue("Plugins/LV2_posix64", self.fLv2Plugins)
  1811. settingsDB.sync()
  1812. if self.fCheckWin32:
  1813. self._checkLV2(carla_discovery_win32, not WINDOWS)
  1814. settingsDB.setValue("Plugins/LV2_win32", self.fLv2Plugins)
  1815. settingsDB.sync()
  1816. if self.fCheckWin64:
  1817. self._checkLV2(carla_discovery_win64, not WINDOWS)
  1818. settingsDB.setValue("Plugins/LV2_win64", self.fLv2Plugins)
  1819. settingsDB.sync()
  1820. if self.fCheckVST:
  1821. if self.fCheckNative:
  1822. self._checkVST(OS, carla_discovery_native)
  1823. settingsDB.setValue("Plugins/VST_native", self.fVstPlugins)
  1824. settingsDB.sync()
  1825. if self.fCheckPosix32:
  1826. self._checkVST(OS, carla_discovery_posix32)
  1827. settingsDB.setValue("Plugins/VST_posix32", self.fVstPlugins)
  1828. settingsDB.sync()
  1829. if self.fCheckPosix64:
  1830. self._checkVST(OS, carla_discovery_posix64)
  1831. settingsDB.setValue("Plugins/VST_posix64", self.fVstPlugins)
  1832. settingsDB.sync()
  1833. if self.fCheckWin32:
  1834. self._checkVST("WINDOWS", carla_discovery_win32, not WINDOWS)
  1835. settingsDB.setValue("Plugins/VST_win32", self.fVstPlugins)
  1836. settingsDB.sync()
  1837. if self.fCheckWin64:
  1838. self._checkVST("WINDOWS", carla_discovery_win64, not WINDOWS)
  1839. settingsDB.setValue("Plugins/VST_win64", self.fVstPlugins)
  1840. settingsDB.sync()
  1841. if self.fCheckGIG:
  1842. self._checkKIT(GIG_PATH, "gig")
  1843. settingsDB.setValue("Plugins/GIG", self.fKitPlugins)
  1844. settingsDB.sync()
  1845. if self.fCheckSF2:
  1846. self._checkKIT(SF2_PATH, "sf2")
  1847. settingsDB.setValue("Plugins/SF2", self.fKitPlugins)
  1848. settingsDB.sync()
  1849. if self.fCheckSFZ:
  1850. self._checkKIT(SFZ_PATH, "sfz")
  1851. settingsDB.setValue("Plugins/SFZ", self.fKitPlugins)
  1852. settingsDB.sync()
  1853. def _checkLADSPA(self, OS, tool, isWine=False):
  1854. global LADSPA_PATH
  1855. ladspaBinaries = []
  1856. self.fLadspaPlugins = []
  1857. for iPATH in LADSPA_PATH:
  1858. binaries = findBinaries(iPATH, OS)
  1859. for binary in binaries:
  1860. if binary not in ladspaBinaries:
  1861. ladspaBinaries.append(binary)
  1862. ladspaBinaries.sort()
  1863. for i in range(len(ladspaBinaries)):
  1864. ladspa = ladspaBinaries[i]
  1865. percent = ( float(i) / len(ladspaBinaries) ) * self.fCurPercentValue
  1866. self._pluginLook((self.fLastCheckValue + percent) * 0.9, ladspa)
  1867. plugins = checkPluginLADSPA(ladspa, tool, isWine)
  1868. if plugins:
  1869. self.fLadspaPlugins.append(plugins)
  1870. self.fSomethingChanged = True
  1871. self.fLastCheckValue += self.fCurPercentValue
  1872. def _checkDSSI(self, OS, tool, isWine=False):
  1873. global DSSI_PATH
  1874. dssiBinaries = []
  1875. self.fDssiPlugins = []
  1876. for iPATH in DSSI_PATH:
  1877. binaries = findBinaries(iPATH, OS)
  1878. for binary in binaries:
  1879. if binary not in dssiBinaries:
  1880. dssiBinaries.append(binary)
  1881. dssiBinaries.sort()
  1882. for i in range(len(dssiBinaries)):
  1883. dssi = dssiBinaries[i]
  1884. percent = ( float(i) / len(dssiBinaries) ) * self.fCurPercentValue
  1885. self._pluginLook(self.fLastCheckValue + percent, dssi)
  1886. plugins = checkPluginDSSI(dssi, tool, isWine)
  1887. if plugins:
  1888. self.fDssiPlugins.append(plugins)
  1889. self.fSomethingChanged = True
  1890. self.fLastCheckValue += self.fCurPercentValue
  1891. def _checkLV2(self, tool, isWine=False):
  1892. global LV2_PATH
  1893. lv2Bundles = []
  1894. self.fLv2Plugins = []
  1895. self._pluginLook(self.fLastCheckValue, "LV2 bundles...")
  1896. for iPATH in LV2_PATH:
  1897. bundles = findLV2Bundles(iPATH)
  1898. for bundle in bundles:
  1899. if bundle not in lv2Bundles:
  1900. lv2Bundles.append(bundle)
  1901. lv2Bundles.sort()
  1902. for i in range(len(lv2Bundles)):
  1903. lv2 = lv2Bundles[i]
  1904. percent = ( float(i) / len(lv2Bundles) ) * self.fCurPercentValue
  1905. self._pluginLook(self.fLastCheckValue + percent, lv2)
  1906. plugins = checkPluginLV2(lv2, tool, isWine)
  1907. if plugins:
  1908. self.fLv2Plugins.append(plugins)
  1909. self.fSomethingChanged = True
  1910. self.fLastCheckValue += self.fCurPercentValue
  1911. def _checkVST(self, OS, tool, isWine=False):
  1912. global VST_PATH
  1913. vstBinaries = []
  1914. self.fVstPlugins = []
  1915. for iPATH in VST_PATH:
  1916. binaries = findBinaries(iPATH, OS)
  1917. for binary in binaries:
  1918. if binary not in vstBinaries:
  1919. vstBinaries.append(binary)
  1920. vstBinaries.sort()
  1921. for i in range(len(vstBinaries)):
  1922. vst = vstBinaries[i]
  1923. percent = ( float(i) / len(vstBinaries) ) * self.fCurPercentValue
  1924. self._pluginLook(self.fLastCheckValue + percent, vst)
  1925. plugins = checkPluginVST(vst, tool, isWine)
  1926. if plugins:
  1927. self.fVstPlugins.append(plugins)
  1928. self.fSomethingChanged = True
  1929. self.fLastCheckValue += self.fCurPercentValue
  1930. def _checkKIT(self, kPATH, kType):
  1931. kitFiles = []
  1932. self.fKitPlugins = []
  1933. for iPATH in kPATH:
  1934. files = findSoundKits(iPATH, kType)
  1935. for file_ in files:
  1936. if file_ not in kitFiles:
  1937. kitFiles.append(file_)
  1938. kitFiles.sort()
  1939. for i in range(len(kitFiles)):
  1940. kit = kitFiles[i]
  1941. percent = ( float(i) / len(kitFiles) ) * self.fCurPercentValue
  1942. self._pluginLook(self.fLastCheckValue + percent, kit)
  1943. if kType == "gig":
  1944. plugins = checkPluginGIG(kit, self.fToolNative)
  1945. elif kType == "sf2":
  1946. plugins = checkPluginSF2(kit, self.fToolNative)
  1947. elif kType == "sfz":
  1948. plugins = checkPluginSFZ(kit, self.fToolNative)
  1949. else:
  1950. plugins = None
  1951. if plugins:
  1952. self.fKitPlugins.append(plugins)
  1953. self.fSomethingChanged = True
  1954. self.fLastCheckValue += self.fCurPercentValue
  1955. def _pluginLook(self, percent, plugin):
  1956. self.emit(SIGNAL("PluginLook(int, QString)"), percent, plugin)
  1957. # ------------------------------------------------------------------------------------------------------------
  1958. # Plugin Refresh Dialog
  1959. class PluginRefreshW(QDialog):
  1960. def __init__(self, parent):
  1961. QDialog.__init__(self, parent)
  1962. self.ui = ui_carla_refresh.Ui_PluginRefreshW()
  1963. self.ui.setupUi(self)
  1964. self._loadSettings()
  1965. self.ui.b_skip.setVisible(False)
  1966. if HAIKU:
  1967. self.ui.ch_posix32.setText("Haiku 32bit")
  1968. self.ui.ch_posix64.setText("Haiku 64bit")
  1969. elif LINUX:
  1970. self.ui.ch_posix32.setText("Linux 32bit")
  1971. self.ui.ch_posix64.setText("Linux 64bit")
  1972. elif MACOS:
  1973. self.ui.ch_posix32.setText("MacOS 32bit")
  1974. self.ui.ch_posix64.setText("MacOS 64bit")
  1975. self.fThread = SearchPluginsThread(self)
  1976. if carla_discovery_posix32 and not WINDOWS:
  1977. self.ui.ico_posix32.setPixmap(getIcon("dialog-ok-apply").pixmap(16, 16))
  1978. else:
  1979. self.ui.ico_posix32.setPixmap(getIcon("dialog-error").pixmap(16, 16))
  1980. self.ui.ch_posix32.setChecked(False)
  1981. self.ui.ch_posix32.setEnabled(False)
  1982. if carla_discovery_posix64 and not WINDOWS:
  1983. self.ui.ico_posix64.setPixmap(getIcon("dialog-ok-apply").pixmap(16, 16))
  1984. else:
  1985. self.ui.ico_posix64.setPixmap(getIcon("dialog-error").pixmap(16, 16))
  1986. self.ui.ch_posix64.setChecked(False)
  1987. self.ui.ch_posix64.setEnabled(False)
  1988. if carla_discovery_win32:
  1989. self.ui.ico_win32.setPixmap(getIcon("dialog-ok-apply").pixmap(16, 16))
  1990. else:
  1991. self.ui.ico_win32.setPixmap(getIcon("dialog-error").pixmap(16, 16))
  1992. self.ui.ch_win32.setChecked(False)
  1993. self.ui.ch_win32.setEnabled(False)
  1994. if carla_discovery_win64:
  1995. self.ui.ico_win64.setPixmap(getIcon("dialog-ok-apply").pixmap(16, 16))
  1996. else:
  1997. self.ui.ico_win64.setPixmap(getIcon("dialog-error").pixmap(16, 16))
  1998. self.ui.ch_win64.setChecked(False)
  1999. self.ui.ch_win64.setEnabled(False)
  2000. if haveLRDF:
  2001. self.ui.ico_rdflib.setPixmap(getIcon("dialog-ok-apply").pixmap(16, 16))
  2002. else:
  2003. self.ui.ico_rdflib.setPixmap(getIcon("dialog-error").pixmap(16, 16))
  2004. hasNative = bool(carla_discovery_native)
  2005. hasNonNative = False
  2006. if WINDOWS:
  2007. if kIs64bit:
  2008. hasNative = bool(carla_discovery_win64)
  2009. hasNonNative = bool(carla_discovery_win32)
  2010. self.fThread.setSearchToolNative(carla_discovery_win64)
  2011. self.ui.ch_win64.setChecked(False)
  2012. self.ui.ch_win64.setVisible(False)
  2013. self.ui.ico_win64.setVisible(False)
  2014. self.ui.label_win64.setVisible(False)
  2015. else:
  2016. hasNative = bool(carla_discovery_win32)
  2017. hasNonNative = bool(carla_discovery_win64)
  2018. self.fThread.setSearchToolNative(carla_discovery_win32)
  2019. self.ui.ch_win32.setChecked(False)
  2020. self.ui.ch_win32.setVisible(False)
  2021. self.ui.ico_win32.setVisible(False)
  2022. self.ui.label_win32.setVisible(False)
  2023. elif LINUX or MACOS:
  2024. if kIs64bit:
  2025. hasNonNative = bool(carla_discovery_posix32 or carla_discovery_win32 or carla_discovery_win64)
  2026. self.ui.ch_posix64.setChecked(False)
  2027. self.ui.ch_posix64.setVisible(False)
  2028. self.ui.ico_posix64.setVisible(False)
  2029. self.ui.label_posix64.setVisible(False)
  2030. else:
  2031. hasNonNative = bool(carla_discovery_posix64 or carla_discovery_win32 or carla_discovery_win64)
  2032. self.ui.ch_posix32.setChecked(False)
  2033. self.ui.ch_posix32.setVisible(False)
  2034. self.ui.ico_posix32.setVisible(False)
  2035. self.ui.label_posix32.setVisible(False)
  2036. if hasNative:
  2037. self.ui.ico_native.setPixmap(getIcon("dialog-ok-apply").pixmap(16, 16))
  2038. else:
  2039. self.ui.ico_native.setPixmap(getIcon("dialog-error").pixmap(16, 16))
  2040. self.ui.ch_native.setChecked(False)
  2041. self.ui.ch_native.setEnabled(False)
  2042. self.ui.ch_gig.setChecked(False)
  2043. self.ui.ch_gig.setEnabled(False)
  2044. self.ui.ch_sf2.setChecked(False)
  2045. self.ui.ch_sf2.setEnabled(False)
  2046. self.ui.ch_sfz.setChecked(False)
  2047. self.ui.ch_sfz.setEnabled(False)
  2048. if not hasNonNative:
  2049. self.ui.ch_ladspa.setChecked(False)
  2050. self.ui.ch_ladspa.setEnabled(False)
  2051. self.ui.ch_dssi.setChecked(False)
  2052. self.ui.ch_dssi.setEnabled(False)
  2053. self.ui.ch_lv2.setChecked(False)
  2054. self.ui.ch_lv2.setEnabled(False)
  2055. self.ui.ch_vst.setChecked(False)
  2056. self.ui.ch_vst.setEnabled(False)
  2057. self.ui.b_start.setEnabled(False)
  2058. self.connect(self.ui.b_start, SIGNAL("clicked()"), SLOT("slot_start()"))
  2059. self.connect(self.ui.b_skip, SIGNAL("clicked()"), SLOT("slot_skip()"))
  2060. self.connect(self.fThread, SIGNAL("PluginLook(int, QString)"), SLOT("slot_handlePluginLook(int, QString)"))
  2061. self.connect(self.fThread, SIGNAL("finished()"), SLOT("slot_handlePluginThreadFinished()"))
  2062. @pyqtSlot()
  2063. def slot_start(self):
  2064. self.ui.progressBar.setMinimum(0)
  2065. self.ui.progressBar.setMaximum(100)
  2066. self.ui.progressBar.setValue(0)
  2067. self.ui.b_start.setEnabled(False)
  2068. self.ui.b_skip.setVisible(True)
  2069. self.ui.b_close.setVisible(False)
  2070. native, posix32, posix64, win32, win64 = (self.ui.ch_native.isChecked(), self.ui.ch_posix32.isChecked(), self.ui.ch_posix64.isChecked(), self.ui.ch_win32.isChecked(), self.ui.ch_win64.isChecked())
  2071. ladspa, dssi, lv2, vst, gig, sf2, sfz = (self.ui.ch_ladspa.isChecked(), self.ui.ch_dssi.isChecked(), self.ui.ch_lv2.isChecked(), self.ui.ch_vst.isChecked(),
  2072. self.ui.ch_gig.isChecked(), self.ui.ch_sf2.isChecked(), self.ui.ch_sfz.isChecked())
  2073. self.fThread.setSearchBinaryTypes(native, posix32, posix64, win32, win64)
  2074. self.fThread.setSearchPluginTypes(ladspa, dssi, lv2, vst, gig, sf2, sfz)
  2075. self.fThread.start()
  2076. @pyqtSlot()
  2077. def slot_skip(self):
  2078. self.fThread.skipPlugin()
  2079. @pyqtSlot(int, str)
  2080. def slot_handlePluginLook(self, percent, plugin):
  2081. self.ui.progressBar.setFormat("%s" % plugin)
  2082. self.ui.progressBar.setValue(percent)
  2083. @pyqtSlot()
  2084. def slot_handlePluginThreadFinished(self):
  2085. self.ui.progressBar.setMinimum(0)
  2086. self.ui.progressBar.setMaximum(1)
  2087. self.ui.progressBar.setValue(1)
  2088. self.ui.progressBar.setFormat(self.tr("Done"))
  2089. self.ui.b_start.setEnabled(True)
  2090. self.ui.b_skip.setVisible(False)
  2091. self.ui.b_close.setVisible(True)
  2092. def _saveSettings(self):
  2093. settings = QSettings()
  2094. settings.setValue("PluginDatabase/SearchLADSPA", self.ui.ch_ladspa.isChecked())
  2095. settings.setValue("PluginDatabase/SearchDSSI", self.ui.ch_dssi.isChecked())
  2096. settings.setValue("PluginDatabase/SearchLV2", self.ui.ch_lv2.isChecked())
  2097. settings.setValue("PluginDatabase/SearchVST", self.ui.ch_vst.isChecked())
  2098. settings.setValue("PluginDatabase/SearchGIG", self.ui.ch_gig.isChecked())
  2099. settings.setValue("PluginDatabase/SearchSF2", self.ui.ch_sf2.isChecked())
  2100. settings.setValue("PluginDatabase/SearchSFZ", self.ui.ch_sfz.isChecked())
  2101. settings.setValue("PluginDatabase/SearchNative", self.ui.ch_native.isChecked())
  2102. settings.setValue("PluginDatabase/SearchPOSIX32", self.ui.ch_posix32.isChecked())
  2103. settings.setValue("PluginDatabase/SearchPOSIX64", self.ui.ch_posix64.isChecked())
  2104. settings.setValue("PluginDatabase/SearchWin32", self.ui.ch_win32.isChecked())
  2105. settings.setValue("PluginDatabase/SearchWin64", self.ui.ch_win64.isChecked())
  2106. def _loadSettings(self):
  2107. settings = QSettings()
  2108. self.ui.ch_ladspa.setChecked(settings.value("PluginDatabase/SearchLADSPA", True, type=bool))
  2109. self.ui.ch_dssi.setChecked(settings.value("PluginDatabase/SearchDSSI", True, type=bool))
  2110. self.ui.ch_lv2.setChecked(settings.value("PluginDatabase/SearchLV2", True, type=bool))
  2111. self.ui.ch_vst.setChecked(settings.value("PluginDatabase/SearchVST", True, type=bool))
  2112. self.ui.ch_gig.setChecked(settings.value("PluginDatabase/SearchGIG", False, type=bool))
  2113. self.ui.ch_sf2.setChecked(settings.value("PluginDatabase/SearchSF2", False, type=bool))
  2114. self.ui.ch_sfz.setChecked(settings.value("PluginDatabase/SearchSFZ", False, type=bool))
  2115. self.ui.ch_native.setChecked(settings.value("PluginDatabase/SearchNative", True, type=bool))
  2116. self.ui.ch_posix32.setChecked(settings.value("PluginDatabase/SearchPOSIX32", False, type=bool))
  2117. self.ui.ch_posix64.setChecked(settings.value("PluginDatabase/SearchPOSIX64", False, type=bool))
  2118. self.ui.ch_win32.setChecked(settings.value("PluginDatabase/SearchWin32", False, type=bool))
  2119. self.ui.ch_win64.setChecked(settings.value("PluginDatabase/SearchWin64", False, type=bool))
  2120. def closeEvent(self, event):
  2121. if self.fThread.isRunning():
  2122. self.fThread.terminate()
  2123. self.fThread.wait()
  2124. self._saveSettings()
  2125. if self.fThread.somethingChanged():
  2126. self.accept()
  2127. else:
  2128. self.reject()
  2129. QDialog.closeEvent(self, event)
  2130. def done(self, r):
  2131. QDialog.done(self, r)
  2132. self.close()
  2133. # ------------------------------------------------------------------------------------------------------------
  2134. # Plugin Database Dialog
  2135. class PluginDatabaseW(QDialog):
  2136. def __init__(self, parent):
  2137. QDialog.__init__(self, parent)
  2138. self.ui = ui_carla_database.Ui_PluginDatabaseW()
  2139. self.ui.setupUi(self)
  2140. self.fLastTableIndex = 0
  2141. self.fRetPlugin = None
  2142. self.fRealParent = parent
  2143. self.ui.b_add.setEnabled(False)
  2144. if BINARY_NATIVE in (BINARY_POSIX32, BINARY_WIN32):
  2145. self.ui.ch_bridged.setText(self.tr("Bridged (64bit)"))
  2146. else:
  2147. self.ui.ch_bridged.setText(self.tr("Bridged (32bit)"))
  2148. if not (LINUX or MACOS):
  2149. self.ui.ch_bridged_wine.setChecked(False)
  2150. self.ui.ch_bridged_wine.setEnabled(False)
  2151. self._loadSettings()
  2152. self.connect(self.ui.b_add, SIGNAL("clicked()"), SLOT("slot_addPlugin()"))
  2153. self.connect(self.ui.b_refresh, SIGNAL("clicked()"), SLOT("slot_refreshPlugins()"))
  2154. self.connect(self.ui.tb_filters, SIGNAL("clicked()"), SLOT("slot_maybeShowFilters()"))
  2155. self.connect(self.ui.tableWidget, SIGNAL("currentCellChanged(int, int, int, int)"), SLOT("slot_checkPlugin(int)"))
  2156. self.connect(self.ui.tableWidget, SIGNAL("cellDoubleClicked(int, int)"), SLOT("slot_addPlugin()"))
  2157. self.connect(self.ui.lineEdit, SIGNAL("textChanged(QString)"), SLOT("slot_checkFilters()"))
  2158. self.connect(self.ui.ch_effects, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2159. self.connect(self.ui.ch_instruments, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2160. self.connect(self.ui.ch_midi, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2161. self.connect(self.ui.ch_other, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2162. self.connect(self.ui.ch_kits, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2163. self.connect(self.ui.ch_internal, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2164. self.connect(self.ui.ch_ladspa, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2165. self.connect(self.ui.ch_dssi, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2166. self.connect(self.ui.ch_lv2, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2167. self.connect(self.ui.ch_vst, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2168. self.connect(self.ui.ch_native, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2169. self.connect(self.ui.ch_bridged, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2170. self.connect(self.ui.ch_bridged_wine, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2171. self.connect(self.ui.ch_gui, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2172. self.connect(self.ui.ch_stereo, SIGNAL("clicked()"), SLOT("slot_checkFilters()"))
  2173. @pyqtSlot()
  2174. def slot_addPlugin(self):
  2175. if self.ui.tableWidget.currentRow() >= 0:
  2176. self.fRetPlugin = self.ui.tableWidget.item(self.ui.tableWidget.currentRow(), 0).pluginData
  2177. self.accept()
  2178. else:
  2179. self.reject()
  2180. @pyqtSlot(int)
  2181. def slot_checkPlugin(self, row):
  2182. self.ui.b_add.setEnabled(row >= 0)
  2183. @pyqtSlot()
  2184. def slot_checkFilters(self):
  2185. self._checkFilters()
  2186. @pyqtSlot()
  2187. def slot_maybeShowFilters(self):
  2188. self._showFilters(not self.ui.frame.isVisible())
  2189. @pyqtSlot()
  2190. def slot_refreshPlugins(self):
  2191. if PluginRefreshW(self).exec_():
  2192. self._reAddPlugins()
  2193. if self.fRealParent:
  2194. self.fRealParent.loadRDFs()
  2195. def _checkFilters(self):
  2196. text = self.ui.lineEdit.text().lower()
  2197. hideEffects = not self.ui.ch_effects.isChecked()
  2198. hideInstruments = not self.ui.ch_instruments.isChecked()
  2199. hideMidi = not self.ui.ch_midi.isChecked()
  2200. hideOther = not self.ui.ch_other.isChecked()
  2201. hideInternal = not self.ui.ch_internal.isChecked()
  2202. hideLadspa = not self.ui.ch_ladspa.isChecked()
  2203. hideDssi = not self.ui.ch_dssi.isChecked()
  2204. hideLV2 = not self.ui.ch_lv2.isChecked()
  2205. hideVST = not self.ui.ch_vst.isChecked()
  2206. hideKits = not self.ui.ch_kits.isChecked()
  2207. hideNative = not self.ui.ch_native.isChecked()
  2208. hideBridged = not self.ui.ch_bridged.isChecked()
  2209. hideBridgedWine = not self.ui.ch_bridged_wine.isChecked()
  2210. hideNonGui = self.ui.ch_gui.isChecked()
  2211. hideNonStereo = self.ui.ch_stereo.isChecked()
  2212. if HAIKU or LINUX or MACOS:
  2213. nativeBins = [BINARY_POSIX32, BINARY_POSIX64]
  2214. wineBins = [BINARY_WIN32, BINARY_WIN64]
  2215. elif WINDOWS:
  2216. nativeBins = [BINARY_WIN32, BINARY_WIN64]
  2217. wineBins = []
  2218. else:
  2219. nativeBins = []
  2220. wineBins = []
  2221. rowCount = self.ui.tableWidget.rowCount()
  2222. for i in range(rowCount):
  2223. self.ui.tableWidget.showRow(i)
  2224. plugin = self.ui.tableWidget.item(i, 0).pluginData
  2225. aIns = plugin['audio.ins']
  2226. aOuts = plugin['audio.outs']
  2227. mIns = plugin['midi.ins']
  2228. mOuts = plugin['midi.outs']
  2229. ptype = self.ui.tableWidget.item(i, 12).text()
  2230. isSynth = bool(plugin['hints'] & PLUGIN_IS_SYNTH)
  2231. isEffect = bool(aIns > 0 < aOuts and not isSynth)
  2232. isMidi = bool(aIns == 0 and aOuts == 0 and mIns > 0 < mOuts)
  2233. isKit = bool(ptype in ("GIG", "SF2", "SFZ"))
  2234. isOther = bool(not (isEffect or isSynth or isMidi or isKit))
  2235. isNative = bool(plugin['build'] == BINARY_NATIVE)
  2236. isStereo = bool(aIns == 2 and aOuts == 2) or (isSynth and aOuts == 2)
  2237. hasGui = bool(plugin['hints'] & PLUGIN_HAS_GUI)
  2238. isBridged = bool(not isNative and plugin['build'] in nativeBins)
  2239. isBridgedWine = bool(not isNative and plugin['build'] in wineBins)
  2240. if (hideEffects and isEffect):
  2241. self.ui.tableWidget.hideRow(i)
  2242. elif (hideInstruments and isSynth):
  2243. self.ui.tableWidget.hideRow(i)
  2244. elif (hideMidi and isMidi):
  2245. self.ui.tableWidget.hideRow(i)
  2246. elif (hideOther and isOther):
  2247. self.ui.tableWidget.hideRow(i)
  2248. elif (hideKits and isKit):
  2249. self.ui.tableWidget.hideRow(i)
  2250. elif (hideInternal and ptype == self.tr("Internal")):
  2251. self.ui.tableWidget.hideRow(i)
  2252. elif (hideLadspa and ptype == "LADSPA"):
  2253. self.ui.tableWidget.hideRow(i)
  2254. elif (hideDssi and ptype == "DSSI"):
  2255. self.ui.tableWidget.hideRow(i)
  2256. elif (hideLV2 and ptype == "LV2"):
  2257. self.ui.tableWidget.hideRow(i)
  2258. elif (hideVST and ptype == "VST"):
  2259. self.ui.tableWidget.hideRow(i)
  2260. elif (hideNative and isNative):
  2261. self.ui.tableWidget.hideRow(i)
  2262. elif (hideBridged and isBridged):
  2263. self.ui.tableWidget.hideRow(i)
  2264. elif (hideBridgedWine and isBridgedWine):
  2265. self.ui.tableWidget.hideRow(i)
  2266. elif (hideNonGui and not hasGui):
  2267. self.ui.tableWidget.hideRow(i)
  2268. elif (hideNonStereo and not isStereo):
  2269. self.ui.tableWidget.hideRow(i)
  2270. elif (text and not (
  2271. text in self.ui.tableWidget.item(i, 0).text().lower() or
  2272. text in self.ui.tableWidget.item(i, 1).text().lower() or
  2273. text in self.ui.tableWidget.item(i, 2).text().lower() or
  2274. text in self.ui.tableWidget.item(i, 3).text().lower() or
  2275. text in self.ui.tableWidget.item(i, 13).text().lower())):
  2276. self.ui.tableWidget.hideRow(i)
  2277. def _showFilters(self, yesNo):
  2278. self.ui.tb_filters.setArrowType(Qt.UpArrow if yesNo else Qt.DownArrow)
  2279. self.ui.frame.setVisible(yesNo)
  2280. def _reAddPlugins(self):
  2281. settingsDB = QSettings("falkTX", "CarlaPlugins")
  2282. rowCount = self.ui.tableWidget.rowCount()
  2283. for x in range(rowCount):
  2284. self.ui.tableWidget.removeRow(0)
  2285. self.fLastTableIndex = 0
  2286. self.ui.tableWidget.setSortingEnabled(False)
  2287. internalCount = 0
  2288. ladspaCount = 0
  2289. dssiCount = 0
  2290. lv2Count = 0
  2291. vstCount = 0
  2292. kitCount = 0
  2293. # ---------------------------------------------------------------------------
  2294. # Internal
  2295. internalPlugins = toList(settingsDB.value("Plugins/Internal", []))
  2296. for plugins in internalPlugins:
  2297. for plugin in plugins:
  2298. internalCount += 1
  2299. if (not Carla.isControl) and internalCount != Carla.host.get_internal_plugin_count():
  2300. internalCount = Carla.host.get_internal_plugin_count()
  2301. internalPlugins = []
  2302. for i in range(Carla.host.get_internal_plugin_count()):
  2303. descInfo = Carla.host.get_internal_plugin_info(i)
  2304. plugins = checkPluginInternal(descInfo)
  2305. if plugins:
  2306. internalPlugins.append(plugins)
  2307. settingsDB.setValue("Plugins/Internal", internalPlugins)
  2308. for plugins in internalPlugins:
  2309. for plugin in plugins:
  2310. self._addPluginToTable(plugin, self.tr("Internal"))
  2311. # ---------------------------------------------------------------------------
  2312. # LADSPA
  2313. ladspaPlugins = []
  2314. ladspaPlugins += toList(settingsDB.value("Plugins/LADSPA_native", []))
  2315. ladspaPlugins += toList(settingsDB.value("Plugins/LADSPA_posix32", []))
  2316. ladspaPlugins += toList(settingsDB.value("Plugins/LADSPA_posix64", []))
  2317. ladspaPlugins += toList(settingsDB.value("Plugins/LADSPA_win32", []))
  2318. ladspaPlugins += toList(settingsDB.value("Plugins/LADSPA_win64", []))
  2319. for plugins in ladspaPlugins:
  2320. for plugin in plugins:
  2321. self._addPluginToTable(plugin, "LADSPA")
  2322. ladspaCount += 1
  2323. # ---------------------------------------------------------------------------
  2324. # DSSI
  2325. dssiPlugins = []
  2326. dssiPlugins += toList(settingsDB.value("Plugins/DSSI_native", []))
  2327. dssiPlugins += toList(settingsDB.value("Plugins/DSSI_posix32", []))
  2328. dssiPlugins += toList(settingsDB.value("Plugins/DSSI_posix64", []))
  2329. dssiPlugins += toList(settingsDB.value("Plugins/DSSI_win32", []))
  2330. dssiPlugins += toList(settingsDB.value("Plugins/DSSI_win64", []))
  2331. for plugins in dssiPlugins:
  2332. for plugin in plugins:
  2333. self._addPluginToTable(plugin, "DSSI")
  2334. dssiCount += 1
  2335. # ---------------------------------------------------------------------------
  2336. # LV2
  2337. lv2Plugins = []
  2338. lv2Plugins += toList(settingsDB.value("Plugins/LV2_native", []))
  2339. lv2Plugins += toList(settingsDB.value("Plugins/LV2_posix32", []))
  2340. lv2Plugins += toList(settingsDB.value("Plugins/LV2_posix64", []))
  2341. lv2Plugins += toList(settingsDB.value("Plugins/LV2_win32", []))
  2342. lv2Plugins += toList(settingsDB.value("Plugins/LV2_win64", []))
  2343. for plugins in lv2Plugins:
  2344. for plugin in plugins:
  2345. self._addPluginToTable(plugin, "LV2")
  2346. lv2Count += 1
  2347. # ---------------------------------------------------------------------------
  2348. # VST
  2349. vstPlugins = []
  2350. vstPlugins += toList(settingsDB.value("Plugins/VST_native", []))
  2351. vstPlugins += toList(settingsDB.value("Plugins/VST_posix32", []))
  2352. vstPlugins += toList(settingsDB.value("Plugins/VST_posix64", []))
  2353. vstPlugins += toList(settingsDB.value("Plugins/VST_win32", []))
  2354. vstPlugins += toList(settingsDB.value("Plugins/VST_win64", []))
  2355. for plugins in vstPlugins:
  2356. for plugin in plugins:
  2357. self._addPluginToTable(plugin, "VST")
  2358. vstCount += 1
  2359. # ---------------------------------------------------------------------------
  2360. # Kits
  2361. gigs = toList(settingsDB.value("Plugins/GIG", []))
  2362. sf2s = toList(settingsDB.value("Plugins/SF2", []))
  2363. sfzs = toList(settingsDB.value("Plugins/SFZ", []))
  2364. for gig in gigs:
  2365. for gig_i in gig:
  2366. self._addPluginToTable(gig_i, "GIG")
  2367. kitCount += 1
  2368. for sf2 in sf2s:
  2369. for sf2_i in sf2:
  2370. self._addPluginToTable(sf2_i, "SF2")
  2371. kitCount += 1
  2372. for sfz in sfzs:
  2373. for sfz_i in sfz:
  2374. self._addPluginToTable(sfz_i, "SFZ")
  2375. kitCount += 1
  2376. # ---------------------------------------------------------------------------
  2377. self.ui.tableWidget.setSortingEnabled(True)
  2378. self.ui.tableWidget.sortByColumn(0, Qt.AscendingOrder)
  2379. self.ui.label.setText(self.tr("Have %i Internal, %i LADSPA, %i DSSI, %i LV2, %i VST and %i Sound Kits" % (internalCount, ladspaCount, dssiCount, lv2Count, vstCount, kitCount)))
  2380. self._checkFilters()
  2381. def _addPluginToTable(self, plugin, ptype):
  2382. index = self.fLastTableIndex
  2383. if plugin['build'] == BINARY_NATIVE:
  2384. bridgeText = self.tr("No")
  2385. else:
  2386. typeText = self.tr("Unknown")
  2387. if LINUX or MACOS:
  2388. if plugin['build'] == BINARY_POSIX32:
  2389. typeText = "32bit"
  2390. elif plugin['build'] == BINARY_POSIX64:
  2391. typeText = "64bit"
  2392. elif plugin['build'] == BINARY_WIN32:
  2393. typeText = "Windows 32bit"
  2394. elif plugin['build'] == BINARY_WIN64:
  2395. typeText = "Windows 64bit"
  2396. elif WINDOWS:
  2397. if plugin['build'] == BINARY_WIN32:
  2398. typeText = "32bit"
  2399. elif plugin['build'] == BINARY_WIN64:
  2400. typeText = "64bit"
  2401. bridgeText = self.tr("Yes (%s)" % typeText)
  2402. self.ui.tableWidget.insertRow(index)
  2403. self.ui.tableWidget.setItem(index, 0, QTableWidgetItem(plugin['name']))
  2404. self.ui.tableWidget.setItem(index, 1, QTableWidgetItem(plugin['label']))
  2405. self.ui.tableWidget.setItem(index, 2, QTableWidgetItem(plugin['maker']))
  2406. self.ui.tableWidget.setItem(index, 3, QTableWidgetItem(str(plugin['uniqueId'])))
  2407. self.ui.tableWidget.setItem(index, 4, QTableWidgetItem(str(plugin['audio.ins'])))
  2408. self.ui.tableWidget.setItem(index, 5, QTableWidgetItem(str(plugin['audio.outs'])))
  2409. self.ui.tableWidget.setItem(index, 6, QTableWidgetItem(str(plugin['parameters.ins'])))
  2410. self.ui.tableWidget.setItem(index, 7, QTableWidgetItem(str(plugin['parameters.outs'])))
  2411. self.ui.tableWidget.setItem(index, 8, QTableWidgetItem(str(plugin['programs.total'])))
  2412. self.ui.tableWidget.setItem(index, 9, QTableWidgetItem(self.tr("Yes") if (plugin['hints'] & PLUGIN_HAS_GUI) else self.tr("No")))
  2413. self.ui.tableWidget.setItem(index, 10, QTableWidgetItem(self.tr("Yes") if (plugin['hints'] & PLUGIN_IS_SYNTH) else self.tr("No")))
  2414. self.ui.tableWidget.setItem(index, 11, QTableWidgetItem(bridgeText))
  2415. self.ui.tableWidget.setItem(index, 12, QTableWidgetItem(ptype))
  2416. self.ui.tableWidget.setItem(index, 13, QTableWidgetItem(plugin['binary']))
  2417. self.ui.tableWidget.item(self.fLastTableIndex, 0).pluginData = plugin
  2418. self.fLastTableIndex += 1
  2419. def _saveSettings(self):
  2420. settings = QSettings()
  2421. settings.setValue("PluginDatabase/Geometry", self.saveGeometry())
  2422. settings.setValue("PluginDatabase/TableGeometry", self.ui.tableWidget.horizontalHeader().saveState())
  2423. settings.setValue("PluginDatabase/ShowFilters", (self.ui.tb_filters.arrowType() == Qt.UpArrow))
  2424. settings.setValue("PluginDatabase/ShowEffects", self.ui.ch_effects.isChecked())
  2425. settings.setValue("PluginDatabase/ShowInstruments", self.ui.ch_instruments.isChecked())
  2426. settings.setValue("PluginDatabase/ShowMIDI", self.ui.ch_midi.isChecked())
  2427. settings.setValue("PluginDatabase/ShowOther", self.ui.ch_other.isChecked())
  2428. settings.setValue("PluginDatabase/ShowInternal", self.ui.ch_internal.isChecked())
  2429. settings.setValue("PluginDatabase/ShowLADSPA", self.ui.ch_ladspa.isChecked())
  2430. settings.setValue("PluginDatabase/ShowDSSI", self.ui.ch_dssi.isChecked())
  2431. settings.setValue("PluginDatabase/ShowLV2", self.ui.ch_lv2.isChecked())
  2432. settings.setValue("PluginDatabase/ShowVST", self.ui.ch_vst.isChecked())
  2433. settings.setValue("PluginDatabase/ShowKits", self.ui.ch_kits.isChecked())
  2434. settings.setValue("PluginDatabase/ShowNative", self.ui.ch_native.isChecked())
  2435. settings.setValue("PluginDatabase/ShowBridged", self.ui.ch_bridged.isChecked())
  2436. settings.setValue("PluginDatabase/ShowBridgedWine", self.ui.ch_bridged_wine.isChecked())
  2437. settings.setValue("PluginDatabase/ShowHasGUI", self.ui.ch_gui.isChecked())
  2438. settings.setValue("PluginDatabase/ShowStereoOnly", self.ui.ch_stereo.isChecked())
  2439. def _loadSettings(self):
  2440. settings = QSettings()
  2441. self.restoreGeometry(settings.value("PluginDatabase/Geometry", ""))
  2442. self.ui.tableWidget.horizontalHeader().restoreState(settings.value("PluginDatabase/TableGeometry", ""))
  2443. self.ui.ch_effects.setChecked(settings.value("PluginDatabase/ShowEffects", True, type=bool))
  2444. self.ui.ch_instruments.setChecked(settings.value("PluginDatabase/ShowInstruments", True, type=bool))
  2445. self.ui.ch_midi.setChecked(settings.value("PluginDatabase/ShowMIDI", True, type=bool))
  2446. self.ui.ch_other.setChecked(settings.value("PluginDatabase/ShowOther", True, type=bool))
  2447. self.ui.ch_internal.setChecked(settings.value("PluginDatabase/ShowInternal", True, type=bool))
  2448. self.ui.ch_ladspa.setChecked(settings.value("PluginDatabase/ShowLADSPA", True, type=bool))
  2449. self.ui.ch_dssi.setChecked(settings.value("PluginDatabase/ShowDSSI", True, type=bool))
  2450. self.ui.ch_lv2.setChecked(settings.value("PluginDatabase/ShowLV2", True, type=bool))
  2451. self.ui.ch_vst.setChecked(settings.value("PluginDatabase/ShowVST", True, type=bool))
  2452. self.ui.ch_kits.setChecked(settings.value("PluginDatabase/ShowKits", True, type=bool))
  2453. self.ui.ch_native.setChecked(settings.value("PluginDatabase/ShowNative", True, type=bool))
  2454. self.ui.ch_bridged.setChecked(settings.value("PluginDatabase/ShowBridged", True, type=bool))
  2455. self.ui.ch_bridged_wine.setChecked(settings.value("PluginDatabase/ShowBridgedWine", True, type=bool))
  2456. self.ui.ch_gui.setChecked(settings.value("PluginDatabase/ShowHasGUI", False, type=bool))
  2457. self.ui.ch_stereo.setChecked(settings.value("PluginDatabase/ShowStereoOnly", False, type=bool))
  2458. self._showFilters(settings.value("PluginDatabase/ShowFilters", False, type=bool))
  2459. self._reAddPlugins()
  2460. def closeEvent(self, event):
  2461. self._saveSettings()
  2462. QDialog.closeEvent(self, event)
  2463. def done(self, r):
  2464. QDialog.done(self, r)
  2465. self.close()
  2466. # ------------------------------------------------------------------------------------------------------------
  2467. # TESTING
  2468. #from PyQt4.QtGui import QApplication
  2469. #Carla.isControl = True
  2470. #ptest = {
  2471. #'index': 0,
  2472. #'name': "",
  2473. #'symbol': "",
  2474. #'current': 0.1,
  2475. #'default': 0.3,
  2476. #'minimum': 0.0,
  2477. #'maximum': 1.0,
  2478. #'midiChannel': 7,
  2479. #'midiCC': 2,
  2480. #'type': PARAMETER_INPUT,
  2481. #'hints': PARAMETER_IS_ENABLED | PARAMETER_IS_AUTOMABLE,
  2482. #'scalepoints': [],
  2483. #'step': 0.01,
  2484. #'stepSmall': 0.001,
  2485. #'stepLarge': 0.1,
  2486. #'unit': "un",
  2487. #}
  2488. #app = QApplication(sys.argv)
  2489. #app.setApplicationName("Carla")
  2490. #app.setApplicationVersion(VERSION)
  2491. #app.setOrganizationName("falkTX")
  2492. #gui = CarlaAboutW(None)
  2493. #gui = PluginParameter(None, ptest, 0, 0)
  2494. #gui = PluginEdit(None, 0)
  2495. #gui = PluginWidget(None, 0)
  2496. #gui = PluginDatabaseW(None)
  2497. #gui.show()
  2498. #if gui.exec_():
  2499. #print(gui.fRetPlugin)
  2500. #gui = PluginRefreshW(None)
  2501. #app.exec_()