DISTRHO Plugin Framework
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.

421 lines
13KB

  1. /*
  2. * DISTRHO Plugin Framework (DPF)
  3. * Copyright (C) 2012-2022 Filipe Coelho <falktx@falktx.com>
  4. *
  5. * Permission to use, copy, modify, and/or distribute this software for any purpose with
  6. * or without fee is hereby granted, provided that the above copyright notice and this
  7. * permission notice appear in all copies.
  8. *
  9. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH REGARD
  10. * TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS. IN
  11. * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL
  12. * DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER
  13. * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
  14. * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  15. */
  16. #include "src/DistrhoPluginChecks.h"
  17. #include "src/DistrhoDefines.h"
  18. #if !defined(DGL_FILE_BROWSER_DISABLED) && !defined(DISTRHO_UI_FILE_BROWSER) && !defined(DISTRHO_OS_MAC)
  19. # define DISTRHO_PUGL_NAMESPACE_MACRO_HELPER(NS, SEP, FUNCTION) NS ## SEP ## FUNCTION
  20. # define DISTRHO_PUGL_NAMESPACE_MACRO(NS, FUNCTION) DISTRHO_PUGL_NAMESPACE_MACRO_HELPER(NS, _, FUNCTION)
  21. # define DISTRHO_FILE_BROWSER_DIALOG_EXTRA_NAMESPACE Plugin
  22. # define x_fib_add_recent DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_add_recent)
  23. # define x_fib_cfg_buttons DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_cfg_buttons)
  24. # define x_fib_cfg_filter_callback DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_cfg_filter_callback)
  25. # define x_fib_close DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_close)
  26. # define x_fib_configure DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_configure)
  27. # define x_fib_filename DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_filename)
  28. # define x_fib_free_recent DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_free_recent)
  29. # define x_fib_handle_events DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_handle_events)
  30. # define x_fib_load_recent DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_load_recent)
  31. # define x_fib_recent_at DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_recent_at)
  32. # define x_fib_recent_count DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_recent_count)
  33. # define x_fib_recent_file DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_recent_file)
  34. # define x_fib_save_recent DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_save_recent)
  35. # define x_fib_show DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_show)
  36. # define x_fib_status DISTRHO_PUGL_NAMESPACE_MACRO(Plugin, x_fib_status)
  37. # include "../extra/FileBrowserDialog.cpp"
  38. #endif
  39. #if DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  40. # if defined(DISTRHO_OS_WINDOWS)
  41. # include <winsock2.h>
  42. # include <windows.h>
  43. # elif defined(HAVE_X11)
  44. # include <X11/Xresource.h>
  45. # endif
  46. #else
  47. # include "src/TopLevelWidgetPrivateData.hpp"
  48. # include "src/WindowPrivateData.hpp"
  49. #endif
  50. #include "DistrhoUIPrivateData.hpp"
  51. START_NAMESPACE_DISTRHO
  52. /* ------------------------------------------------------------------------------------------------------------
  53. * Static data, see DistrhoUIInternal.hpp */
  54. const char* g_nextBundlePath = nullptr;
  55. #if DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  56. uintptr_t g_nextWindowId = 0;
  57. double g_nextScaleFactor = 1.0;
  58. #endif
  59. #if DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  60. /* ------------------------------------------------------------------------------------------------------------
  61. * get global scale factor */
  62. #ifdef DISTRHO_OS_MAC
  63. double getDesktopScaleFactor(uintptr_t parentWindowHandle);
  64. #else
  65. static double getDesktopScaleFactor(const uintptr_t parentWindowHandle)
  66. {
  67. // allow custom scale for testing
  68. if (const char* const scale = getenv("DPF_SCALE_FACTOR"))
  69. return std::max(1.0, std::atof(scale));
  70. #if defined(DISTRHO_OS_WINDOWS)
  71. if (const HMODULE Shcore = LoadLibraryA("Shcore.dll"))
  72. {
  73. typedef HRESULT(WINAPI* PFN_GetProcessDpiAwareness)(HANDLE, DWORD*);
  74. typedef HRESULT(WINAPI* PFN_GetScaleFactorForMonitor)(HMONITOR, DWORD*);
  75. # if defined(__GNUC__) && (__GNUC__ >= 9)
  76. # pragma GCC diagnostic push
  77. # pragma GCC diagnostic ignored "-Wcast-function-type"
  78. # endif
  79. const PFN_GetProcessDpiAwareness GetProcessDpiAwareness
  80. = (PFN_GetProcessDpiAwareness)GetProcAddress(Shcore, "GetProcessDpiAwareness");
  81. const PFN_GetScaleFactorForMonitor GetScaleFactorForMonitor
  82. = (PFN_GetScaleFactorForMonitor)GetProcAddress(Shcore, "GetScaleFactorForMonitor");
  83. # if defined(__GNUC__) && (__GNUC__ >= 9)
  84. # pragma GCC diagnostic pop
  85. # endif
  86. DWORD dpiAware = 0;
  87. if (GetProcessDpiAwareness && GetScaleFactorForMonitor
  88. && GetProcessDpiAwareness(NULL, &dpiAware) == 0 && dpiAware != 0)
  89. {
  90. const HMONITOR hMon = parentWindowHandle != 0
  91. ? MonitorFromWindow((HWND)parentWindowHandle, MONITOR_DEFAULTTOPRIMARY)
  92. : MonitorFromPoint(POINT{0,0}, MONITOR_DEFAULTTOPRIMARY);
  93. DWORD scaleFactor = 0;
  94. if (GetScaleFactorForMonitor(hMon, &scaleFactor) == 0 && scaleFactor != 0)
  95. {
  96. FreeLibrary(Shcore);
  97. return static_cast<double>(scaleFactor) / 100.0;
  98. }
  99. }
  100. FreeLibrary(Shcore);
  101. }
  102. #elif defined(HAVE_X11)
  103. ::Display* const display = XOpenDisplay(nullptr);
  104. DISTRHO_SAFE_ASSERT_RETURN(display != nullptr, 1.0);
  105. XrmInitialize();
  106. if (char* const rms = XResourceManagerString(display))
  107. {
  108. if (const XrmDatabase sdb = XrmGetStringDatabase(rms))
  109. {
  110. char* type = nullptr;
  111. XrmValue ret;
  112. if (XrmGetResource(sdb, "Xft.dpi", "String", &type, &ret)
  113. && ret.addr != nullptr
  114. && type != nullptr
  115. && std::strncmp("String", type, 6) == 0)
  116. {
  117. if (const double dpi = std::atof(ret.addr))
  118. {
  119. XCloseDisplay(display);
  120. return dpi / 96;
  121. }
  122. }
  123. }
  124. }
  125. XCloseDisplay(display);
  126. #endif
  127. return 1.0;
  128. // might be unused
  129. (void)parentWindowHandle;
  130. }
  131. #endif // !DISTRHO_OS_MAC
  132. #endif
  133. /* ------------------------------------------------------------------------------------------------------------
  134. * UI::PrivateData special handling */
  135. UI::PrivateData* UI::PrivateData::s_nextPrivateData = nullptr;
  136. #if DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  137. ExternalWindow::PrivateData
  138. #else
  139. PluginWindow&
  140. #endif
  141. UI::PrivateData::createNextWindow(UI* const ui, const uint width, const uint height)
  142. {
  143. UI::PrivateData* const pData = s_nextPrivateData;
  144. #if DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  145. pData->window = new PluginWindow(ui, pData->app);
  146. ExternalWindow::PrivateData ewData;
  147. ewData.parentWindowHandle = pData->winId;
  148. ewData.width = width;
  149. ewData.height = height;
  150. ewData.scaleFactor = pData->scaleFactor != 0.0 ? pData->scaleFactor : getDesktopScaleFactor(pData->winId);
  151. ewData.title = DISTRHO_PLUGIN_NAME;
  152. ewData.isStandalone = DISTRHO_UI_IS_STANDALONE;
  153. return ewData;
  154. #else
  155. pData->window = new PluginWindow(ui, pData->app, pData->winId, width, height, pData->scaleFactor);
  156. // If there are no callbacks, this is most likely a temporary window, so ignore idle callbacks
  157. if (pData->callbacksPtr == nullptr)
  158. pData->window->setIgnoreIdleCallbacks();
  159. return pData->window.getObject();
  160. #endif
  161. }
  162. /* ------------------------------------------------------------------------------------------------------------
  163. * UI */
  164. UI::UI(const uint width, const uint height, const bool automaticallyScaleAndSetAsMinimumSize)
  165. : UIWidget(UI::PrivateData::createNextWindow(this, width, height)),
  166. uiData(UI::PrivateData::s_nextPrivateData)
  167. {
  168. #if !DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  169. if (width != 0 && height != 0)
  170. {
  171. Widget::setSize(width, height);
  172. if (automaticallyScaleAndSetAsMinimumSize)
  173. setGeometryConstraints(width, height, true, true, true);
  174. }
  175. #else
  176. // unused
  177. (void)automaticallyScaleAndSetAsMinimumSize;
  178. #endif
  179. }
  180. UI::~UI()
  181. {
  182. }
  183. /* ------------------------------------------------------------------------------------------------------------
  184. * Host state */
  185. bool UI::isResizable() const noexcept
  186. {
  187. #if DISTRHO_UI_USER_RESIZABLE
  188. # if DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  189. return true;
  190. # else
  191. return uiData->window->isResizable();
  192. # endif
  193. #else
  194. return false;
  195. #endif
  196. }
  197. uint UI::getBackgroundColor() const noexcept
  198. {
  199. return uiData->bgColor;
  200. }
  201. uint UI::getForegroundColor() const noexcept
  202. {
  203. return uiData->fgColor;
  204. }
  205. double UI::getSampleRate() const noexcept
  206. {
  207. return uiData->sampleRate;
  208. }
  209. const char* UI::getBundlePath() const noexcept
  210. {
  211. return uiData->bundlePath;
  212. }
  213. void UI::editParameter(uint32_t index, bool started)
  214. {
  215. uiData->editParamCallback(index + uiData->parameterOffset, started);
  216. }
  217. void UI::setParameterValue(uint32_t index, float value)
  218. {
  219. uiData->setParamCallback(index + uiData->parameterOffset, value);
  220. }
  221. #if DISTRHO_PLUGIN_WANT_STATE
  222. void UI::setState(const char* key, const char* value)
  223. {
  224. uiData->setStateCallback(key, value);
  225. }
  226. #endif
  227. #if DISTRHO_PLUGIN_WANT_STATE
  228. bool UI::requestStateFile(const char* key)
  229. {
  230. return uiData->fileRequestCallback(key);
  231. }
  232. #endif
  233. #if DISTRHO_PLUGIN_WANT_MIDI_INPUT
  234. void UI::sendNote(uint8_t channel, uint8_t note, uint8_t velocity)
  235. {
  236. uiData->sendNoteCallback(channel, note, velocity);
  237. }
  238. #endif
  239. #ifndef DGL_FILE_BROWSER_DISABLED
  240. bool UI::openFileBrowser(const FileBrowserOptions& options)
  241. {
  242. # if DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  243. // TODO
  244. return false;
  245. (void)options;
  246. # else
  247. return getWindow().openFileBrowser(options);
  248. # endif
  249. }
  250. #endif
  251. #if DISTRHO_PLUGIN_WANT_DIRECT_ACCESS
  252. /* ------------------------------------------------------------------------------------------------------------
  253. * Direct DSP access */
  254. void* UI::getPluginInstancePointer() const noexcept
  255. {
  256. return uiData->dspPtr;
  257. }
  258. #endif
  259. #if DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  260. /* ------------------------------------------------------------------------------------------------------------
  261. * External UI helpers (static calls) */
  262. const char* UI::getNextBundlePath() noexcept
  263. {
  264. return g_nextBundlePath;
  265. }
  266. double UI::getNextScaleFactor() noexcept
  267. {
  268. return g_nextScaleFactor;
  269. }
  270. # if DISTRHO_PLUGIN_HAS_EMBED_UI
  271. uintptr_t UI::getNextWindowId() noexcept
  272. {
  273. return g_nextWindowId;
  274. }
  275. # endif
  276. #endif // DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  277. /* ------------------------------------------------------------------------------------------------------------
  278. * DSP/Plugin Callbacks (optional) */
  279. void UI::sampleRateChanged(double)
  280. {
  281. }
  282. /* ------------------------------------------------------------------------------------------------------------
  283. * UI Callbacks (optional) */
  284. void UI::uiScaleFactorChanged(double)
  285. {
  286. }
  287. #if !DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  288. std::vector<DGL_NAMESPACE::ClipboardDataOffer> UI::getClipboardDataOfferTypes()
  289. {
  290. return uiData->window->getClipboardDataOfferTypes();
  291. }
  292. uint32_t UI::uiClipboardDataOffer()
  293. {
  294. std::vector<DGL_NAMESPACE::ClipboardDataOffer> offers(uiData->window->getClipboardDataOfferTypes());
  295. for (std::vector<DGL_NAMESPACE::ClipboardDataOffer>::iterator it=offers.begin(), end=offers.end(); it != end;++it)
  296. {
  297. const DGL_NAMESPACE::ClipboardDataOffer offer = *it;
  298. if (std::strcmp(offer.type, "text/plain") == 0)
  299. return offer.id;
  300. }
  301. return 0;
  302. }
  303. void UI::uiFocus(bool, DGL_NAMESPACE::CrossingMode)
  304. {
  305. }
  306. void UI::uiReshape(uint, uint)
  307. {
  308. // NOTE this must be the same as Window::onReshape
  309. pData->fallbackOnResize();
  310. }
  311. #endif // !DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  312. #ifndef DGL_FILE_BROWSER_DISABLED
  313. void UI::uiFileBrowserSelected(const char*)
  314. {
  315. }
  316. #endif
  317. /* ------------------------------------------------------------------------------------------------------------
  318. * UI Resize Handling, internal */
  319. #if DISTRHO_PLUGIN_HAS_EXTERNAL_UI
  320. void UI::sizeChanged(const uint width, const uint height)
  321. {
  322. UIWidget::sizeChanged(width, height);
  323. uiData->setSizeCallback(width, height);
  324. }
  325. #else
  326. void UI::onResize(const ResizeEvent& ev)
  327. {
  328. UIWidget::onResize(ev);
  329. #ifndef DISTRHO_PLUGIN_TARGET_VST3
  330. if (uiData->initializing)
  331. return;
  332. const uint width = ev.size.getWidth();
  333. const uint height = ev.size.getHeight();
  334. uiData->setSizeCallback(width, height);
  335. #endif
  336. }
  337. // NOTE: only used for VST3
  338. void UI::requestSizeChange(const uint width, const uint height)
  339. {
  340. # ifdef DISTRHO_PLUGIN_TARGET_VST3
  341. if (uiData->initializing)
  342. uiData->window->setSizeForVST3(width, height);
  343. else
  344. uiData->setSizeCallback(width, height);
  345. # else
  346. // unused
  347. (void)width;
  348. (void)height;
  349. # endif
  350. }
  351. #endif
  352. // -----------------------------------------------------------------------------------------------------------
  353. END_NAMESPACE_DISTRHO