blob: 81accaba1fd13a6319842889953ed062e7aeaa2d [file] [log] [blame]
/*
* Copyright (C) 2018 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 "WKShareSheet.h"
#if PLATFORM(COCOA) && !PLATFORM(WATCHOS) && !PLATFORM(APPLETV)
#import "QuarantineSPI.h"
#import "WKWebViewInternal.h"
#import "WebPageProxy.h"
#import <WebCore/RuntimeApplicationChecks.h>
#import <WebCore/ShareData.h>
#import <wtf/RetainPtr.h>
#import <wtf/Scope.h>
#import <wtf/UUID.h>
#import <wtf/WeakObjCPtr.h>
#if PLATFORM(IOS_FAMILY)
#import "UIKitSPI.h"
#import "WKContentViewInteraction.h"
#else
#import <pal/spi/mac/NSSharingServicePickerSPI.h>
#endif
#if PLATFORM(MAC)
@interface WKShareSheet () <NSSharingServiceDelegate, NSSharingServicePickerDelegate>
@end
#endif
@implementation WKShareSheet {
RetainPtr<NSURL> _temporaryFileShareDirectory;
WeakObjCPtr<WKWebView> _webView;
WeakObjCPtr<id <WKShareSheetDelegate> > _delegate;
WTF::CompletionHandler<void(bool)> _completionHandler;
#if PLATFORM(MAC)
RetainPtr<NSSharingServicePicker> _sharingServicePicker;
#else
RetainPtr<UIActivityViewController> _shareSheetViewController;
RetainPtr<UIViewController> _presentationViewController;
#endif
}
- (id<WKShareSheetDelegate>)delegate
{
return _delegate.get().get();
}
- (void)setDelegate:(id<WKShareSheetDelegate>)delegate
{
_delegate = delegate;
}
- (instancetype)initWithView:(WKWebView *)view
{
if (!(self = [super init]))
return nil;
_webView = view;
return self;
}
- (void)presentWithParameters:(const WebCore::ShareDataWithParsedURL &)data inRect:(WTF::Optional<WebCore::FloatRect>)rect completionHandler:(WTF::CompletionHandler<void(bool)>&&)completionHandler
{
auto shareDataArray = adoptNS([[NSMutableArray alloc] init]);
if (!data.shareData.text.isEmpty())
[shareDataArray addObject:(NSString *)data.shareData.text];
if (data.url) {
NSURL *url = (NSURL *)data.url.value();
#if PLATFORM(IOS_FAMILY)
if (!data.shareData.title.isEmpty())
url._title = data.shareData.title;
#endif
[shareDataArray addObject:url];
}
if (!data.shareData.title.isEmpty() && ![shareDataArray count])
[shareDataArray addObject:(NSString *)data.shareData.title];
_completionHandler = WTFMove(completionHandler);
if (auto resolution = [_webView _resolutionForShareSheetImmediateCompletionForTesting]) {
[self _didCompleteWithSuccess:*resolution];
[self dismiss];
return;
}
if (data.files.size()) {
_temporaryFileShareDirectory = [WKShareSheet createTemporarySharingDirectory];
auto fileWriteGroup = adoptOSObject(dispatch_group_create());
auto queue = adoptOSObject(dispatch_queue_create("com.apple.WebKit.WKShareSheet.ShareableFileWriter", DISPATCH_QUEUE_SERIAL));
__block bool successful = true;
int index = 0;
for (auto file : data.files) {
dispatch_group_async(fileWriteGroup.get(), queue.get(), ^{
if (!successful)
return;
NSURL *fileURL = [WKShareSheet writeFileToShareableURL:WebCore::ResourceResponseBase::sanitizeSuggestedFilename(file.fileName) data:file.fileData->createNSData().get() temporaryDirectory:_temporaryFileShareDirectory.get()];
if (!fileURL) {
successful = false;
return;
}
[shareDataArray addObject:fileURL];
});
index++;
}
dispatch_group_notify(fileWriteGroup.get(), dispatch_get_main_queue(), ^{
if (!successful) {
[self _didCompleteWithSuccess:NO];
[self dismiss];
return;
}
[self presentWithShareDataArray:shareDataArray.get() inRect:rect];
});
return;
}
[self presentWithShareDataArray:shareDataArray.get() inRect:rect];
}
- (void)presentWithShareDataArray:(NSArray *)sharingItems inRect:(WTF::Optional<WebCore::FloatRect>)rect
{
WKWebView *webView = _webView.getAutoreleased();
#if PLATFORM(MAC)
_sharingServicePicker = adoptNS([[NSSharingServicePicker alloc] initWithItems:sharingItems]);
_sharingServicePicker.get().delegate = self;
// WKShareSheet can be released under NSSharingServicePicker delegate callbacks.
RetainPtr<WKShareSheet> protector(self);
NSRect presentationRect;
if (rect)
presentationRect = *rect;
else {
NSPoint location = [NSEvent mouseLocation];
NSRect mouseLocationRect = NSMakeRect(location.x, location.y, 1.0, 1.0);
NSRect mouseLocationInWindow = [webView.window convertRectFromScreen:mouseLocationRect];
presentationRect = [webView convertRect:mouseLocationInWindow fromView:nil];
}
[_sharingServicePicker showRelativeToRect:presentationRect ofView:webView preferredEdge:NSMinYEdge];
#else
_shareSheetViewController = adoptNS([[UIActivityViewController alloc] initWithActivityItems:sharingItems applicationActivities:nil]);
[_shareSheetViewController setCompletionWithItemsHandler:^(NSString *, BOOL completed, NSArray *, NSError *) {
[self _didCompleteWithSuccess:completed];
[self dispatchDidDismiss];
}];
UIPopoverPresentationController *popoverController = [_shareSheetViewController popoverPresentationController];
if (rect) {
popoverController.sourceView = webView;
popoverController.sourceRect = *rect;
} else
popoverController._centersPopoverIfSourceViewNotSet = YES;
_presentationViewController = [UIViewController _viewControllerForFullScreenPresentationFromView:webView];
[_presentationViewController presentViewController:_shareSheetViewController.get() animated:YES completion:nil];
#endif
}
#if PLATFORM(MAC)
- (void)sharingServicePicker:(NSSharingServicePicker *)sharingServicePicker didChooseSharingService:(NSSharingService *)service
{
if (service)
return;
[self _didCompleteWithSuccess:NO];
[self dispatchDidDismiss];
}
- (id <NSSharingServiceDelegate>)sharingServicePicker:(NSSharingServicePicker *)sharingServicePicker delegateForSharingService:(NSSharingService *)sharingService
{
return self;
}
- (NSWindow *)sharingService:(NSSharingService *)sharingService sourceWindowForShareItems:(NSArray *)items sharingContentScope:(NSSharingContentScope *)sharingContentScope
{
return [_webView window];
}
- (void)sharingService:(NSSharingService *)sharingService didFailToShareItems:(NSArray *)items error:(NSError *)error
{
[self _didCompleteWithSuccess:NO];
[self dispatchDidDismiss];
}
- (void)sharingService:(NSSharingService *)sharingService didShareItems:(NSArray *)items
{
[self _didCompleteWithSuccess:YES];
[self dispatchDidDismiss];
}
#endif
- (void)_didCompleteWithSuccess:(BOOL)success
{
auto completionHandler = WTFMove(_completionHandler);
if (completionHandler)
completionHandler(success);
if (success) {
// <rdar://problem/63030288>: didShareItems callback for NSSharingServiceDelegate currently is called
// before the temporary files are copied, so we can't delete them here. UIActivityViewController doesn't
// have this problem, so we can delete immediately for iOS.
#if PLATFORM(IOS_FAMILY)
[[NSFileManager defaultManager] removeItemAtURL:_temporaryFileShareDirectory.get() error:nil];
#endif
} else
[[NSFileManager defaultManager] removeItemAtURL:_temporaryFileShareDirectory.get() error:nil];
_temporaryFileShareDirectory = nullptr;
}
- (void)dismiss
{
// If programmatically dismissed without having already called the
// completion handler, call it assuming failure.
[self _didCompleteWithSuccess:NO];
#if PLATFORM(MAC)
[_sharingServicePicker hide];
[self dispatchDidDismiss];
#else
if (!_presentationViewController)
return;
UIViewController *currentPresentedViewController = [_presentationViewController presentedViewController];
if (currentPresentedViewController != _shareSheetViewController)
return;
[currentPresentedViewController dismissViewControllerAnimated:YES completion:^{
[self dispatchDidDismiss];
_presentationViewController = nil;
}];
#endif
}
- (void)dispatchDidDismiss
{
if ([_delegate respondsToSelector:@selector(shareSheetDidDismiss:)])
[_delegate shareSheetDidDismiss:self];
}
#if PLATFORM(MAC)
+ (BOOL)setQuarantineInformationForFilePath:(NSURL *)fileURL
{
auto quarantineProperties = @{
(__bridge NSString *)kLSQuarantineTypeKey: (__bridge NSString *)kLSQuarantineTypeWebDownload,
(__bridge NSString *)kLSQuarantineAgentBundleIdentifierKey: WebCore::applicationBundleIdentifier()
};
if (![fileURL setResourceValue:quarantineProperties forKey:NSURLQuarantinePropertiesKey error:nil])
return NO;
// Whether the file was downloaded by sandboxed WebProcess or not, LSSetItemAttribute resets the flags to 0 (advisory QTN_FLAG_DOWNLOAD,
// which can be then removed by WebProcess). Replace the flags with sandbox quarantine ones, which cannot be removed by sandboxed processes.
return [WKShareSheet applyQuarantineSandboxAndDownloadFlagsToFileAtPath:fileURL];
}
+ (BOOL)applyQuarantineSandboxAndDownloadFlagsToFileAtPath:(NSURL *)fileURL
{
qtn_file_t fq = qtn_file_alloc();
auto scopeExit = WTF::makeScopeExit([&] {
qtn_file_free(fq);
});
int quarantineError = qtn_file_init_with_path(fq, fileURL.fileSystemRepresentation);
if (quarantineError)
return NO;
quarantineError = qtn_file_set_flags(fq, QTN_FLAG_SANDBOX | QTN_FLAG_DOWNLOAD);
if (quarantineError)
return NO;
quarantineError = qtn_file_apply_to_path(fq, fileURL.fileSystemRepresentation);
return YES;
}
#endif
+ (NSURL *)createTemporarySharingDirectory
{
NSString *temporaryDirectory = FileSystem::createTemporaryDirectory(@"WKFileShare");
if (![temporaryDirectory length])
return nil;
return [NSURL fileURLWithPath:temporaryDirectory isDirectory:YES];
}
+ (NSURL *)createRandomSharingDirectoryForFile:(NSURL *)temporaryDirectory
{
NSString *randomDirectory = createCanonicalUUIDString();
if (![randomDirectory length] || !temporaryDirectory)
return nil;
NSURL *dataPath = [temporaryDirectory URLByAppendingPathComponent:randomDirectory];
if (![[NSFileManager defaultManager] createDirectoryAtURL:dataPath withIntermediateDirectories:NO attributes:nil error:nil])
return nil;
return dataPath;
}
+ (NSURL *)writeFileToShareableURL:(NSString *)fileName data:(NSData *)fileData temporaryDirectory:(NSURL *)temporaryDirectory
{
ASSERT(!RunLoop::isMain());
if (!temporaryDirectory || ![fileName length] || !fileData)
return nil;
NSURL *temporaryDirectoryForFile = [WKShareSheet createRandomSharingDirectoryForFile:temporaryDirectory];
if (!temporaryDirectoryForFile)
return nil;
NSURL *fileURL = [temporaryDirectoryForFile URLByAppendingPathComponent:fileName];
if (![fileData writeToURL:fileURL options:NSDataWritingAtomic error:nil])
return nil;
#if PLATFORM(MAC)
if (![WKShareSheet setQuarantineInformationForFilePath:fileURL])
return nil;
#endif
return fileURL;
}
@end
#endif // PLATFORM(COCOA) && !PLATFORM(WATCHOS) && !PLATFORM(APPLETV)