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.

3094 lines
116KB

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