blob: 5342d902d1abc9e61efb3004ad44097deb5569e3 [file] [log] [blame]
/*
* Copyright (C) 2019 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"
#if USE(UICONTEXTMENU)
#import "TestContextMenuDriver.h"
#import "TestWKWebView.h"
#import "TestWKWebViewController.h"
#import "Utilities.h"
#import <WebKit/WKWebViewConfigurationPrivate.h>
#import <WebKit/WebKit.h>
static bool contextMenuRequested;
static bool contextMenuSPIRequested;
static bool willPresentCalled;
static bool willCommitCalled;
static bool previewingViewControllerCalled;
static bool previewActionItemsCalled;
static bool didEndCalled;
static bool didAskForHintPreviewContainer;
static bool alternateURLRequested;
static RetainPtr<NSURL> linkURL;
static RetainPtr<TestContextMenuDriver> contextMenuWebViewDriver(Class delegateClass, NSString *customHTMLString = nil)
{
static auto window = adoptNS([[UIWindow alloc] initWithFrame:NSMakeRect(0, 0, 800, 600)]);
static auto driver = adoptNS([TestContextMenuDriver new]);
static auto uiDelegate = adoptNS((NSObject<WKUIDelegate> *)[delegateClass new]);
static auto configuration = adoptNS([WKWebViewConfiguration new]);
[configuration _setClickInteractionDriverForTesting:driver.get()];
static auto webViewController = adoptNS([[TestWKWebViewController alloc] initWithFrame:CGRectMake(0, 0, 200, 200) configuration:configuration.get()]);
TestWKWebView *webView = [webViewController webView];
[window addSubview:webView];
[webView setUIDelegate:uiDelegate.get()];
if (!customHTMLString) {
linkURL = [NSURL URLWithString:@"http://127.0.0.1/"];
[webView synchronouslyLoadHTMLString:[NSString stringWithFormat:@"<a href='%@'>This is a link</a>", linkURL.get()]];
} else
[webView synchronouslyLoadHTMLString:customHTMLString];
return driver;
}
@interface TestContextMenuUIDelegate : NSObject <WKUIDelegate>
@end
@implementation TestContextMenuUIDelegate
- (void)webView:(WKWebView *)webView contextMenuConfigurationForElement:(WKContextMenuElementInfo *)elementInfo completionHandler:(void(^)(UIContextMenuConfiguration *))completionHandler
{
EXPECT_TRUE([elementInfo.linkURL.absoluteString isEqualToString:[linkURL absoluteString]]);
contextMenuRequested = true;
UIContextMenuContentPreviewProvider previewProvider = ^UIViewController * ()
{
return adoptNS([[UIViewController alloc] init]).autorelease();
};
UIContextMenuActionProvider actionProvider = ^UIMenu *(NSArray<UIMenuElement *> *suggestedActions)
{
return [UIMenu menuWithTitle:@"" children:suggestedActions];
};
completionHandler([UIContextMenuConfiguration configurationWithIdentifier:nil previewProvider:previewProvider actionProvider:actionProvider]);
}
- (void)webView:(WKWebView *)webView contextMenuWillPresentForElement:(WKContextMenuElementInfo *)elementInfo
{
willPresentCalled = true;
}
- (void)webView:(WKWebView *)webView contextMenuForElement:(WKContextMenuElementInfo *)elementInfo willCommitWithAnimator:(id<UIContextMenuInteractionCommitAnimating>)animator
{
willCommitCalled = true;
}
- (void)webView:(WKWebView *)webView contextMenuDidEndForElement:(WKContextMenuElementInfo *)elementInfo
{
didEndCalled = true;
}
@end
TEST(ContextMenu, Click)
{
auto driver = contextMenuWebViewDriver([TestContextMenuUIDelegate class]);
[driver begin:^(BOOL result) {
EXPECT_FALSE(result);
[driver clickDown];
[driver clickUp];
}];
TestWebKitAPI::Util::run(&willPresentCalled);
EXPECT_TRUE(contextMenuRequested);
EXPECT_TRUE(willPresentCalled);
EXPECT_FALSE(willCommitCalled);
EXPECT_FALSE(didEndCalled);
}
TEST(ContextMenu, End)
{
auto driver = contextMenuWebViewDriver([TestContextMenuUIDelegate class]);
[driver begin:^(BOOL result) {
EXPECT_FALSE(result);
[driver end];
}];
TestWebKitAPI::Util::run(&didEndCalled);
EXPECT_TRUE(contextMenuRequested);
EXPECT_FALSE(willPresentCalled);
EXPECT_FALSE(willCommitCalled);
EXPECT_TRUE(didEndCalled);
}
@interface TestContextMenuAPIBeforeSPIUIDelegate : NSObject <WKUIDelegate>
@end
@implementation TestContextMenuAPIBeforeSPIUIDelegate
- (void)webView:(WKWebView *)webView contextMenuConfigurationForElement:(WKContextMenuElementInfo *)elementInfo completionHandler:(void(^)(UIContextMenuConfiguration *))completionHandler
{
contextMenuRequested = true;
UIContextMenuContentPreviewProvider previewProvider = ^UIViewController * ()
{
return adoptNS([[UIViewController alloc] init]).autorelease();
};
UIContextMenuActionProvider actionProvider = ^UIMenu *(NSArray<UIMenuElement *> *suggestedActions)
{
return [UIMenu menuWithTitle:@"" children:suggestedActions];
};
completionHandler([UIContextMenuConfiguration configurationWithIdentifier:nil previewProvider:previewProvider actionProvider:actionProvider]);
}
- (void)_webView:(WKWebView *)webView contextMenuConfigurationForElement:(WKContextMenuElementInfo *)elementInfo completionHandler:(void(^)(UIContextMenuConfiguration *))completionHandler
{
contextMenuSPIRequested = true;
completionHandler(nil);
}
- (void)webView:(WKWebView *)webView contextMenuWillPresentForElement:(WKContextMenuElementInfo *)elementInfo
{
willPresentCalled = true;
}
@end
TEST(ContextMenu, APIBeforeSPI)
{
auto driver = contextMenuWebViewDriver([TestContextMenuAPIBeforeSPIUIDelegate class]);
[driver begin:^(BOOL result) {
EXPECT_FALSE(result);
[driver clickDown];
[driver clickUp];
}];
TestWebKitAPI::Util::run(&willPresentCalled);
EXPECT_TRUE(contextMenuRequested);
EXPECT_FALSE(contextMenuSPIRequested);
}
@interface TestContextMenuImageUIDelegate : NSObject <WKUIDelegate>
@end
@implementation TestContextMenuImageUIDelegate
- (void)_webView:(WKWebView *)webView contextMenuConfigurationForElement:(WKContextMenuElementInfo *)elementInfo completionHandler:(void(^)(UIContextMenuConfiguration *))completionHandler
{
contextMenuRequested = true;
completionHandler(nil);
}
- (NSURL *)_webView:(WKWebView *)webView alternateURLFromImage:(UIImage *)image userInfo:(NSDictionary **)userInfo
{
alternateURLRequested = true;
return linkURL.get();
}
- (void)webView:(WKWebView *)webView contextMenuWillPresentForElement:(WKContextMenuElementInfo *)elementInfo
{
willPresentCalled = true;
EXPECT_TRUE([elementInfo.linkURL.absoluteString isEqualToString:[linkURL absoluteString]]);
}
- (void)webView:(WKWebView *)webView contextMenuForElement:(WKContextMenuElementInfo *)elementInfo willCommitWithAnimator:(id<UIContextMenuInteractionCommitAnimating>)animator
{
willCommitCalled = true;
}
- (void)webView:(WKWebView *)webView contextMenuDidEndForElement:(WKContextMenuElementInfo *)elementInfo
{
didEndCalled = true;
}
@end
TEST(ContextMenu, Image)
{
linkURL = [NSURL URLWithString:@"http://127.0.0.1/image"];
auto driver = contextMenuWebViewDriver([TestContextMenuImageUIDelegate class], @"<img style='width:400px;height:400px' src=''>");
[driver begin:^(BOOL result) {
EXPECT_FALSE(result);
[driver clickDown];
[driver clickUp];
}];
TestWebKitAPI::Util::run(&willPresentCalled);
EXPECT_TRUE(contextMenuRequested);
EXPECT_TRUE(alternateURLRequested);
EXPECT_TRUE(willPresentCalled);
EXPECT_FALSE(willCommitCalled);
EXPECT_FALSE(didEndCalled);
}
#pragma clang diagnostic push
#pragma clang diagnostic ignored "-Wdeprecated-declarations"
#pragma clang diagnostic ignored "-Wdeprecated-implementations"
@interface LegacyPreviewViewController : UIViewController
@end
@implementation LegacyPreviewViewController
- (NSArray<UIPreviewAction *> *)previewActionItems
{
previewActionItemsCalled = true;
return @[
[UIPreviewAction actionWithTitle:@"Action" style:UIPreviewActionStyleDefault handler:^(UIPreviewAction *, UIViewController *) { }]
];
}
@end
@interface LegacyContextMenuUIDelegate : NSObject <WKUIDelegate>
@end
@implementation LegacyContextMenuUIDelegate
- (BOOL)webView:(WKWebView *)webView shouldPreviewElement:(WKPreviewElementInfo *)elementInfo
{
EXPECT_TRUE([elementInfo.linkURL.absoluteString isEqualToString:[linkURL absoluteString]]);
contextMenuRequested = true;
return YES;
}
- (UIViewController *)webView:(WKWebView *)webView previewingViewControllerForElement:(WKPreviewElementInfo *)elementInfo defaultActions:(NSArray<id <WKPreviewActionItem>> *)previewActions
{
EXPECT_TRUE(previewActions.count);
previewingViewControllerCalled = true;
return [LegacyPreviewViewController new];
}
/* Even though this is non-legacy API, it should not be enough to trigger the non-legacy flow. */
- (void)webView:(WKWebView *)webView contextMenuWillPresentForElement:(WKContextMenuElementInfo *)elementInfo
{
willPresentCalled = true;
}
/* Even though this is non-legacy API, it should not be enough to trigger the non-legacy flow. */
- (void)_webView:(WKWebView *)webView contextMenuDidEndForElement:(WKContextMenuElementInfo *)elementInfo
{
}
@end
TEST(ContextMenu, Legacy)
{
auto driver = contextMenuWebViewDriver([LegacyContextMenuUIDelegate class]);
[driver begin:^(BOOL result) {
EXPECT_FALSE(result);
[driver clickDown];
[driver clickUp];
}];
TestWebKitAPI::Util::run(&previewActionItemsCalled);
EXPECT_TRUE(contextMenuRequested);
EXPECT_TRUE(previewingViewControllerCalled);
EXPECT_TRUE(willPresentCalled);
}
#pragma clang diagnostic pop
@interface TestContextMenuSuggestedActionsUIDelegate : NSObject <WKUIDelegate>
@end
@implementation TestContextMenuSuggestedActionsUIDelegate
- (void)webView:(WKWebView *)webView contextMenuConfigurationForElement:(WKContextMenuElementInfo *)elementInfo completionHandler:(void(^)(UIContextMenuConfiguration *))completionHandler
{
EXPECT_TRUE([elementInfo.linkURL.absoluteString isEqualToString:[linkURL absoluteString]]);
contextMenuRequested = true;
UIContextMenuContentPreviewProvider previewProvider = ^UIViewController * ()
{
return adoptNS([[UIViewController alloc] init]).autorelease();
};
UIContextMenuActionProvider actionProvider = ^UIMenu *(NSArray<UIMenuElement *> *suggestedActions)
{
NSArray<NSString *> *expectedIdentifiers = @[
@"WKElementActionTypeOpen",
@"WKElementActionTypeAddToReadingList",
@"WKElementActionTypeCopy",
@"WKElementActionTypeShare"
];
EXPECT_TRUE(expectedIdentifiers.count == suggestedActions.count);
[suggestedActions enumerateObjectsUsingBlock:^(UIMenuElement *menuElement, NSUInteger index, BOOL *) {
EXPECT_TRUE([menuElement isKindOfClass:[UIAction class]]);
EXPECT_TRUE([[(UIAction *)menuElement identifier] isEqualToString:expectedIdentifiers[index]]);
}];
return [UIMenu menuWithTitle:@"" children:suggestedActions];
};
completionHandler([UIContextMenuConfiguration configurationWithIdentifier:nil previewProvider:previewProvider actionProvider:actionProvider]);
}
- (void)webView:(WKWebView *)webView contextMenuWillPresentForElement:(WKContextMenuElementInfo *)elementInfo
{
willPresentCalled = true;
}
@end
TEST(ContextMenu, SuggestedActions)
{
auto driver = contextMenuWebViewDriver([TestContextMenuSuggestedActionsUIDelegate class]);
[driver begin:^(BOOL result) {
EXPECT_FALSE(result);
[driver clickDown];
[driver clickUp];
}];
TestWebKitAPI::Util::run(&willPresentCalled);
EXPECT_TRUE(contextMenuRequested);
EXPECT_TRUE(willPresentCalled);
}
@interface TestContextMenuHintPreviewContainerUIDelegate : NSObject <WKUIDelegate>
@end
@implementation TestContextMenuHintPreviewContainerUIDelegate
- (void)webView:(WKWebView *)webView contextMenuWillPresentForElement:(WKContextMenuElementInfo *)elementInfo
{
willPresentCalled = true;
}
- (UIView *)_contextMenuHintPreviewContainerViewForWebView:(WKWebView *)webView
{
didAskForHintPreviewContainer = true;
return nil;
}
@end
TEST(ContextMenu, HintPreviewContainer)
{
auto driver = contextMenuWebViewDriver([TestContextMenuHintPreviewContainerUIDelegate class]);
[driver begin:^(BOOL result) {
EXPECT_FALSE(result);
[driver clickDown];
[driver clickUp];
}];
TestWebKitAPI::Util::run(&willPresentCalled);
EXPECT_TRUE(didAskForHintPreviewContainer);
}
#endif // USE(UICONTEXTMENU)