| /* |
| * Copyright (C) 2014-2020 Apple Inc. All rights reserved. |
| * |
| * Redistribution and use in source and binary forms, with or without |
| * modification, are permitted provided that the following conditions |
| * are met: |
| * 1. Redistributions of source code must retain the above copyright |
| * notice, this list of conditions and the following disclaimer. |
| * 2. Redistributions in binary form must reproduce the above copyright |
| * notice, this list of conditions and the following disclaimer in the |
| * documentation and/or other materials provided with the distribution. |
| * |
| * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' |
| * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, |
| * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR |
| * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS |
| * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR |
| * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF |
| * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS |
| * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN |
| * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) |
| * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF |
| * THE POSSIBILITY OF SUCH DAMAGE. |
| */ |
| |
| #import "config.h" |
| #import "DictionaryLookup.h" |
| |
| #import "SimpleRange.h" |
| |
| #if PLATFORM(COCOA) |
| |
| #if ENABLE(REVEAL) |
| |
| #import "Document.h" |
| #import "Editing.h" |
| #import "FocusController.h" |
| #import "Frame.h" |
| #import "FrameSelection.h" |
| #import "GraphicsContextCG.h" |
| #import "HitTestResult.h" |
| #import "NotImplemented.h" |
| #import "Page.h" |
| #import "Range.h" |
| #import "RenderObject.h" |
| #import "TextIterator.h" |
| #import "VisiblePosition.h" |
| #import "VisibleSelection.h" |
| #import "VisibleUnits.h" |
| #import <pal/cocoa/RevealSoftLink.h> |
| #import <pal/spi/cg/CoreGraphicsSPI.h> |
| #import <wtf/BlockObjCExceptions.h> |
| #import <wtf/RefPtr.h> |
| |
| #if PLATFORM(MAC) |
| #import <Quartz/Quartz.h> |
| #else |
| #import <PDFKit/PDFKit.h> |
| #import <pal/spi/ios/UIKitSPI.h> |
| |
| #import <pal/ios/UIKitSoftLink.h> |
| #endif |
| |
| #if PLATFORM(MACCATALYST) |
| #import <UIKitMacHelper/UINSRevealController.h> |
| SOFT_LINK_PRIVATE_FRAMEWORK(UIKitMacHelper) |
| SOFT_LINK(UIKitMacHelper, UINSSharedRevealController, id<UINSRevealController>, (void), ()) |
| #endif // PLATFORM(MACCATALYST) |
| |
| #if PLATFORM(MAC) |
| |
| @interface WebRevealHighlight : NSObject<RVPresenterHighlightDelegate> |
| |
| @property (nonatomic, readonly) NSRect highlightRect; |
| @property (nonatomic, readonly) BOOL useDefaultHighlight; |
| @property (nonatomic, readonly) RetainPtr<NSAttributedString> attributedString; |
| |
| - (instancetype)initWithHighlightRect:(NSRect)highlightRect useDefaultHighlight:(BOOL)useDefaultHighlight attributedString:(NSAttributedString *) attributedString; |
| - (void)setClearTextIndicator:(Function<void()>&&)clearTextIndicator; |
| |
| @end |
| |
| @implementation WebRevealHighlight { |
| Function<void()> _clearTextIndicator; |
| } |
| |
| - (instancetype)initWithHighlightRect:(NSRect)highlightRect useDefaultHighlight:(BOOL)useDefaultHighlight attributedString:(NSAttributedString *) attributedString |
| { |
| if (!(self = [super init])) |
| return nil; |
| |
| _highlightRect = highlightRect; |
| _useDefaultHighlight = useDefaultHighlight; |
| _attributedString = adoptNS([attributedString copy]); |
| |
| return self; |
| } |
| |
| - (void)setClearTextIndicator:(Function<void()>&&)clearTextIndicator |
| { |
| _clearTextIndicator = WTFMove(clearTextIndicator); |
| } |
| |
| - (NSArray<NSValue *> *)revealContext:(RVPresentingContext *)context rectsForItem:(RVItem *)item |
| { |
| UNUSED_PARAM(context); |
| UNUSED_PARAM(item); |
| return @[[NSValue valueWithRect:self.highlightRect]]; |
| } |
| |
| - (void)revealContext:(RVPresentingContext *)context drawRectsForItem:(RVItem *)item |
| { |
| UNUSED_PARAM(item); |
| |
| for (NSValue *rectVal in context.itemRectsInView) { |
| NSRect rect = rectVal.rectValue; |
| |
| // Get current font attributes from the attributed string above, and add paragraph style attribute in order to center text. |
| auto attributes = adoptNS([[NSMutableDictionary alloc] initWithDictionary:[self.attributedString fontAttributesInRange:NSMakeRange(0, [self.attributedString length])]]); |
| auto paragraph = adoptNS([[NSMutableParagraphStyle alloc] init]); |
| [paragraph setAlignment:NSTextAlignmentCenter]; |
| [attributes setObject:paragraph.get() forKey:NSParagraphStyleAttributeName]; |
| |
| auto string = adoptNS([[NSAttributedString alloc] initWithString:[self.attributedString string] attributes:attributes.get()]); |
| [string drawInRect:rect]; |
| } |
| } |
| |
| - (BOOL)revealContext:(RVPresentingContext *)context shouldUseDefaultHighlightForItem:(RVItem *)item |
| { |
| UNUSED_PARAM(context); |
| UNUSED_PARAM(item); |
| return self.useDefaultHighlight; |
| } |
| |
| - (void)revealContext:(RVPresentingContext *)context stopHighlightingItem:(RVItem *)item |
| { |
| UNUSED_PARAM(context); |
| UNUSED_PARAM(item); |
| auto block = WTFMove(_clearTextIndicator); |
| if (block) |
| block(); |
| } |
| |
| @end |
| |
| #elif PLATFORM(MACCATALYST) // PLATFORM(MAC) |
| |
| @interface WebRevealHighlight : NSObject<UIRVPresenterHighlightDelegate> |
| |
| - (instancetype)initWithHighlightRect:(NSRect)highlightRect view:(UIView *)view image:(RefPtr<WebCore::Image>&&)image; |
| |
| @end |
| |
| @implementation WebRevealHighlight { |
| RefPtr<WebCore::Image> _image; |
| CGRect _highlightRect; |
| BOOL _highlighting; |
| UIView *_view; |
| } |
| |
| - (instancetype)initWithHighlightRect:(NSRect)highlightRect view:(UIView *)view image:(RefPtr<WebCore::Image>&&)image |
| { |
| if (!(self = [super init])) |
| return nil; |
| |
| _highlightRect = highlightRect; |
| _view = view; |
| _highlighting = NO; |
| _image = image; |
| |
| return self; |
| } |
| |
| - (void)setImage:(RefPtr<WebCore::Image>&&)image |
| { |
| _image = WTFMove(image); |
| } |
| |
| - (NSArray<NSValue *> *)highlightRectsForItem:(RVItem *)item |
| { |
| UNUSED_PARAM(item); |
| return @[[NSValue valueWithCGRect:_highlightRect]]; |
| } |
| |
| - (void)startHighlightingItem:(RVItem *)item |
| { |
| UNUSED_PARAM(item); |
| _highlighting = YES; |
| } |
| |
| - (void)highlightItem:(RVItem *)item withProgress:(CGFloat)progress |
| { |
| UNUSED_PARAM(item); |
| UNUSED_PARAM(progress); |
| } |
| |
| - (void)completeHighlightingItem:(RVItem *)item |
| { |
| UNUSED_PARAM(item); |
| } |
| |
| - (void)stopHighlightingItem:(RVItem *)item |
| { |
| UNUSED_PARAM(item); |
| _highlighting = NO; |
| } |
| |
| - (void)highlightRangeChangedForItem:(RVItem *)item |
| { |
| UNUSED_PARAM(item); |
| } |
| |
| - (BOOL)highlighting |
| { |
| return _highlighting; |
| } |
| |
| - (void)drawHighlightContentForItem:(RVItem *)item context:(CGContextRef)context |
| { |
| NSArray <NSValue *> *rects = [self highlightRectsForItem:item]; |
| if (!rects.count) |
| return; |
| |
| CGRect highlightRect = rects.firstObject.CGRectValue; |
| for (NSValue *rect in rects) |
| highlightRect = CGRectUnion(highlightRect, rect.CGRectValue); |
| highlightRect = [_view convertRect:highlightRect fromView:nil]; |
| |
| WebCore::CGContextStateSaver saveState(context); |
| CGAffineTransform contextTransform = CGContextGetCTM(context); |
| CGFloat backingScale = contextTransform.a; |
| CGFloat macCatalystScaleFactor = [PAL::getUIApplicationClass() sharedApplication]._iOSMacScale; |
| CGAffineTransform transform = CGAffineTransformMakeScale(macCatalystScaleFactor * backingScale, macCatalystScaleFactor * backingScale); |
| CGContextSetCTM(context, transform); |
| |
| for (NSValue *v in rects) { |
| CGRect imgSrcRect = [_view convertRect:v.CGRectValue fromView:nil]; |
| auto nativeImage = _image->nativeImage(); |
| CGRect origin = CGRectMake(imgSrcRect.origin.x - highlightRect.origin.x, imgSrcRect.origin.y - highlightRect.origin.y, highlightRect.size.width, highlightRect.size.height); |
| CGContextDrawImage(context, origin, nativeImage->platformImage().get()); |
| } |
| } |
| |
| @end |
| |
| #endif // PLATFORM(MACCATALYST) |
| |
| #endif // ENABLE(REVEAL) |
| |
| namespace WebCore { |
| |
| #if ENABLE(REVEAL) |
| |
| static bool canCreateRevealItems() |
| { |
| static bool result; |
| static std::once_flag onceFlag; |
| std::call_once(onceFlag, [&] { |
| result = PAL::isRevealCoreFrameworkAvailable() && PAL::getRVItemClass(); |
| }); |
| return result; |
| } |
| |
| std::optional<std::tuple<SimpleRange, NSDictionary *>> DictionaryLookup::rangeForSelection(const VisibleSelection& selection) |
| { |
| BEGIN_BLOCK_OBJC_EXCEPTIONS |
| |
| if (!canCreateRevealItems()) |
| return std::nullopt; |
| |
| // Since we already have the range we want, we just need to grab the returned options. |
| auto selectionStart = selection.visibleStart(); |
| auto selectionEnd = selection.visibleEnd(); |
| |
| // As context, we are going to use the surrounding paragraphs of text. |
| auto paragraphStart = startOfParagraph(selectionStart); |
| auto paragraphEnd = endOfParagraph(selectionEnd); |
| if (paragraphStart.isNull() || paragraphEnd.isNull()) |
| return std::nullopt; |
| |
| auto lengthToSelectionStart = characterCount(*makeSimpleRange(paragraphStart, selectionStart)); |
| auto selectionCharacterCount = characterCount(*makeSimpleRange(selectionStart, selectionEnd)); |
| NSRange rangeToPass = NSMakeRange(lengthToSelectionStart, selectionCharacterCount); |
| |
| auto fullCharacterRange = *makeSimpleRange(paragraphStart, paragraphEnd); |
| String itemString = plainText(fullCharacterRange); |
| NSRange highlightRange = adoptNS([PAL::allocRVItemInstance() initWithText:itemString selectedRange:rangeToPass]).get().highlightRange; |
| |
| return { { resolveCharacterRange(fullCharacterRange, highlightRange), nil } }; |
| |
| END_BLOCK_OBJC_EXCEPTIONS |
| |
| return std::nullopt; |
| } |
| |
| std::optional<std::tuple<SimpleRange, NSDictionary *>> DictionaryLookup::rangeAtHitTestResult(const HitTestResult& hitTestResult) |
| { |
| BEGIN_BLOCK_OBJC_EXCEPTIONS |
| |
| if (!canCreateRevealItems()) |
| return std::nullopt; |
| |
| auto* node = hitTestResult.innerNonSharedNode(); |
| if (!node || !node->renderer()) |
| return std::nullopt; |
| |
| auto* frame = node->document().frame(); |
| if (!frame) |
| return std::nullopt; |
| |
| // Don't do anything if there is no character at the point. |
| auto framePoint = hitTestResult.roundedPointInInnerNodeFrame(); |
| if (!frame->rangeForPoint(framePoint)) |
| return std::nullopt; |
| |
| auto position = frame->visiblePositionForPoint(framePoint); |
| if (position.isNull()) |
| position = firstPositionInOrBeforeNode(node); |
| |
| auto selection = CheckedRef(frame->page()->focusController())->focusedOrMainFrame().selection().selection(); |
| NSRange selectionRange; |
| NSUInteger hitIndex; |
| std::optional<SimpleRange> fullCharacterRange; |
| |
| if (selection.isRange()) { |
| auto selectionStart = selection.visibleStart(); |
| auto selectionEnd = selection.visibleEnd(); |
| |
| // As context, we are going to use the surrounding paragraphs of text. |
| fullCharacterRange = makeSimpleRange(startOfParagraph(selectionStart), endOfParagraph(selectionEnd)); |
| if (!fullCharacterRange) |
| return std::nullopt; |
| |
| selectionRange = NSMakeRange(characterCount(*makeSimpleRange(fullCharacterRange->start, selectionStart)), |
| characterCount(*makeSimpleRange(selectionStart, selectionEnd))); |
| hitIndex = characterCount(*makeSimpleRange(fullCharacterRange->start, position)); |
| } else { |
| VisibleSelection selectionAccountingForLineRules { position }; |
| selectionAccountingForLineRules.expandUsingGranularity(TextGranularity::WordGranularity); |
| position = selectionAccountingForLineRules.start(); |
| |
| // As context, we are going to use 250 characters of text before and after the point. |
| fullCharacterRange = rangeExpandedAroundPositionByCharacters(position, 250); |
| if (!fullCharacterRange) |
| return std::nullopt; |
| |
| selectionRange = NSMakeRange(NSNotFound, 0); |
| hitIndex = characterCount(*makeSimpleRange(fullCharacterRange->start, position)); |
| } |
| |
| NSRange selectedRange = [PAL::getRVSelectionClass() revealRangeAtIndex:hitIndex selectedRanges:@[[NSValue valueWithRange:selectionRange]] shouldUpdateSelection:nil]; |
| |
| String itemString = plainText(*fullCharacterRange); |
| auto highlightRange = adoptNS([PAL::allocRVItemInstance() initWithText:itemString selectedRange:selectedRange]).get().highlightRange; |
| |
| if (highlightRange.location == NSNotFound || !highlightRange.length) |
| return std::nullopt; |
| |
| return { { resolveCharacterRange(*fullCharacterRange, highlightRange), nil } }; |
| |
| END_BLOCK_OBJC_EXCEPTIONS |
| |
| return std::nullopt; |
| |
| } |
| |
| static void expandSelectionByCharacters(PDFSelection *selection, NSInteger numberOfCharactersToExpand, NSInteger& charactersAddedBeforeStart, NSInteger& charactersAddedAfterEnd) |
| { |
| BEGIN_BLOCK_OBJC_EXCEPTIONS |
| |
| size_t originalLength = selection.string.length; |
| [selection extendSelectionAtStart:numberOfCharactersToExpand]; |
| |
| charactersAddedBeforeStart = selection.string.length - originalLength; |
| |
| [selection extendSelectionAtEnd:numberOfCharactersToExpand]; |
| charactersAddedAfterEnd = selection.string.length - originalLength - charactersAddedBeforeStart; |
| |
| END_BLOCK_OBJC_EXCEPTIONS |
| } |
| |
| std::tuple<NSString *, NSDictionary *> DictionaryLookup::stringForPDFSelection(PDFSelection *selection) |
| { |
| BEGIN_BLOCK_OBJC_EXCEPTIONS |
| |
| if (!canCreateRevealItems()) |
| return { nullptr, nil }; |
| |
| // Don't do anything if there is no character at the point. |
| if (!selection || !selection.string.length) |
| return { @"", nil }; |
| |
| RetainPtr<PDFSelection> selectionForLookup = adoptNS([selection copy]); |
| |
| // As context, we are going to use 250 characters of text before and after the point. |
| auto originalLength = [selectionForLookup string].length; |
| NSInteger charactersAddedBeforeStart = 0; |
| NSInteger charactersAddedAfterEnd = 0; |
| expandSelectionByCharacters(selectionForLookup.get(), 250, charactersAddedBeforeStart, charactersAddedAfterEnd); |
| |
| auto fullPlainTextString = [selectionForLookup string]; |
| auto rangeToPass = NSMakeRange(charactersAddedBeforeStart, 0); |
| |
| auto item = adoptNS([PAL::allocRVItemInstance() initWithText:fullPlainTextString selectedRange:rangeToPass]); |
| NSRange extractedRange = item.get().highlightRange; |
| |
| if (extractedRange.location == NSNotFound) |
| return { selection.string, nil }; |
| |
| NSInteger lookupAddedBefore = rangeToPass.location - extractedRange.location; |
| NSInteger lookupAddedAfter = (extractedRange.location + extractedRange.length) - (rangeToPass.location + originalLength); |
| |
| [selection extendSelectionAtStart:lookupAddedBefore]; |
| [selection extendSelectionAtEnd:lookupAddedAfter]; |
| |
| ASSERT([selection.string isEqualToString:[fullPlainTextString substringWithRange:extractedRange]]); |
| return { selection.string, nil }; |
| |
| END_BLOCK_OBJC_EXCEPTIONS |
| |
| return { @"", nil }; |
| } |
| |
| static WKRevealController showPopupOrCreateAnimationController(bool createAnimationController, const DictionaryPopupInfo& dictionaryPopupInfo, CocoaView *view, const WTF::Function<void(TextIndicator&)>& textIndicatorInstallationCallback, const WTF::Function<FloatRect(FloatRect)>& rootViewToViewConversionCallback, WTF::Function<void()>&& clearTextIndicator) |
| { |
| BEGIN_BLOCK_OBJC_EXCEPTIONS |
| |
| #if PLATFORM(MAC) |
| if (!PAL::isRevealFrameworkAvailable() || !canCreateRevealItems() || !PAL::getRVPresenterClass()) |
| return nil; |
| |
| auto mutableOptions = adoptNS([[NSMutableDictionary alloc] init]); |
| if (NSDictionary *options = dictionaryPopupInfo.options.get()) |
| [mutableOptions addEntriesFromDictionary:options]; |
| |
| auto textIndicator = TextIndicator::create(dictionaryPopupInfo.textIndicator); |
| |
| auto presenter = adoptNS([PAL::allocRVPresenterInstance() init]); |
| |
| NSRect highlightRect; |
| NSPoint pointerLocation; |
| |
| if (textIndicator.get().contentImage()) { |
| textIndicatorInstallationCallback(textIndicator.get()); |
| |
| FloatRect firstTextRectInViewCoordinates = textIndicator.get().textRectsInBoundingRectCoordinates()[0]; |
| FloatRect textBoundingRectInViewCoordinates = textIndicator.get().textBoundingRectInRootViewCoordinates(); |
| FloatRect selectionBoundingRectInViewCoordinates = textIndicator.get().selectionRectInRootViewCoordinates(); |
| |
| if (rootViewToViewConversionCallback) { |
| textBoundingRectInViewCoordinates = rootViewToViewConversionCallback(textBoundingRectInViewCoordinates); |
| selectionBoundingRectInViewCoordinates = rootViewToViewConversionCallback(selectionBoundingRectInViewCoordinates); |
| } |
| |
| firstTextRectInViewCoordinates.moveBy(textBoundingRectInViewCoordinates.location()); |
| highlightRect = selectionBoundingRectInViewCoordinates; |
| pointerLocation = firstTextRectInViewCoordinates.location(); |
| } else { |
| NSPoint textBaselineOrigin = dictionaryPopupInfo.origin; |
| highlightRect = textIndicator->selectionRectInRootViewCoordinates(); |
| pointerLocation = [view convertPoint:textBaselineOrigin toView:nil]; |
| } |
| |
| auto webHighlight = adoptNS([[WebRevealHighlight alloc] initWithHighlightRect: highlightRect useDefaultHighlight:!textIndicator.get().contentImage() attributedString:dictionaryPopupInfo.attributedString.get()]); |
| auto context = adoptNS([PAL::allocRVPresentingContextInstance() initWithPointerLocationInView:pointerLocation inView:view highlightDelegate:webHighlight.get()]); |
| auto item = adoptNS([PAL::allocRVItemInstance() initWithText:dictionaryPopupInfo.attributedString.get().string selectedRange:NSMakeRange(0, dictionaryPopupInfo.attributedString.get().string.length)]); |
| |
| [webHighlight setClearTextIndicator:[webHighlight = WTFMove(webHighlight), clearTextIndicator = WTFMove(clearTextIndicator)] { |
| if (clearTextIndicator) |
| clearTextIndicator(); |
| }]; |
| |
| if (createAnimationController) |
| return [presenter animationControllerForItem:item.get() documentContext:nil presentingContext:context.get() options:nil]; |
| |
| [presenter revealItem:item.get() documentContext:nil presentingContext:context.get() options:@{ @"forceLookup": @YES }]; |
| return nil; |
| #elif PLATFORM(MACCATALYST) |
| UNUSED_PARAM(textIndicatorInstallationCallback); |
| UNUSED_PARAM(rootViewToViewConversionCallback); |
| UNUSED_PARAM(clearTextIndicator); |
| ASSERT_UNUSED(createAnimationController, !createAnimationController); |
| auto textIndicator = TextIndicator::create(dictionaryPopupInfo.textIndicator); |
| auto webHighlight = adoptNS([[WebRevealHighlight alloc] initWithHighlightRect:[view convertRect:textIndicator->selectionRectInRootViewCoordinates() toView:nil] view:view image:textIndicator->contentImage()]); |
| auto item = adoptNS([PAL::allocRVItemInstance() initWithText:dictionaryPopupInfo.attributedString.get().string selectedRange:NSMakeRange(0, dictionaryPopupInfo.attributedString.get().string.length)]); |
| |
| [UINSSharedRevealController() revealItem:item.get() locationInWindow:dictionaryPopupInfo.origin window:view.window highlighter:webHighlight.get()]; |
| return nil; |
| #else // PLATFORM(IOS_FAMILY) |
| UNUSED_PARAM(createAnimationController); |
| UNUSED_PARAM(dictionaryPopupInfo); |
| UNUSED_PARAM(view); |
| UNUSED_PARAM(textIndicatorInstallationCallback); |
| UNUSED_PARAM(rootViewToViewConversionCallback); |
| UNUSED_PARAM(clearTextIndicator); |
| return nil; |
| #endif // PLATFORM(IOS_FAMILY) |
| |
| END_BLOCK_OBJC_EXCEPTIONS |
| } |
| |
| void DictionaryLookup::showPopup(const DictionaryPopupInfo& dictionaryPopupInfo, CocoaView *view, const WTF::Function<void(TextIndicator&)>& textIndicatorInstallationCallback, const WTF::Function<FloatRect(FloatRect)>& rootViewToViewConversionCallback, WTF::Function<void()>&& clearTextIndicator) |
| { |
| showPopupOrCreateAnimationController(false, dictionaryPopupInfo, view, textIndicatorInstallationCallback, rootViewToViewConversionCallback, WTFMove(clearTextIndicator)); |
| } |
| |
| void DictionaryLookup::hidePopup() |
| { |
| notImplemented(); |
| } |
| |
| #if PLATFORM(MAC) |
| |
| WKRevealController DictionaryLookup::animationControllerForPopup(const DictionaryPopupInfo& dictionaryPopupInfo, NSView *view, const WTF::Function<void(TextIndicator&)>& textIndicatorInstallationCallback, const WTF::Function<FloatRect(FloatRect)>& rootViewToViewConversionCallback, WTF::Function<void()>&& clearTextIndicator) |
| { |
| return showPopupOrCreateAnimationController(true, dictionaryPopupInfo, view, textIndicatorInstallationCallback, rootViewToViewConversionCallback, WTFMove(clearTextIndicator)); |
| } |
| |
| #endif // PLATFORM(MAC) |
| |
| #elif PLATFORM(IOS_FAMILY) // PLATFORM(IOS_FAMILY) && !ENABLE(REVEAL) |
| |
| std::optional<std::tuple<SimpleRange, NSDictionary *>> DictionaryLookup::rangeForSelection(const VisibleSelection&) |
| { |
| return std::nullopt; |
| } |
| |
| std::optional<std::tuple<SimpleRange, NSDictionary *>> DictionaryLookup::rangeAtHitTestResult(const HitTestResult&) |
| { |
| return std::nullopt; |
| } |
| |
| #endif |
| |
| } // namespace WebCore |
| |
| #endif // PLATFORM(COCOA) |