The JUCE cross-platform C++ framework, with DISTRHO/KXStudio specific changes
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.

112 lines
3.7KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE library.
  4. Copyright (c) 2017 - ROLI Ltd.
  5. JUCE is an open source library subject to commercial or open-source
  6. licensing.
  7. By using JUCE, you agree to the terms of both the JUCE 5 End-User License
  8. Agreement and JUCE 5 Privacy Policy (both updated and effective as of the
  9. 27th April 2017).
  10. End User License Agreement: www.juce.com/juce-5-licence
  11. Privacy Policy: www.juce.com/juce-5-privacy-policy
  12. Or: You may also use this code under the terms of the GPL v3 (see
  13. www.gnu.org/licenses).
  14. JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
  15. EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
  16. DISCLAIMED.
  17. ==============================================================================
  18. */
  19. #if JUCE_WINDOWS
  20. namespace juce
  21. {
  22. // This function is in juce_win32_Windowing.cpp
  23. extern bool offerKeyMessageToJUCEWindow (MSG&);
  24. namespace
  25. {
  26. static HHOOK mouseWheelHook = 0, keyboardHook = 0;
  27. static int numHookUsers = 0;
  28. struct WindowsHooks
  29. {
  30. WindowsHooks()
  31. {
  32. if (numHookUsers++ == 0)
  33. {
  34. mouseWheelHook = SetWindowsHookEx (WH_MOUSE, mouseWheelHookCallback,
  35. (HINSTANCE) juce::Process::getCurrentModuleInstanceHandle(),
  36. GetCurrentThreadId());
  37. keyboardHook = SetWindowsHookEx (WH_GETMESSAGE, keyboardHookCallback,
  38. (HINSTANCE) juce::Process::getCurrentModuleInstanceHandle(),
  39. GetCurrentThreadId());
  40. }
  41. }
  42. ~WindowsHooks()
  43. {
  44. if (--numHookUsers == 0)
  45. {
  46. if (mouseWheelHook != 0)
  47. {
  48. UnhookWindowsHookEx (mouseWheelHook);
  49. mouseWheelHook = 0;
  50. }
  51. if (keyboardHook != 0)
  52. {
  53. UnhookWindowsHookEx (keyboardHook);
  54. keyboardHook = 0;
  55. }
  56. }
  57. }
  58. static LRESULT CALLBACK mouseWheelHookCallback (int nCode, WPARAM wParam, LPARAM lParam)
  59. {
  60. if (nCode >= 0 && wParam == WM_MOUSEWHEEL)
  61. {
  62. // using a local copy of this struct to support old mingw libraries
  63. struct MOUSEHOOKSTRUCTEX_ : public MOUSEHOOKSTRUCT { DWORD mouseData; };
  64. auto& hs = *(MOUSEHOOKSTRUCTEX_*) lParam;
  65. if (auto* comp = Desktop::getInstance().findComponentAt ({ hs.pt.x, hs.pt.y }))
  66. if (comp->getWindowHandle() != 0)
  67. return PostMessage ((HWND) comp->getWindowHandle(), WM_MOUSEWHEEL,
  68. hs.mouseData & 0xffff0000, (hs.pt.x & 0xffff) | (hs.pt.y << 16));
  69. }
  70. return CallNextHookEx (mouseWheelHook, nCode, wParam, lParam);
  71. }
  72. static LRESULT CALLBACK keyboardHookCallback (int nCode, WPARAM wParam, LPARAM lParam)
  73. {
  74. MSG& msg = *(MSG*) lParam;
  75. if (nCode == HC_ACTION && wParam == PM_REMOVE
  76. && offerKeyMessageToJUCEWindow (msg))
  77. {
  78. zerostruct (msg);
  79. msg.message = WM_USER;
  80. return 1;
  81. }
  82. return CallNextHookEx (keyboardHook, nCode, wParam, lParam);
  83. }
  84. };
  85. }
  86. } // juce namespace
  87. #endif