|
- /*
- ==============================================================================
-
- This file is part of the JUCE library.
- Copyright (c) 2020 - Raw Material Software Limited
-
- JUCE is an open source library subject to commercial or open-source
- licensing.
-
- By using JUCE, you agree to the terms of both the JUCE 6 End-User License
- Agreement and JUCE Privacy Policy (both effective as of the 16th June 2020).
-
- End User License Agreement: www.juce.com/juce-6-licence
- Privacy Policy: www.juce.com/juce-privacy-policy
-
- Or: You may also use this code under the terms of the GPL v3 (see
- www.gnu.org/licenses).
-
- JUCE IS PROVIDED "AS IS" WITHOUT ANY WARRANTY, AND ALL WARRANTIES, WHETHER
- EXPRESSED OR IMPLIED, INCLUDING MERCHANTABILITY AND FITNESS FOR PURPOSE, ARE
- DISCLAIMED.
-
- ==============================================================================
- */
-
- namespace juce
- {
-
- //==============================================================================
- class UIATextProvider : public UIAProviderBase,
- public ComBaseClassHelper<ITextProvider2>
- {
- public:
- explicit UIATextProvider (AccessibilityNativeHandle* nativeHandle)
- : UIAProviderBase (nativeHandle)
- {
- }
-
- //==============================================================================
- JUCE_COMRESULT QueryInterface (REFIID iid, void** result) override
- {
- if (iid == _uuidof (IUnknown) || iid == _uuidof (ITextProvider))
- return castToType<ITextProvider> (result);
-
- if (iid == _uuidof (ITextProvider2))
- return castToType<ITextProvider2> (result);
-
- *result = nullptr;
- return E_NOINTERFACE;
- }
-
- //=============================================================================
- JUCE_COMRESULT get_DocumentRange (ITextRangeProvider** pRetVal) override
- {
- return withTextInterface (pRetVal, [&] (const AccessibilityTextInterface& textInterface)
- {
- *pRetVal = new UIATextRangeProvider (*this, { 0, textInterface.getTotalNumCharacters() });
- return S_OK;
- });
- }
-
- JUCE_COMRESULT get_SupportedTextSelection (SupportedTextSelection* pRetVal) override
- {
- return withCheckedComArgs (pRetVal, *this, [&]
- {
- *pRetVal = SupportedTextSelection_Single;
- return S_OK;
- });
- }
-
- JUCE_COMRESULT GetSelection (SAFEARRAY** pRetVal) override
- {
- return withTextInterface (pRetVal, [&] (const AccessibilityTextInterface& textInterface)
- {
- *pRetVal = SafeArrayCreateVector (VT_UNKNOWN, 0, 1);
-
- if (pRetVal != nullptr)
- {
- auto selection = textInterface.getSelection();
- auto hasSelection = ! selection.isEmpty();
- auto cursorPos = textInterface.getTextInsertionOffset();
-
- auto* rangeProvider = new UIATextRangeProvider (*this,
- { hasSelection ? selection.getStart() : cursorPos,
- hasSelection ? selection.getEnd() : cursorPos });
-
- LONG pos = 0;
- auto hr = SafeArrayPutElement (*pRetVal, &pos, static_cast<IUnknown*> (rangeProvider));
-
- if (FAILED (hr))
- return E_FAIL;
-
- rangeProvider->Release();
- }
-
- return S_OK;
- });
- }
-
- JUCE_COMRESULT GetVisibleRanges (SAFEARRAY** pRetVal) override
- {
- return withTextInterface (pRetVal, [&] (const AccessibilityTextInterface& textInterface)
- {
- *pRetVal = SafeArrayCreateVector (VT_UNKNOWN, 0, 1);
-
- if (pRetVal != nullptr)
- {
- auto* rangeProvider = new UIATextRangeProvider (*this, { 0, textInterface.getTotalNumCharacters() });
-
- LONG pos = 0;
- auto hr = SafeArrayPutElement (*pRetVal, &pos, static_cast<IUnknown*> (rangeProvider));
-
- if (FAILED (hr))
- return E_FAIL;
-
- rangeProvider->Release();
- }
-
- return S_OK;
- });
- }
-
- JUCE_COMRESULT RangeFromChild (IRawElementProviderSimple*, ITextRangeProvider** pRetVal) override
- {
- return withCheckedComArgs (pRetVal, *this, []
- {
- return S_OK;
- });
- }
-
- JUCE_COMRESULT RangeFromPoint (UiaPoint point, ITextRangeProvider** pRetVal) override
- {
- return withTextInterface (pRetVal, [&] (const AccessibilityTextInterface& textInterface)
- {
- auto offset = textInterface.getOffsetAtPoint ({ roundToInt (point.x), roundToInt (point.y) });
-
- if (offset > 0)
- *pRetVal = new UIATextRangeProvider (*this, { offset, offset });
-
- return S_OK;
- });
- }
-
- //==============================================================================
- JUCE_COMRESULT GetCaretRange (BOOL* isActive, ITextRangeProvider** pRetVal) override
- {
- return withTextInterface (pRetVal, [&] (const AccessibilityTextInterface& textInterface)
- {
- *isActive = getHandler().hasFocus (false);
-
- auto cursorPos = textInterface.getTextInsertionOffset();
- *pRetVal = new UIATextRangeProvider (*this, { cursorPos, cursorPos });
-
- return S_OK;
- });
- }
-
- JUCE_COMRESULT RangeFromAnnotation (IRawElementProviderSimple*, ITextRangeProvider** pRetVal) override
- {
- return withCheckedComArgs (pRetVal, *this, []
- {
- return S_OK;
- });
- }
-
- private:
- //==============================================================================
- template <typename Value, typename Callback>
- JUCE_COMRESULT withTextInterface (Value* pRetVal, Callback&& callback) const
- {
- return withCheckedComArgs (pRetVal, *this, [&]() -> HRESULT
- {
- if (auto* textInterface = getHandler().getTextInterface())
- return callback (*textInterface);
-
- return UIA_E_NOTSUPPORTED;
- });
- }
-
- //==============================================================================
- class UIATextRangeProvider : public UIAProviderBase,
- public ComBaseClassHelper<ITextRangeProvider>
- {
- public:
- UIATextRangeProvider (UIATextProvider& textProvider, Range<int> range)
- : UIAProviderBase (textProvider.getHandler().getNativeImplementation()),
- owner (&textProvider),
- selectionRange (range)
- {
- }
-
- //==============================================================================
- Range<int> getSelectionRange() const noexcept { return selectionRange; }
-
- //==============================================================================
- JUCE_COMRESULT AddToSelection() override
- {
- return Select();
- }
-
- JUCE_COMRESULT Clone (ITextRangeProvider** pRetVal) override
- {
- return withCheckedComArgs (pRetVal, *this, [&]
- {
- *pRetVal = new UIATextRangeProvider (*owner, selectionRange);
- return S_OK;
- });
- }
-
- JUCE_COMRESULT Compare (ITextRangeProvider* range, BOOL* pRetVal) override
- {
- return withCheckedComArgs (pRetVal, *this, [&]
- {
- *pRetVal = (selectionRange == static_cast<UIATextRangeProvider*> (range)->getSelectionRange());
- return S_OK;
- });
- }
-
- JUCE_COMRESULT CompareEndpoints (TextPatternRangeEndpoint endpoint,
- ITextRangeProvider* targetRange,
- TextPatternRangeEndpoint targetEndpoint,
- int* pRetVal) override
- {
- if (targetRange == nullptr)
- return E_INVALIDARG;
-
- return withCheckedComArgs (pRetVal, *this, [&]
- {
- auto offset = (endpoint == TextPatternRangeEndpoint_Start ? selectionRange.getStart()
- : selectionRange.getEnd());
-
- auto otherRange = static_cast<UIATextRangeProvider*> (targetRange)->getSelectionRange();
- auto otherOffset = (targetEndpoint == TextPatternRangeEndpoint_Start ? otherRange.getStart()
- : otherRange.getEnd());
-
- *pRetVal = offset - otherOffset;
- return S_OK;
- });
- }
-
- JUCE_COMRESULT ExpandToEnclosingUnit (TextUnit unit) override
- {
- if (! isElementValid())
- return UIA_E_ELEMENTNOTAVAILABLE;
-
- if (auto* textInterface = getHandler().getTextInterface())
- {
- auto numCharacters = textInterface->getTotalNumCharacters();
-
- if (numCharacters == 0)
- {
- selectionRange = {};
- return S_OK;
- }
-
- if (unit == TextUnit_Character)
- {
- selectionRange.setStart (jlimit (0, numCharacters - 1, selectionRange.getStart()));
- selectionRange.setEnd (selectionRange.getStart() + 1);
-
- return S_OK;
- }
-
- if (unit == TextUnit_Paragraph
- || unit == TextUnit_Page
- || unit == TextUnit_Document)
- {
- selectionRange = { 0, textInterface->getTotalNumCharacters() };
- return S_OK;
- }
-
- auto start = getNextEndpointPosition (*textInterface,
- selectionRange.getStart(),
- unit,
- NextEndpointDirection::backwards);
-
- if (start >= 0)
- {
- auto end = getNextEndpointPosition (*textInterface,
- start,
- unit,
- NextEndpointDirection::forwards);
-
- if (end >= 0)
- selectionRange = Range<int> (start, end);
- }
-
- return S_OK;
- }
-
- return UIA_E_NOTSUPPORTED;
- }
-
- JUCE_COMRESULT FindAttribute (TEXTATTRIBUTEID, VARIANT, BOOL, ITextRangeProvider** pRetVal) override
- {
- return withCheckedComArgs (pRetVal, *this, []
- {
- return S_OK;
- });
- }
-
- JUCE_COMRESULT FindText (BSTR text, BOOL backward, BOOL ignoreCase,
- ITextRangeProvider** pRetVal) override
- {
- return owner->withTextInterface (pRetVal, [&] (const AccessibilityTextInterface& textInterface)
- {
- auto selectionText = textInterface.getText (selectionRange);
- String textToSearchFor (text);
-
- auto offset = (backward ? (ignoreCase ? selectionText.lastIndexOfIgnoreCase (textToSearchFor) : selectionText.lastIndexOf (textToSearchFor))
- : (ignoreCase ? selectionText.indexOfIgnoreCase (textToSearchFor) : selectionText.indexOf (textToSearchFor)));
-
- if (offset != -1)
- *pRetVal = new UIATextRangeProvider (*owner, { offset, offset + textToSearchFor.length() });
-
- return S_OK;
- });
- }
-
- JUCE_COMRESULT GetAttributeValue (TEXTATTRIBUTEID attributeId, VARIANT* pRetVal) override
- {
- return owner->withTextInterface (pRetVal, [&] (const AccessibilityTextInterface& textInterface)
- {
- VariantHelpers::clear (pRetVal);
-
- const auto& handler = getHandler();
-
- switch (attributeId)
- {
- case UIA_IsReadOnlyAttributeId:
- {
- const auto readOnly = [&]
- {
- if (auto* valueInterface = handler.getValueInterface())
- return valueInterface->isReadOnly();
-
- return false;
- }();
-
- VariantHelpers::setBool (readOnly, pRetVal);
- break;
- }
- case UIA_CaretPositionAttributeId:
- {
- auto cursorPos = textInterface.getTextInsertionOffset();
-
- auto caretPos = [&]
- {
- if (cursorPos == 0)
- return CaretPosition_BeginningOfLine;
-
- if (cursorPos == textInterface.getTotalNumCharacters())
- return CaretPosition_EndOfLine;
-
- return CaretPosition_Unknown;
- }();
-
- VariantHelpers::setInt (caretPos, pRetVal);
- break;
- }
- default:
- break;
- }
-
- return S_OK;
- });
- }
-
- JUCE_COMRESULT GetBoundingRectangles (SAFEARRAY** pRetVal) override
- {
- return owner->withTextInterface (pRetVal, [&] (const AccessibilityTextInterface& textInterface)
- {
- auto rectangleList = textInterface.getTextBounds (selectionRange);
- auto numRectangles = rectangleList.getNumRectangles();
-
- *pRetVal = SafeArrayCreateVector (VT_R8, 0, 4 * numRectangles);
-
- if (*pRetVal == nullptr)
- return E_FAIL;
-
- if (numRectangles > 0)
- {
- double* doubleArr = nullptr;
-
- if (FAILED (SafeArrayAccessData (*pRetVal, reinterpret_cast<void**> (&doubleArr))))
- {
- SafeArrayDestroy (*pRetVal);
- return E_FAIL;
- }
-
- for (int i = 0; i < numRectangles; ++i)
- {
- auto r = Desktop::getInstance().getDisplays().logicalToPhysical (rectangleList.getRectangle (i));
-
- doubleArr[i * 4] = r.getX();
- doubleArr[i * 4 + 1] = r.getY();
- doubleArr[i * 4 + 2] = r.getWidth();
- doubleArr[i * 4 + 3] = r.getHeight();
- }
-
- if (FAILED (SafeArrayUnaccessData (*pRetVal)))
- {
- SafeArrayDestroy (*pRetVal);
- return E_FAIL;
- }
- }
-
- return S_OK;
- });
- }
-
- JUCE_COMRESULT GetChildren (SAFEARRAY** pRetVal) override
- {
- return withCheckedComArgs (pRetVal, *this, [&]
- {
- *pRetVal = SafeArrayCreateVector (VT_UNKNOWN, 0, 0);
- return S_OK;
- });
- }
-
- JUCE_COMRESULT GetEnclosingElement (IRawElementProviderSimple** pRetVal) override
- {
- return withCheckedComArgs (pRetVal, *this, [&]
- {
- getHandler().getNativeImplementation()->QueryInterface (IID_PPV_ARGS (pRetVal));
- return S_OK;
- });
- }
-
- JUCE_COMRESULT GetText (int maxLength, BSTR* pRetVal) override
- {
- return owner->withTextInterface (pRetVal, [&] (const AccessibilityTextInterface& textInterface)
- {
- auto text = textInterface.getText (selectionRange);
-
- if (maxLength >= 0 && text.length() > maxLength)
- text = text.substring (0, maxLength);
-
- *pRetVal = SysAllocString ((const OLECHAR*) text.toWideCharPointer());
- return S_OK;
- });
- }
-
- JUCE_COMRESULT Move (TextUnit unit, int count, int* pRetVal) override
- {
- return owner->withTextInterface (pRetVal, [&] (const AccessibilityTextInterface&)
- {
- if (count > 0)
- {
- MoveEndpointByUnit (TextPatternRangeEndpoint_End, unit, count, pRetVal);
- MoveEndpointByUnit (TextPatternRangeEndpoint_Start, unit, count, pRetVal);
- }
- else if (count < 0)
- {
- MoveEndpointByUnit (TextPatternRangeEndpoint_Start, unit, count, pRetVal);
- MoveEndpointByUnit (TextPatternRangeEndpoint_End, unit, count, pRetVal);
- }
-
- return S_OK;
- });
- }
-
- JUCE_COMRESULT MoveEndpointByRange (TextPatternRangeEndpoint endpoint,
- ITextRangeProvider* targetRange,
- TextPatternRangeEndpoint targetEndpoint) override
- {
- if (targetRange == nullptr)
- return E_INVALIDARG;
-
- if (! isElementValid())
- return UIA_E_ELEMENTNOTAVAILABLE;
-
- if (auto* textInterface = getHandler().getTextInterface())
- {
- auto otherRange = static_cast<UIATextRangeProvider*> (targetRange)->getSelectionRange();
- auto targetPoint = (targetEndpoint == TextPatternRangeEndpoint_Start ? otherRange.getStart()
- : otherRange.getEnd());
-
- setEndpointChecked (endpoint, targetPoint);
- return S_OK;
- }
-
- return UIA_E_NOTSUPPORTED;
- }
-
- JUCE_COMRESULT MoveEndpointByUnit (TextPatternRangeEndpoint endpoint,
- TextUnit unit,
- int count,
- int* pRetVal) override
- {
- return owner->withTextInterface (pRetVal, [&] (const AccessibilityTextInterface& textInterface)
- {
- auto numCharacters = textInterface.getTotalNumCharacters();
-
- if (count == 0 || numCharacters == 0)
- return S_OK;
-
- auto isStart = (endpoint == TextPatternRangeEndpoint_Start);
- auto endpointToMove = (isStart ? selectionRange.getStart() : selectionRange.getEnd());
-
- if (unit == TextUnit_Character)
- {
- auto targetPoint = jlimit (0, numCharacters, endpointToMove + count);
-
- *pRetVal = targetPoint - endpointToMove;
- setEndpointChecked (endpoint, targetPoint);
-
- return S_OK;
- }
-
- auto direction = (count > 0 ? NextEndpointDirection::forwards
- : NextEndpointDirection::backwards);
-
- if (unit == TextUnit_Paragraph
- || unit == TextUnit_Page
- || unit == TextUnit_Document)
- {
- *pRetVal = (direction == NextEndpointDirection::forwards ? 1 : -1);
- setEndpointChecked (endpoint, numCharacters);
- return S_OK;
- }
-
- for (int i = 0; i < std::abs (count); ++i)
- {
- auto nextEndpoint = getNextEndpointPosition (textInterface,
- endpointToMove,
- unit,
- direction);
-
- if (nextEndpoint < 0)
- {
- *pRetVal = (direction == NextEndpointDirection::forwards ? i : -i);
- setEndpointChecked (endpoint, endpointToMove);
- return S_OK;
- }
-
- endpointToMove = nextEndpoint;
- }
-
- *pRetVal = count;
- setEndpointChecked (endpoint, endpointToMove);
-
- return S_OK;
- });
- }
-
- JUCE_COMRESULT RemoveFromSelection() override
- {
- if (! isElementValid())
- return UIA_E_ELEMENTNOTAVAILABLE;
-
- if (auto* textInterface = getHandler().getTextInterface())
- {
- textInterface->setSelection ({});
- return S_OK;
- }
-
- return UIA_E_NOTSUPPORTED;
- }
-
- JUCE_COMRESULT ScrollIntoView (BOOL) override
- {
- if (! isElementValid())
- return UIA_E_ELEMENTNOTAVAILABLE;
-
- return UIA_E_NOTSUPPORTED;
- }
-
- JUCE_COMRESULT Select() override
- {
- if (! isElementValid())
- return UIA_E_ELEMENTNOTAVAILABLE;
-
- if (auto* textInterface = getHandler().getTextInterface())
- {
- textInterface->setSelection ({});
- textInterface->setSelection (selectionRange);
-
- return S_OK;
- }
-
- return UIA_E_NOTSUPPORTED;
- }
-
- private:
- enum class NextEndpointDirection { forwards, backwards };
-
- static int getNextEndpointPosition (const AccessibilityTextInterface& textInterface,
- int currentPosition,
- TextUnit unit,
- NextEndpointDirection direction)
- {
- auto isTextUnitSeparator = [unit] (const juce_wchar c)
- {
- return ((unit == TextUnit_Word || unit == TextUnit_Format) && CharacterFunctions::isWhitespace (c))
- || (unit == TextUnit_Line && (c == '\r' || c == '\n'));
- };
-
- constexpr int textBufferSize = 1024;
- int numChars = 0;
-
- if (direction == NextEndpointDirection::forwards)
- {
- auto textBuffer = textInterface.getText ({ currentPosition,
- jmin (textInterface.getTotalNumCharacters(), currentPosition + textBufferSize) });
-
- for (auto charPtr = textBuffer.getCharPointer(); ! charPtr.isEmpty();)
- {
- auto character = charPtr.getAndAdvance();
- ++numChars;
-
- if (isTextUnitSeparator (character))
- return currentPosition + numChars;
- }
- }
- else
- {
- auto textBuffer = textInterface.getText ({ jmax (0, currentPosition - textBufferSize),
- currentPosition });
-
- for (auto charPtr = textBuffer.end() - 1; charPtr != textBuffer.begin(); --charPtr)
- {
- auto character = *charPtr;
-
- if (isTextUnitSeparator (character))
- return currentPosition - numChars;
-
- ++numChars;
- }
- }
-
- return -1;
- }
-
- void setEndpointChecked (TextPatternRangeEndpoint endpoint, int newEndpoint)
- {
- if (endpoint == TextPatternRangeEndpoint_Start)
- {
- if (selectionRange.getEnd() < newEndpoint)
- selectionRange.setEnd (newEndpoint);
-
- selectionRange.setStart (newEndpoint);
- }
- else
- {
- if (selectionRange.getStart() > newEndpoint)
- selectionRange.setStart (newEndpoint);
-
- selectionRange.setEnd (newEndpoint);
- }
- }
-
- ComSmartPtr<UIATextProvider> owner;
- Range<int> selectionRange;
-
- //==============================================================================
- JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (UIATextRangeProvider)
- };
-
- //==============================================================================
- JUCE_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR (UIATextProvider)
- };
-
- } // namespace juce
|