openURL doesn't work in Share extension

2019-04-29 03:54发布

问题:

Trying to use [self.extensionContext openURL:... completionHandler:...]; in an iOS 8 Share extension to open the containing app never opens the app, and always calls the completion handler with success = NO.

Here is the same issue with Action extensions, but I think it is more reasonable for Share extensions to be able to open the containing app than Action extensions. The point of a Share extension is to upload a potentially large piece of data, and the only way to do that without opening the app is through NSURLSession, which can only do HTTP(S) uploads. But an app may wish to share content through a different mechanism than HTTP(S) uploads.

Apple documentation doesn't state that openURL... cannot be used for any particular type of extension. It's hard to know whether this is a bug or intended behavior. There is no official information about this.

回答1:

Here are some possible workarounds... Communicating with/opening Containing app from Share extension

The ones that actually work can be summed up as:

  • Using a UIDocumentInteractionController and registering your container app with a special extension type to open your container app from your extension
  • Using a dummy NSURLSessionTask to get the application:handleEventsForBackgroundURLSession:completionHandler: method of the UIApplicationDelegate class of your container app to be called. Then, in that method, you can call [[UIApplication sharedApplication] openURL:url] to open whatever you want.

Neither of these methods are perfect, but they do work (not sure if Apple would like them though). For more details you can check out the link. Hope this helps :)

Edit: You can also use a UIWebView as described https://stackoverflow.com/a/24614589/3923882



回答2:

The documentation does indicate that the "extension point determines whether to support this method" and in practice only those who use the Today Extension are able to launch their host app using openUrl. Share, Action, Keyboard, and Document provider do not work for anyone (beta 5).



回答3:

I'm using this method which does not involve any background task:

  • use a UIViewController (instead of the default SLComposeServiceViewController) subclass for share extension in order to add custom layout;
  • add a UIWebView and load an HTML string with a link pointing to your app's custom URL;
  • implement webView:shouldStartLoadWithRequest:navigationType: method to intercept the click on the link and before returning YES use NSUserDefaults with initWithSuiteName: (providing the name of the app group) to save data in the container shared by your extension and the containing app.

The click on the web link launches the app, which can fetch the data in NSUserDefaults directly.



回答4:

This is what I use from my keyboard extension. I hope it helps here too:

UIWebView * webView = [[UIWebView alloc] initWithFrame:CGRectMake(0, 0, 0, 0)];
NSString *urlString = @"https://itunes.apple.com/us/app/watuu/id304697459";
NSString * content = [NSString stringWithFormat : @"<head><meta http-equiv='refresh' content='0; URL=%@'></head>", urlString];
[webView loadHTMLString:content baseURL:nil];
[self.view addSubview:webView];
[webView performSelector:@selector(removeFromSuperview) withObject:nil afterDelay:2.0];

Please note that in this case I am instantiating this call from the UIInputViewController.

This method should also work using the URL scheme from the containing app