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.

154 lines
7.3KB

  1. /*
  2. ==============================================================================
  3. This file is part of the JUCE 7 technical preview.
  4. Copyright (c) 2022 - Raw Material Software Limited
  5. You may use this code under the terms of the GPL v3
  6. (see www.gnu.org/licenses).
  7. For the technical preview this file cannot be licensed commercially.
  8. JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
  9. EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
  10. DISCLAIMED.
  11. ==============================================================================
  12. */
  13. namespace juce
  14. {
  15. class WindowsUIAWrapper : public DeletedAtShutdown
  16. {
  17. public:
  18. bool isLoaded() const noexcept
  19. {
  20. return uiaReturnRawElementProvider != nullptr
  21. && uiaHostProviderFromHwnd != nullptr
  22. && uiaRaiseAutomationPropertyChangedEvent != nullptr
  23. && uiaRaiseAutomationEvent != nullptr
  24. && uiaClientsAreListening != nullptr
  25. && uiaDisconnectProvider != nullptr
  26. && uiaDisconnectAllProviders != nullptr;
  27. }
  28. //==============================================================================
  29. LRESULT returnRawElementProvider (HWND hwnd, WPARAM wParam, LPARAM lParam, IRawElementProviderSimple* provider)
  30. {
  31. return uiaReturnRawElementProvider != nullptr ? uiaReturnRawElementProvider (hwnd, wParam, lParam, provider)
  32. : (LRESULT) nullptr;
  33. }
  34. JUCE_COMRESULT hostProviderFromHwnd (HWND hwnd, IRawElementProviderSimple** provider)
  35. {
  36. return uiaHostProviderFromHwnd != nullptr ? uiaHostProviderFromHwnd (hwnd, provider)
  37. : (HRESULT) UIA_E_NOTSUPPORTED;
  38. }
  39. JUCE_COMRESULT raiseAutomationPropertyChangedEvent (IRawElementProviderSimple* provider, PROPERTYID propID, VARIANT oldValue, VARIANT newValue)
  40. {
  41. return uiaRaiseAutomationPropertyChangedEvent != nullptr ? uiaRaiseAutomationPropertyChangedEvent (provider, propID, oldValue, newValue)
  42. : (HRESULT) UIA_E_NOTSUPPORTED;
  43. }
  44. JUCE_COMRESULT raiseAutomationEvent (IRawElementProviderSimple* provider, EVENTID eventID)
  45. {
  46. return uiaRaiseAutomationEvent != nullptr ? uiaRaiseAutomationEvent (provider, eventID)
  47. : (HRESULT) UIA_E_NOTSUPPORTED;
  48. }
  49. BOOL clientsAreListening()
  50. {
  51. return uiaClientsAreListening != nullptr ? uiaClientsAreListening()
  52. : false;
  53. }
  54. JUCE_COMRESULT disconnectProvider (IRawElementProviderSimple* provider)
  55. {
  56. if (uiaDisconnectProvider != nullptr)
  57. {
  58. const ScopedValueSetter<IRawElementProviderSimple*> disconnectingProviderSetter (disconnectingProvider, provider);
  59. return uiaDisconnectProvider (provider);
  60. }
  61. return (HRESULT) UIA_E_NOTSUPPORTED;
  62. }
  63. JUCE_COMRESULT disconnectAllProviders()
  64. {
  65. if (uiaDisconnectAllProviders != nullptr)
  66. {
  67. const ScopedValueSetter<bool> disconnectingAllProvidersSetter (disconnectingAllProviders, true);
  68. return uiaDisconnectAllProviders();
  69. }
  70. return (HRESULT) UIA_E_NOTSUPPORTED;
  71. }
  72. //==============================================================================
  73. bool isProviderDisconnecting (IRawElementProviderSimple* provider)
  74. {
  75. return disconnectingProvider == provider || disconnectingAllProviders;
  76. }
  77. //==============================================================================
  78. JUCE_DECLARE_SINGLETON_SINGLETHREADED_MINIMAL (WindowsUIAWrapper)
  79. private:
  80. //==============================================================================
  81. WindowsUIAWrapper()
  82. {
  83. // force UIA COM library initialisation here to prevent an exception when calling methods from SendMessage()
  84. if (isLoaded())
  85. returnRawElementProvider (nullptr, 0, 0, nullptr);
  86. else
  87. jassertfalse; // UIAutomationCore could not be loaded!
  88. }
  89. ~WindowsUIAWrapper()
  90. {
  91. disconnectAllProviders();
  92. if (uiaHandle != nullptr)
  93. ::FreeLibrary (uiaHandle);
  94. clearSingletonInstance();
  95. }
  96. //==============================================================================
  97. template <typename FuncType>
  98. static FuncType getUiaFunction (HMODULE module, LPCSTR funcName)
  99. {
  100. JUCE_BEGIN_IGNORE_WARNINGS_GCC_LIKE ("-Wcast-function-type")
  101. return (FuncType) GetProcAddress (module, funcName);
  102. JUCE_END_IGNORE_WARNINGS_GCC_LIKE
  103. }
  104. //==============================================================================
  105. using UiaReturnRawElementProviderFunc = LRESULT (WINAPI*) (HWND, WPARAM, LPARAM, IRawElementProviderSimple*);
  106. using UiaHostProviderFromHwndFunc = HRESULT (WINAPI*) (HWND, IRawElementProviderSimple**);
  107. using UiaRaiseAutomationPropertyChangedEventFunc = HRESULT (WINAPI*) (IRawElementProviderSimple*, PROPERTYID, VARIANT, VARIANT);
  108. using UiaRaiseAutomationEventFunc = HRESULT (WINAPI*) (IRawElementProviderSimple*, EVENTID);
  109. using UiaClientsAreListeningFunc = BOOL (WINAPI*) ();
  110. using UiaDisconnectProviderFunc = HRESULT (WINAPI*) (IRawElementProviderSimple*);
  111. using UiaDisconnectAllProvidersFunc = HRESULT (WINAPI*) ();
  112. HMODULE uiaHandle = ::LoadLibraryA ("UIAutomationCore.dll");
  113. UiaReturnRawElementProviderFunc uiaReturnRawElementProvider = getUiaFunction<UiaReturnRawElementProviderFunc> (uiaHandle, "UiaReturnRawElementProvider");
  114. UiaHostProviderFromHwndFunc uiaHostProviderFromHwnd = getUiaFunction<UiaHostProviderFromHwndFunc> (uiaHandle, "UiaHostProviderFromHwnd");
  115. UiaRaiseAutomationPropertyChangedEventFunc uiaRaiseAutomationPropertyChangedEvent = getUiaFunction<UiaRaiseAutomationPropertyChangedEventFunc> (uiaHandle, "UiaRaiseAutomationPropertyChangedEvent");
  116. UiaRaiseAutomationEventFunc uiaRaiseAutomationEvent = getUiaFunction<UiaRaiseAutomationEventFunc> (uiaHandle, "UiaRaiseAutomationEvent");
  117. UiaClientsAreListeningFunc uiaClientsAreListening = getUiaFunction<UiaClientsAreListeningFunc> (uiaHandle, "UiaClientsAreListening");
  118. UiaDisconnectProviderFunc uiaDisconnectProvider = getUiaFunction<UiaDisconnectProviderFunc> (uiaHandle, "UiaDisconnectProvider");
  119. UiaDisconnectAllProvidersFunc uiaDisconnectAllProviders = getUiaFunction<UiaDisconnectAllProvidersFunc> (uiaHandle, "UiaDisconnectAllProviders");
  120. IRawElementProviderSimple* disconnectingProvider = nullptr;
  121. bool disconnectingAllProviders = false;
  122. //==============================================================================
  123. JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (WindowsUIAWrapper)
  124. };
  125. } // namespace juce