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.

817 lines
29KB

  1. #!/usr/bin/env python3
  2. # -*- coding: utf-8 -*-
  3. # Common Carla code
  4. # Copyright (C) 2011-2019 Filipe Coelho <falktx@falktx.com>
  5. #
  6. # This program is free software; you can redistribute it and/or
  7. # modify it under the terms of the GNU General Public License as
  8. # published by the Free Software Foundation; either version 2 of
  9. # the License, or any later version.
  10. #
  11. # This program is distributed in the hope that it will be useful,
  12. # but WITHOUT ANY WARRANTY; without even the implied warranty of
  13. # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  14. # GNU General Public License for more details.
  15. #
  16. # For a full copy of the GNU General Public License see the doc/GPL.txt file.
  17. # ------------------------------------------------------------------------------------------------------------
  18. # Config
  19. # These will be modified during install
  20. X_LIBDIR_X = None
  21. X_DATADIR_X = None
  22. # ------------------------------------------------------------------------------------------------------------
  23. # Imports (Global)
  24. import os
  25. import sys
  26. from PyQt5.Qt import PYQT_VERSION_STR
  27. from PyQt5.QtCore import qFatal, QT_VERSION, QT_VERSION_STR, qWarning, QDir, QSettings
  28. from PyQt5.QtGui import QIcon
  29. from PyQt5.QtWidgets import QFileDialog, QMessageBox
  30. # ------------------------------------------------------------------------------------------------------------
  31. # Import Signal
  32. from signal import signal, SIGINT, SIGTERM
  33. try:
  34. from signal import SIGUSR1
  35. haveSIGUSR1 = True
  36. except:
  37. haveSIGUSR1 = False
  38. # ------------------------------------------------------------------------------------------------------------
  39. # Imports (Custom)
  40. from carla_backend_qt import *
  41. # ------------------------------------------------------------------------------------------------------------
  42. # Platform specific stuff
  43. if WINDOWS:
  44. WINDIR = os.getenv("WINDIR")
  45. # ------------------------------------------------------------------------------------------------------------
  46. # Set Version
  47. VERSION = "2.0.95 (2.1-beta1)"
  48. # ------------------------------------------------------------------------------------------------------------
  49. # Set TMP
  50. envTMP = os.getenv("TMP")
  51. if envTMP is None:
  52. if WINDOWS:
  53. qWarning("TMP variable not set")
  54. TMP = QDir.tempPath()
  55. else:
  56. TMP = envTMP
  57. if not os.path.exists(TMP):
  58. qWarning("TMP does not exist")
  59. TMP = "/"
  60. del envTMP
  61. # ------------------------------------------------------------------------------------------------------------
  62. # Set HOME
  63. envHOME = os.getenv("HOME")
  64. if envHOME is None:
  65. if LINUX or MACOS:
  66. qWarning("HOME variable not set")
  67. HOME = QDir.toNativeSeparators(QDir.homePath())
  68. else:
  69. HOME = envHOME
  70. if not os.path.exists(HOME):
  71. qWarning("HOME does not exist")
  72. HOME = TMP
  73. del envHOME
  74. # ------------------------------------------------------------------------------------------------------------
  75. # Set PATH
  76. envPATH = os.getenv("PATH")
  77. if envPATH is None:
  78. qWarning("PATH variable not set")
  79. if MACOS:
  80. PATH = ("/opt/local/bin", "/usr/local/bin", "/usr/bin", "/bin")
  81. elif WINDOWS:
  82. PATH = (os.path.join(WINDIR, "system32"), WINDIR)
  83. else:
  84. PATH = ("/usr/local/bin", "/usr/bin", "/bin")
  85. else:
  86. PATH = envPATH.split(os.pathsep)
  87. del envPATH
  88. # ------------------------------------------------------------------------------------------------------------
  89. # Static MIDI CC list
  90. MIDI_CC_LIST = (
  91. "0x01 Modulation",
  92. "0x02 Breath",
  93. "0x03 (Undefined)",
  94. "0x04 Foot",
  95. "0x05 Portamento",
  96. "0x07 Volume",
  97. "0x08 Balance",
  98. "0x09 (Undefined)",
  99. "0x0A Pan",
  100. "0x0B Expression",
  101. "0x0C FX Control 1",
  102. "0x0D FX Control 2",
  103. "0x0E (Undefined)",
  104. "0x0F (Undefined)",
  105. "0x10 General Purpose 1",
  106. "0x11 General Purpose 2",
  107. "0x12 General Purpose 3",
  108. "0x13 General Purpose 4",
  109. "0x14 (Undefined)",
  110. "0x15 (Undefined)",
  111. "0x16 (Undefined)",
  112. "0x17 (Undefined)",
  113. "0x18 (Undefined)",
  114. "0x19 (Undefined)",
  115. "0x1A (Undefined)",
  116. "0x1B (Undefined)",
  117. "0x1C (Undefined)",
  118. "0x1D (Undefined)",
  119. "0x1E (Undefined)",
  120. "0x1F (Undefined)",
  121. "0x46 Control 1 [Variation]",
  122. "0x47 Control 2 [Timbre]",
  123. "0x48 Control 3 [Release]",
  124. "0x49 Control 4 [Attack]",
  125. "0x4A Control 5 [Brightness]",
  126. "0x4B Control 6 [Decay]",
  127. "0x4C Control 7 [Vib Rate]",
  128. "0x4D Control 8 [Vib Depth]",
  129. "0x4E Control 9 [Vib Delay]",
  130. "0x4F Control 10 [Undefined]",
  131. "0x50 General Purpose 5",
  132. "0x51 General Purpose 6",
  133. "0x52 General Purpose 7",
  134. "0x53 General Purpose 8",
  135. "0x54 Portamento Control",
  136. "0x5B FX 1 Depth [Reverb]",
  137. "0x5C FX 2 Depth [Tremolo]",
  138. "0x5D FX 3 Depth [Chorus]",
  139. "0x5E FX 4 Depth [Detune]",
  140. "0x5F FX 5 Depth [Phaser]"
  141. )
  142. # ------------------------------------------------------------------------------------------------------------
  143. # PatchCanvas defines
  144. CANVAS_ANTIALIASING_SMALL = 1
  145. CANVAS_EYECANDY_SMALL = 1
  146. # ------------------------------------------------------------------------------------------------------------
  147. # Carla Settings keys
  148. CARLA_KEY_MAIN_PROJECT_FOLDER = "Main/ProjectFolder" # str
  149. CARLA_KEY_MAIN_USE_PRO_THEME = "Main/UseProTheme" # bool
  150. CARLA_KEY_MAIN_PRO_THEME_COLOR = "Main/ProThemeColor" # str
  151. CARLA_KEY_MAIN_REFRESH_INTERVAL = "Main/RefreshInterval" # int
  152. CARLA_KEY_MAIN_CONFIRM_EXIT = "Main/ConfirmExit" # bool
  153. CARLA_KEY_MAIN_SHOW_LOGS = "Main/ShowLogs" # bool
  154. CARLA_KEY_MAIN_EXPERIMENTAL = "Main/Experimental" # bool
  155. CARLA_KEY_CANVAS_THEME = "Canvas/Theme" # str
  156. CARLA_KEY_CANVAS_SIZE = "Canvas/Size" # str "NxN"
  157. CARLA_KEY_CANVAS_USE_BEZIER_LINES = "Canvas/UseBezierLines" # bool
  158. CARLA_KEY_CANVAS_AUTO_HIDE_GROUPS = "Canvas/AutoHideGroups" # bool
  159. CARLA_KEY_CANVAS_AUTO_SELECT_ITEMS = "Canvas/AutoSelectItems" # bool
  160. CARLA_KEY_CANVAS_EYE_CANDY = "Canvas/EyeCandy2" # bool
  161. CARLA_KEY_CANVAS_FANCY_EYE_CANDY = "Canvas/FancyEyeCandy" # bool
  162. CARLA_KEY_CANVAS_USE_OPENGL = "Canvas/UseOpenGL" # bool
  163. CARLA_KEY_CANVAS_ANTIALIASING = "Canvas/Antialiasing" # enum
  164. CARLA_KEY_CANVAS_HQ_ANTIALIASING = "Canvas/HQAntialiasing" # bool
  165. CARLA_KEY_CANVAS_INLINE_DISPLAYS = "Canvas/InlineDisplays" # bool
  166. CARLA_KEY_CANVAS_FULL_REPAINTS = "Canvas/FullRepaints" # bool
  167. CARLA_KEY_ENGINE_DRIVER_PREFIX = "Engine/Driver-"
  168. CARLA_KEY_ENGINE_AUDIO_DRIVER = "Engine/AudioDriver" # str
  169. CARLA_KEY_ENGINE_PROCESS_MODE = "Engine/ProcessMode" # enum
  170. CARLA_KEY_ENGINE_TRANSPORT_MODE = "Engine/TransportMode" # enum
  171. CARLA_KEY_ENGINE_TRANSPORT_EXTRA = "Engine/TransportExtra" # str
  172. CARLA_KEY_ENGINE_FORCE_STEREO = "Engine/ForceStereo" # bool
  173. CARLA_KEY_ENGINE_PREFER_PLUGIN_BRIDGES = "Engine/PreferPluginBridges" # bool
  174. CARLA_KEY_ENGINE_PREFER_UI_BRIDGES = "Engine/PreferUiBridges" # bool
  175. CARLA_KEY_ENGINE_MANAGE_UIS = "Engine/ManageUIs" # bool
  176. CARLA_KEY_ENGINE_UIS_ALWAYS_ON_TOP = "Engine/UIsAlwaysOnTop" # bool
  177. CARLA_KEY_ENGINE_MAX_PARAMETERS = "Engine/MaxParameters" # int
  178. CARLA_KEY_ENGINE_RESET_XRUNS = "Engine/ResetXruns" # bool
  179. CARLA_KEY_ENGINE_UI_BRIDGES_TIMEOUT = "Engine/UiBridgesTimeout" # int
  180. CARLA_KEY_OSC_ENABLED = "OSC/Enabled"
  181. CARLA_KEY_OSC_TCP_PORT_ENABLED = "OSC/TCPEnabled"
  182. CARLA_KEY_OSC_TCP_PORT_NUMBER = "OSC/TCPNumber"
  183. CARLA_KEY_OSC_TCP_PORT_RANDOM = "OSC/TCPRandom"
  184. CARLA_KEY_OSC_UDP_PORT_ENABLED = "OSC/UDPEnabled"
  185. CARLA_KEY_OSC_UDP_PORT_NUMBER = "OSC/UDPNumber"
  186. CARLA_KEY_OSC_UDP_PORT_RANDOM = "OSC/UDPRandom"
  187. CARLA_KEY_PATHS_AUDIO = "Paths/Audio"
  188. CARLA_KEY_PATHS_MIDI = "Paths/MIDI"
  189. CARLA_KEY_PATHS_LADSPA = "Paths/LADSPA"
  190. CARLA_KEY_PATHS_DSSI = "Paths/DSSI"
  191. CARLA_KEY_PATHS_LV2 = "Paths/LV2"
  192. CARLA_KEY_PATHS_VST2 = "Paths/VST2"
  193. CARLA_KEY_PATHS_VST3 = "Paths/VST3"
  194. CARLA_KEY_PATHS_SF2 = "Paths/SF2"
  195. CARLA_KEY_PATHS_SFZ = "Paths/SFZ"
  196. CARLA_KEY_WINE_EXECUTABLE = "Wine/Executable" # str
  197. CARLA_KEY_WINE_AUTO_PREFIX = "Wine/AutoPrefix" # bool
  198. CARLA_KEY_WINE_FALLBACK_PREFIX = "Wine/FallbackPrefix" # str
  199. CARLA_KEY_WINE_RT_PRIO_ENABLED = "Wine/RtPrioEnabled" # bool
  200. CARLA_KEY_WINE_BASE_RT_PRIO = "Wine/BaseRtPrio" # int
  201. CARLA_KEY_WINE_SERVER_RT_PRIO = "Wine/ServerRtPrio" # int
  202. CARLA_KEY_EXPERIMENTAL_PLUGIN_BRIDGES = "Experimental/PluginBridges" # bool
  203. CARLA_KEY_EXPERIMENTAL_WINE_BRIDGES = "Experimental/WineBridges" # bool
  204. CARLA_KEY_EXPERIMENTAL_JACK_APPS = "Experimental/JackApplications" # bool
  205. CARLA_KEY_EXPERIMENTAL_EXPORT_LV2 = "Experimental/ExportLV2" # bool
  206. CARLA_KEY_EXPERIMENTAL_PREVENT_BAD_BEHAVIOUR = "Experimental/PreventBadBehaviour" # bool
  207. CARLA_KEY_EXPERIMENTAL_LOAD_LIB_GLOBAL = "Experimental/LoadLibGlobal" # bool
  208. # if pro theme is on and color is black
  209. CARLA_KEY_CUSTOM_PAINTING = "UseCustomPainting" # bool
  210. # ------------------------------------------------------------------------------------------------------------
  211. # Carla Settings defaults
  212. # Main
  213. CARLA_DEFAULT_MAIN_PROJECT_FOLDER = HOME
  214. CARLA_DEFAULT_MAIN_USE_PRO_THEME = True
  215. CARLA_DEFAULT_MAIN_PRO_THEME_COLOR = "Black"
  216. CARLA_DEFAULT_MAIN_REFRESH_INTERVAL = 20
  217. CARLA_DEFAULT_MAIN_CONFIRM_EXIT = True
  218. CARLA_DEFAULT_MAIN_SHOW_LOGS = bool(not WINDOWS)
  219. CARLA_DEFAULT_MAIN_EXPERIMENTAL = False
  220. # Canvas
  221. CARLA_DEFAULT_CANVAS_THEME = "Modern Dark"
  222. CARLA_DEFAULT_CANVAS_SIZE = "3100x2400"
  223. CARLA_DEFAULT_CANVAS_SIZE_WIDTH = 3100
  224. CARLA_DEFAULT_CANVAS_SIZE_HEIGHT = 2400
  225. CARLA_DEFAULT_CANVAS_USE_BEZIER_LINES = True
  226. CARLA_DEFAULT_CANVAS_AUTO_HIDE_GROUPS = True
  227. CARLA_DEFAULT_CANVAS_AUTO_SELECT_ITEMS = False
  228. CARLA_DEFAULT_CANVAS_EYE_CANDY = True
  229. CARLA_DEFAULT_CANVAS_FANCY_EYE_CANDY = False
  230. CARLA_DEFAULT_CANVAS_USE_OPENGL = False
  231. CARLA_DEFAULT_CANVAS_ANTIALIASING = CANVAS_ANTIALIASING_SMALL
  232. CARLA_DEFAULT_CANVAS_HQ_ANTIALIASING = False
  233. CARLA_DEFAULT_CANVAS_INLINE_DISPLAYS = False
  234. CARLA_DEFAULT_CANVAS_FULL_REPAINTS = False
  235. # Engine
  236. CARLA_DEFAULT_FORCE_STEREO = False
  237. CARLA_DEFAULT_PREFER_PLUGIN_BRIDGES = False
  238. CARLA_DEFAULT_PREFER_UI_BRIDGES = True
  239. CARLA_DEFAULT_MANAGE_UIS = True
  240. CARLA_DEFAULT_UIS_ALWAYS_ON_TOP = False
  241. CARLA_DEFAULT_MAX_PARAMETERS = MAX_DEFAULT_PARAMETERS
  242. CARLA_DEFAULT_RESET_XRUNS = False
  243. CARLA_DEFAULT_UI_BRIDGES_TIMEOUT = 4000
  244. CARLA_DEFAULT_AUDIO_BUFFER_SIZE = 512
  245. CARLA_DEFAULT_AUDIO_SAMPLE_RATE = 44100
  246. CARLA_DEFAULT_AUDIO_TRIPLE_BUFFER = False
  247. if WINDOWS:
  248. CARLA_DEFAULT_AUDIO_DRIVER = "DirectSound"
  249. elif MACOS:
  250. CARLA_DEFAULT_AUDIO_DRIVER = "CoreAudio"
  251. elif os.path.exists("/usr/bin/jackd") or os.path.exists("/usr/bin/jackdbus"):
  252. CARLA_DEFAULT_AUDIO_DRIVER = "JACK"
  253. else:
  254. CARLA_DEFAULT_AUDIO_DRIVER = "PulseAudio"
  255. if CARLA_DEFAULT_AUDIO_DRIVER == "JACK":
  256. CARLA_DEFAULT_PROCESS_MODE = ENGINE_PROCESS_MODE_MULTIPLE_CLIENTS
  257. CARLA_DEFAULT_TRANSPORT_MODE = ENGINE_TRANSPORT_MODE_JACK
  258. else:
  259. CARLA_DEFAULT_PROCESS_MODE = ENGINE_PROCESS_MODE_PATCHBAY
  260. CARLA_DEFAULT_TRANSPORT_MODE = ENGINE_TRANSPORT_MODE_INTERNAL
  261. # OSC
  262. CARLA_DEFAULT_OSC_ENABLED = not WINDOWS
  263. CARLA_DEFAULT_OSC_TCP_PORT_ENABLED = True
  264. CARLA_DEFAULT_OSC_TCP_PORT_NUMBER = 22752
  265. CARLA_DEFAULT_OSC_TCP_PORT_RANDOM = False
  266. CARLA_DEFAULT_OSC_UDP_PORT_ENABLED = True
  267. CARLA_DEFAULT_OSC_UDP_PORT_NUMBER = 22752
  268. CARLA_DEFAULT_OSC_UDP_PORT_RANDOM = False
  269. # Wine
  270. CARLA_DEFAULT_WINE_EXECUTABLE = "wine"
  271. CARLA_DEFAULT_WINE_AUTO_PREFIX = True
  272. CARLA_DEFAULT_WINE_FALLBACK_PREFIX = os.path.expanduser("~/.wine")
  273. CARLA_DEFAULT_WINE_RT_PRIO_ENABLED = True
  274. CARLA_DEFAULT_WINE_BASE_RT_PRIO = 15
  275. CARLA_DEFAULT_WINE_SERVER_RT_PRIO = 10
  276. # Experimental
  277. CARLA_DEFAULT_EXPERIMENTAL_PLUGIN_BRIDGES = False
  278. CARLA_DEFAULT_EXPERIMENTAL_WINE_BRIDGES = False
  279. CARLA_DEFAULT_EXPERIMENTAL_JACK_APPS = False
  280. CARLA_DEFAULT_EXPERIMENTAL_LV2_EXPORT = False
  281. CARLA_DEFAULT_EXPERIMENTAL_PREVENT_BAD_BEHAVIOUR = False
  282. CARLA_DEFAULT_EXPERIMENTAL_LOAD_LIB_GLOBAL = False
  283. # ------------------------------------------------------------------------------------------------------------
  284. # Default File Folders
  285. CARLA_DEFAULT_FILE_PATH_AUDIO = []
  286. CARLA_DEFAULT_FILE_PATH_MIDI = []
  287. # ------------------------------------------------------------------------------------------------------------
  288. # Default Plugin Folders (get)
  289. DEFAULT_LADSPA_PATH = ""
  290. DEFAULT_DSSI_PATH = ""
  291. DEFAULT_LV2_PATH = ""
  292. DEFAULT_VST2_PATH = ""
  293. DEFAULT_VST3_PATH = ""
  294. DEFAULT_SF2_PATH = ""
  295. DEFAULT_SFZ_PATH = ""
  296. if WINDOWS:
  297. splitter = ";"
  298. APPDATA = os.getenv("APPDATA")
  299. PROGRAMFILES = os.getenv("PROGRAMFILES")
  300. PROGRAMFILESx86 = os.getenv("PROGRAMFILES(x86)")
  301. COMMONPROGRAMFILES = os.getenv("COMMONPROGRAMFILES")
  302. COMMONPROGRAMFILESx86 = os.getenv("COMMONPROGRAMFILES(x86)")
  303. # Small integrity tests
  304. if not APPDATA:
  305. qFatal("APPDATA variable not set, cannot continue")
  306. sys.exit(1)
  307. if not PROGRAMFILES:
  308. qFatal("PROGRAMFILES variable not set, cannot continue")
  309. sys.exit(1)
  310. if not COMMONPROGRAMFILES:
  311. qFatal("COMMONPROGRAMFILES variable not set, cannot continue")
  312. sys.exit(1)
  313. DEFAULT_LADSPA_PATH = APPDATA + "\\LADSPA"
  314. DEFAULT_LADSPA_PATH += ";" + PROGRAMFILES + "\\LADSPA"
  315. DEFAULT_DSSI_PATH = APPDATA + "\\DSSI"
  316. DEFAULT_DSSI_PATH += ";" + PROGRAMFILES + "\\DSSI"
  317. DEFAULT_LV2_PATH = APPDATA + "\\LV2"
  318. DEFAULT_LV2_PATH += ";" + COMMONPROGRAMFILES + "\\LV2"
  319. DEFAULT_VST2_PATH = PROGRAMFILES + "\\VstPlugins"
  320. DEFAULT_VST2_PATH += ";" + PROGRAMFILES + "\\Steinberg\\VstPlugins"
  321. if kIs64bit:
  322. DEFAULT_VST2_PATH += ";" + COMMONPROGRAMFILES + "\\VST2"
  323. DEFAULT_VST3_PATH = COMMONPROGRAMFILES + "\\VST3"
  324. DEFAULT_SF2_PATH = APPDATA + "\\SF2"
  325. DEFAULT_SFZ_PATH = APPDATA + "\\SFZ"
  326. if PROGRAMFILESx86:
  327. DEFAULT_LADSPA_PATH += ";" + PROGRAMFILESx86 + "\\LADSPA"
  328. DEFAULT_DSSI_PATH += ";" + PROGRAMFILESx86 + "\\DSSI"
  329. DEFAULT_VST2_PATH += ";" + PROGRAMFILESx86 + "\\VstPlugins"
  330. DEFAULT_VST2_PATH += ";" + PROGRAMFILESx86 + "\\Steinberg\\VstPlugins"
  331. if COMMONPROGRAMFILESx86:
  332. DEFAULT_VST3_PATH += COMMONPROGRAMFILESx86 + "\\VST3"
  333. elif HAIKU:
  334. splitter = ":"
  335. DEFAULT_LADSPA_PATH = HOME + "/.ladspa"
  336. DEFAULT_LADSPA_PATH += ":/system/add-ons/media/ladspaplugins"
  337. DEFAULT_LADSPA_PATH += ":/system/lib/ladspa"
  338. DEFAULT_DSSI_PATH = HOME + "/.dssi"
  339. DEFAULT_DSSI_PATH += ":/system/add-ons/media/dssiplugins"
  340. DEFAULT_DSSI_PATH += ":/system/lib/dssi"
  341. DEFAULT_LV2_PATH = HOME + "/.lv2"
  342. DEFAULT_LV2_PATH += ":/system/add-ons/media/lv2plugins"
  343. DEFAULT_VST2_PATH = HOME + "/.vst"
  344. DEFAULT_VST2_PATH += ":/system/add-ons/media/vstplugins"
  345. DEFAULT_VST3_PATH = HOME + "/.vst3"
  346. DEFAULT_VST3_PATH += ":/system/add-ons/media/vst3plugins"
  347. elif MACOS:
  348. splitter = ":"
  349. DEFAULT_LADSPA_PATH = HOME + "/Library/Audio/Plug-Ins/LADSPA"
  350. DEFAULT_LADSPA_PATH += ":/Library/Audio/Plug-Ins/LADSPA"
  351. DEFAULT_DSSI_PATH = HOME + "/Library/Audio/Plug-Ins/DSSI"
  352. DEFAULT_DSSI_PATH += ":/Library/Audio/Plug-Ins/DSSI"
  353. DEFAULT_LV2_PATH = HOME + "/Library/Audio/Plug-Ins/LV2"
  354. DEFAULT_LV2_PATH += ":/Library/Audio/Plug-Ins/LV2"
  355. DEFAULT_VST2_PATH = HOME + "/Library/Audio/Plug-Ins/VST"
  356. DEFAULT_VST2_PATH += ":/Library/Audio/Plug-Ins/VST"
  357. DEFAULT_VST3_PATH = HOME + "/Library/Audio/Plug-Ins/VST3"
  358. DEFAULT_VST3_PATH += ":/Library/Audio/Plug-Ins/VST3"
  359. else:
  360. splitter = ":"
  361. DEFAULT_LADSPA_PATH = HOME + "/.ladspa"
  362. DEFAULT_LADSPA_PATH += ":/usr/lib/ladspa"
  363. DEFAULT_LADSPA_PATH += ":/usr/local/lib/ladspa"
  364. DEFAULT_DSSI_PATH = HOME + "/.dssi"
  365. DEFAULT_DSSI_PATH += ":/usr/lib/dssi"
  366. DEFAULT_DSSI_PATH += ":/usr/local/lib/dssi"
  367. DEFAULT_LV2_PATH = HOME + "/.lv2"
  368. DEFAULT_LV2_PATH += ":/usr/lib/lv2"
  369. DEFAULT_LV2_PATH += ":/usr/local/lib/lv2"
  370. DEFAULT_VST2_PATH = HOME + "/.vst"
  371. DEFAULT_VST2_PATH += ":/usr/lib/vst"
  372. DEFAULT_VST2_PATH += ":/usr/local/lib/vst"
  373. DEFAULT_VST2_PATH += HOME + "/.lxvst"
  374. DEFAULT_VST2_PATH += ":/usr/lib/lxvst"
  375. DEFAULT_VST2_PATH += ":/usr/local/lib/lxvst"
  376. DEFAULT_VST3_PATH = HOME + "/.vst3"
  377. DEFAULT_VST3_PATH += ":/usr/lib/vst3"
  378. DEFAULT_VST3_PATH += ":/usr/local/lib/vst3"
  379. DEFAULT_SF2_PATH = HOME + "/.sounds/sf2"
  380. DEFAULT_SF2_PATH += ":/usr/share/sounds/sf2"
  381. DEFAULT_SF2_PATH += ":/usr/share/soundfonts"
  382. DEFAULT_SFZ_PATH = HOME + "/.sounds/sfz"
  383. DEFAULT_SFZ_PATH += ":/usr/share/sounds/sfz"
  384. if not WINDOWS:
  385. winePrefix = os.getenv("WINEPREFIX")
  386. if not winePrefix:
  387. winePrefix = HOME + "/.wine"
  388. if os.path.exists(winePrefix):
  389. DEFAULT_VST2_PATH += ":" + winePrefix + "/drive_c/Program Files/VstPlugins"
  390. DEFAULT_VST3_PATH += ":" + winePrefix + "/drive_c/Program Files/Common Files/VST3"
  391. if kIs64bit and os.path.exists(winePrefix + "/drive_c/Program Files (x86)"):
  392. DEFAULT_VST2_PATH += ":" + winePrefix + "/drive_c/Program Files (x86)/VstPlugins"
  393. DEFAULT_VST3_PATH += ":" + winePrefix + "/drive_c/Program Files (x86)/Common Files/VST3"
  394. del winePrefix
  395. # ------------------------------------------------------------------------------------------------------------
  396. # Default Plugin Folders (set)
  397. readEnvVars = True
  398. if WINDOWS:
  399. # Check if running Wine. If yes, ignore env vars
  400. from winreg import ConnectRegistry, OpenKey, CloseKey, HKEY_CURRENT_USER
  401. reg = ConnectRegistry(None, HKEY_CURRENT_USER)
  402. try:
  403. key = OpenKey(reg, r"SOFTWARE\Wine")
  404. CloseKey(key)
  405. del key
  406. readEnvVars = False
  407. except:
  408. pass
  409. CloseKey(reg)
  410. del reg
  411. if readEnvVars:
  412. CARLA_DEFAULT_LADSPA_PATH = os.getenv("LADSPA_PATH", DEFAULT_LADSPA_PATH).split(splitter)
  413. CARLA_DEFAULT_DSSI_PATH = os.getenv("DSSI_PATH", DEFAULT_DSSI_PATH).split(splitter)
  414. CARLA_DEFAULT_LV2_PATH = os.getenv("LV2_PATH", DEFAULT_LV2_PATH).split(splitter)
  415. CARLA_DEFAULT_VST2_PATH = os.getenv("VST_PATH", DEFAULT_VST2_PATH).split(splitter)
  416. CARLA_DEFAULT_VST3_PATH = os.getenv("VST3_PATH", DEFAULT_VST3_PATH).split(splitter)
  417. CARLA_DEFAULT_SF2_PATH = os.getenv("SF2_PATH", DEFAULT_SF2_PATH).split(splitter)
  418. CARLA_DEFAULT_SFZ_PATH = os.getenv("SFZ_PATH", DEFAULT_SFZ_PATH).split(splitter)
  419. else:
  420. CARLA_DEFAULT_LADSPA_PATH = DEFAULT_LADSPA_PATH.split(splitter)
  421. CARLA_DEFAULT_DSSI_PATH = DEFAULT_DSSI_PATH.split(splitter)
  422. CARLA_DEFAULT_LV2_PATH = DEFAULT_LV2_PATH.split(splitter)
  423. CARLA_DEFAULT_VST2_PATH = DEFAULT_VST2_PATH.split(splitter)
  424. CARLA_DEFAULT_VST3_PATH = DEFAULT_VST3_PATH.split(splitter)
  425. CARLA_DEFAULT_SF2_PATH = DEFAULT_SF2_PATH.split(splitter)
  426. CARLA_DEFAULT_SFZ_PATH = DEFAULT_SFZ_PATH.split(splitter)
  427. # ------------------------------------------------------------------------------------------------------------
  428. # Default Plugin Folders (cleanup)
  429. del DEFAULT_LADSPA_PATH
  430. del DEFAULT_DSSI_PATH
  431. del DEFAULT_LV2_PATH
  432. del DEFAULT_VST2_PATH
  433. del DEFAULT_VST3_PATH
  434. del DEFAULT_SF2_PATH
  435. del DEFAULT_SFZ_PATH
  436. # ------------------------------------------------------------------------------------------------------------
  437. # Global Carla object
  438. class CarlaObject(object):
  439. __slots__ = [
  440. 'gui', # Host Window
  441. 'nogui', # Skip UI
  442. 'term', # Terminated by OS signal
  443. 'utils' # Utils object
  444. ]
  445. gCarla = CarlaObject()
  446. gCarla.gui = None
  447. gCarla.nogui = False
  448. gCarla.term = False
  449. gCarla.utils = None
  450. # ------------------------------------------------------------------------------------------------------------
  451. # Set CWD
  452. CWD = sys.path[0]
  453. if not CWD:
  454. CWD = os.path.dirname(sys.argv[0])
  455. # make it work with cxfreeze
  456. if os.path.isfile(CWD):
  457. CWD = os.path.dirname(CWD)
  458. CXFREEZE = True
  459. else:
  460. CXFREEZE = False
  461. # ------------------------------------------------------------------------------------------------------------
  462. # Set DLL_EXTENSION
  463. if WINDOWS:
  464. DLL_EXTENSION = "dll"
  465. elif MACOS:
  466. DLL_EXTENSION = "dylib"
  467. else:
  468. DLL_EXTENSION = "so"
  469. # ------------------------------------------------------------------------------------------------------------
  470. # Check if a value is a number (float support)
  471. def isNumber(value):
  472. try:
  473. float(value)
  474. return True
  475. except:
  476. return False
  477. # ------------------------------------------------------------------------------------------------------------
  478. # Convert a value to a list
  479. def toList(value):
  480. if value is None:
  481. return []
  482. elif not isinstance(value, list):
  483. return [value]
  484. else:
  485. return value
  486. # ------------------------------------------------------------------------------------------------------------
  487. # Get Icon from user theme, using our own as backup (Oxygen)
  488. def getIcon(icon, size = 16):
  489. return QIcon.fromTheme(icon, QIcon(":/%ix%i/%s.png" % (size, size, icon)))
  490. # ------------------------------------------------------------------------------------------------------------
  491. # Handle some basic command-line arguments shared between all carla variants
  492. def handleInitialCommandLineArguments(file):
  493. initName = os.path.basename(file) if (file is not None and os.path.dirname(file) in PATH) else sys.argv[0]
  494. libPrefix = None
  495. for arg in sys.argv[1:]:
  496. if arg.startswith("--with-appname="):
  497. initName = os.path.basename(arg.replace("--with-appname=", ""))
  498. elif arg.startswith("--with-libprefix="):
  499. libPrefix = arg.replace("--with-libprefix=", "")
  500. elif arg == "--gdb":
  501. pass
  502. elif arg in ("-n", "--n", "-no-gui", "--no-gui", "-nogui", "--nogui"):
  503. gCarla.nogui = True
  504. elif arg in ("-h", "--h", "-help", "--help"):
  505. print("Usage: %s [OPTION]... [FILE|URL]" % initName)
  506. print("")
  507. print(" where FILE can be a Carla project or preset file to be loaded, or URL if using Carla-Control")
  508. print("")
  509. print(" and OPTION can be one or more of the following:")
  510. print("")
  511. print(" --gdb \t Run Carla inside gdb.")
  512. print(" -n,--no-gui \t Run Carla headless, don't show UI.")
  513. print("")
  514. print(" -h,--help \t Print this help text and exit.")
  515. print(" -v,--version\t Print version information and exit.")
  516. print("")
  517. sys.exit(0)
  518. elif arg in ("-v", "--v", "-version", "--version"):
  519. pathBinaries, pathResources = getPaths(libPrefix)
  520. print("Using Carla version %s" % VERSION)
  521. print(" Python version: %s" % sys.version.split(" ",1)[0])
  522. print(" Qt version: %s" % QT_VERSION_STR)
  523. print(" PyQt version: %s" % PYQT_VERSION_STR)
  524. print(" Binary dir: %s" % pathBinaries)
  525. print(" Resources dir: %s" % pathResources)
  526. sys.exit(0)
  527. return (initName, libPrefix)
  528. # ------------------------------------------------------------------------------------------------------------
  529. # Get initial project file (as passed in the command-line parameters)
  530. def getInitialProjectFile(app, skipExistCheck = False):
  531. # FIXME - PyQt mishandles unicode characters, we'll use direct sys.argv for now
  532. # see https://riverbankcomputing.com/pipermail/pyqt/2015-January/035395.html
  533. #args = app.arguments()[1:]
  534. args = sys.argv[1:]
  535. for arg in args:
  536. if arg.startswith("--with-appname=") or arg.startswith("--with-libprefix=") or arg == "--gdb":
  537. continue
  538. if arg in ("-n", "--n", "-no-gui", "--no-gui", "-nogui", "--nogui"):
  539. continue
  540. arg = os.path.expanduser(arg)
  541. if skipExistCheck or os.path.exists(arg):
  542. return arg
  543. return None
  544. # ------------------------------------------------------------------------------------------------------------
  545. # Get paths (binaries, resources)
  546. def getPaths(libPrefix = None):
  547. CWDl = CWD.lower()
  548. # adjust for special distros
  549. libdir = os.path.basename(os.path.normpath(X_LIBDIR_X)) if X_LIBDIR_X else "lib"
  550. datadir = os.path.basename(os.path.normpath(X_DATADIR_X)) if X_DATADIR_X else "share"
  551. # standalone, installed system-wide linux
  552. if libPrefix is not None:
  553. pathBinaries = os.path.join(libPrefix, libdir, "carla")
  554. pathResources = os.path.join(libPrefix, datadir, "carla", "resources")
  555. # standalone, local source
  556. elif CWDl.endswith("frontend"):
  557. pathBinaries = os.path.abspath(os.path.join(CWD, "..", "..", "bin"))
  558. pathResources = os.path.join(pathBinaries, "resources")
  559. # plugin
  560. elif CWDl.endswith("resources"):
  561. # installed system-wide linux
  562. if CWDl.endswith("/share/carla/resources"):
  563. pathBinaries = os.path.abspath(os.path.join(CWD, "..", "..", "..", libdir, "carla"))
  564. pathResources = CWD
  565. # local source
  566. elif CWDl.endswith("native-plugins%sresources" % os.sep):
  567. pathBinaries = os.path.abspath(os.path.join(CWD, "..", "..", "..", "bin"))
  568. pathResources = CWD
  569. # other
  570. else:
  571. pathBinaries = os.path.abspath(os.path.join(CWD, ".."))
  572. pathResources = CWD
  573. # everything else
  574. else:
  575. pathBinaries = CWD
  576. pathResources = os.path.join(pathBinaries, "resources")
  577. return (pathBinaries, pathResources)
  578. # ------------------------------------------------------------------------------------------------------------
  579. # Signal handler
  580. # TODO move to carla_host.py or something
  581. def signalHandler(sig, frame):
  582. if sig in (SIGINT, SIGTERM):
  583. gCarla.term = True
  584. if gCarla.gui is not None:
  585. gCarla.gui.SIGTERM.emit()
  586. elif haveSIGUSR1 and sig == SIGUSR1:
  587. if gCarla.gui is not None:
  588. gCarla.gui.SIGUSR1.emit()
  589. def setUpSignals():
  590. signal(SIGINT, signalHandler)
  591. signal(SIGTERM, signalHandler)
  592. if not haveSIGUSR1:
  593. return
  594. signal(SIGUSR1, signalHandler)
  595. # ------------------------------------------------------------------------------------------------------------
  596. # QLineEdit and QPushButton combo
  597. def getAndSetPath(parent, lineEdit):
  598. newPath = QFileDialog.getExistingDirectory(parent, parent.tr("Set Path"), lineEdit.text(), QFileDialog.ShowDirsOnly)
  599. if newPath:
  600. lineEdit.setText(newPath)
  601. return newPath
  602. # ------------------------------------------------------------------------------------------------------------
  603. # Backwards-compatible horizontalAdvance/width call, depending on Qt version
  604. def fontMetricsHorizontalAdvance(fontMetrics, string):
  605. if QT_VERSION >= 0x51100:
  606. return fontMetrics.horizontalAdvance(string)
  607. return fontMetrics.width(string)
  608. # ------------------------------------------------------------------------------------------------------------
  609. # Custom QMessageBox which resizes itself to fit text
  610. class QMessageBoxWithBetterWidth(QMessageBox):
  611. def __init__(self, parent):
  612. QMessageBox.__init__(self, parent)
  613. def showEvent(self, event):
  614. fontMetrics = self.fontMetrics()
  615. lines = self.text().strip().split("\n") + self.informativeText().strip().split("\n")
  616. if len(lines) > 0:
  617. width = 0
  618. for line in lines:
  619. width = max(fontMetricsHorizontalAdvance(fontMetrics, line), width)
  620. self.layout().setColumnMinimumWidth(2, width + 12)
  621. QMessageBox.showEvent(self, event)
  622. # ------------------------------------------------------------------------------------------------------------
  623. # Safer QSettings class, which does not throw if type mismatches
  624. class QSafeSettings(QSettings):
  625. def value(self, key, defaultValue, type):
  626. if not isinstance(defaultValue, type):
  627. print("QSafeSettings.value() - defaultValue type mismatch for key", key)
  628. try:
  629. return QSettings.value(self, key, defaultValue, type)
  630. except:
  631. return defaultValue
  632. # ------------------------------------------------------------------------------------------------------------
  633. # Custom MessageBox
  634. def CustomMessageBox(parent, icon, title, text,
  635. extraText="",
  636. buttons=QMessageBox.Yes|QMessageBox.No,
  637. defButton=QMessageBox.No):
  638. msgBox = QMessageBoxWithBetterWidth(parent)
  639. msgBox.setIcon(icon)
  640. msgBox.setWindowTitle(title)
  641. msgBox.setText(text)
  642. msgBox.setInformativeText(extraText)
  643. msgBox.setStandardButtons(buttons)
  644. msgBox.setDefaultButton(defButton)
  645. return msgBox.exec_()
  646. # ------------------------------------------------------------------------------------------------------------