Sharing Extension in IOS8 beta

后端 未结 7 1908
天命终不由人
天命终不由人 2020-12-24 08:40

I\'m trying to create a sharing extension using the new iOS 8 app extensions. I tried to get the current URL of a Safari site to show it in a UILabel. Simple enough.

相关标签:
7条回答
  • 2020-12-24 09:16

    The other answers are all complicated and incomplete. They only work in Safari and do not work in Google Chrome. This works both in Google Chrome and Safari:

    override func viewDidLoad() {
        super.viewDidLoad()
    
        for item in extensionContext!.inputItems {
            if let attachments = item.attachments {
                for itemProvider in attachments! {
                    itemProvider.loadItemForTypeIdentifier("public.url", options: nil, completionHandler: { (object, error) -> Void in
                        if object != nil {
                            if let url = object as? NSURL {
                            print(url.absoluteString) //This is your URL
                            }
                        }
                    })
                }
            }
        }
    }
    
    0 讨论(0)
  • 2020-12-24 09:19

    You need to be looking for an attachment of type kUTTypePropertyList. Do something like this with the first attachment of the first extension item in your extension:

    NSExtensionItem *extensionItem = self.extensionContext.extensionItems.firstObject;
    NSItemProvider *itemProvider =  self.extensionItem.attachments.firstObject;
    [itemProvider loadItemForTypeIdentifier:kUTTypePropertyList options:nil
      completionHandler:^(NSDictionary *item, NSError *error) {
       // Unpack items from "item" in here
    }];
    

    You'll also need to setup some JavaScript to pick up all of the data you need from the DOM. See the second extensions session from WWDC 14 for more details.

    0 讨论(0)
  • 2020-12-24 09:20

    Your extension view controller should be adopting the NSExtensionRequestHandling protocol. One of this protocol's methods is:

    - (void)beginRequestWithExtensionContext:(NSExtensionContext *)context
    

    You should be waiting for this to be called before you attempt to get the NSExtensionContext. It even provides the context in the method as the context parameter.

    This was outlined in this document.

    0 讨论(0)
  • 2020-12-24 09:32

    All of those previous answers are really good but I just came accross this issue in Swift and felt it was a little tidious to extract the URL from a given NSExtensionContext especially in the CFString to String conversion process and the fact that the completionHandler in loadItemForTypeIdentifier is not executed in the main thread.

    import MobileCoreServices
    
    extension NSExtensionContext {
      private var kTypeURL:String {
          get {
              return kUTTypeURL as NSString as String
          }
      }
    
      func extractURL(completion: ((url:NSURL?) -> Void)?) -> Void {
          var processed:Bool = false
    
          for item in self.inputItems ?? [] {
              if  let item = item as? NSExtensionItem,
                  let attachments = item.attachments,
                  let provider = attachments.first as? NSItemProvider
                  where provider.hasItemConformingToTypeIdentifier(kTypeURL) == true {
                      provider.loadItemForTypeIdentifier(kTypeURL, options: nil, completionHandler: { (output, error) -> Void in
                          dispatch_async(dispatch_get_main_queue(), { () -> Void in
                              processed = true
                              if let url = output as? NSURL {
                                  completion?(url: url)
                              }
                              else {
                                  completion?(url: nil)
                              }
                          })
    
                      })
              }
          }
    
          // make sure the completion block is called even if no url could be extracted
          if (processed == false) {
              completion?(url: nil)
          }
      }
    }
    

    That way you can now simply use it like this in your UIViewController subclass:

    self.extensionContext?.extractURL({ (url) -> Void in
        self.urlLabel.text = url?.absoluteString
        println(url?.absoluteString)
    })
    
    0 讨论(0)
  • 2020-12-24 09:33

    Below is how you can get the url. Notice the type identifier is kUTTypeURL and the block argument is NSURL. Also, the plist needs to be correct like mine also. The documentation was lacking and got help from number4 on the Apple dev forums. (you'll need to be registered and logged in to see it).

    Code:

    NSExtensionItem *item = self.extensionContext.inputItems.firstObject;
    NSItemProvider *itemProvider = item.attachments.firstObject;
    if ([itemProvider hasItemConformingToTypeIdentifier:(NSString *)kUTTypeURL]) {
        [itemProvider loadItemForTypeIdentifier:(NSString *)kUTTypeURL options:nil completionHandler:^(NSURL *url, NSError *error) {
            self.urlString = url.absoluteString;
        }];
    }
    

    Info.plist

    <key>NSExtension</key>
    <dict>
        <key>NSExtensionAttributes</key>
        <dict>
            <key>NSExtensionActivationRule</key>
            <dict>
                <key>NSExtensionActivationSupportsWebURLWithMaxCount</key>
                <integer>1</integer>
            </dict>
            <key>NSExtensionPointName</key>
            <string>com.apple.share-services</string>
            <key>NSExtensionPointVersion</key>
            <string>1.0</string>
        </dict>
        <key>NSExtensionPointIdentifier</key>
        <string>com.apple.share-services</string>
        <key>NSExtensionMainStoryboard</key>
        <string>MainInterface</string>
    </dict>
    
    0 讨论(0)
  • 2020-12-24 09:36

    I've solved it for myself. I was trying with Sharing Image.

    - (void)didSelectPost {
    
    
    // This is called after the user selects Post. Do the upload of contentText and/or NSExtensionContext attachments.
    
    // Inform the host that we're done, so it un-blocks its UI. Note: Alternatively you could call super's -didSelectPost, which will similarly complete the extension context.
    
    // Verify that we have a valid NSExtensionItem
    NSExtensionItem *imageItem = [self.extensionContext.inputItems firstObject];
    if(!imageItem){
        return;
    }
    
    // Verify that we have a valid NSItemProvider
    NSItemProvider *imageItemProvider = [[imageItem attachments] firstObject];
    if(!imageItemProvider){
        return;
    }
    
    // Look for an image inside the NSItemProvider
    if([imageItemProvider hasItemConformingToTypeIdentifier:(NSString *)kUTTypeImage]){
        [imageItemProvider loadItemForTypeIdentifier:(NSString *)kUTTypeImage options:nil completionHandler:^(UIImage *image, NSError *error) {
            if(image){
                NSLog(@"image %@", image);
                // do your stuff here...
    
            }
        }];
        }
    // this line should not be here. Cos it's called before the block finishes.
    // and this is why the console log or any other task won't work inside the block
    [self.extensionContext completeRequestReturningItems:nil completionHandler:nil];
    
    }
    

    So what I did is just moved the [self.extensionContext completeRequestReturningItems:nil completionHandler:nil]; inside the block at the end of other tasks. The final working version look like this (Xcode 6 beta 5 on Mavericks OS X 10.9.4):

    - (void)didSelectPost {
    
    
    // This is called after the user selects Post. Do the upload of contentText and/or NSExtensionContext attachments.
    
    // Inform the host that we're done, so it un-blocks its UI. Note: Alternatively you could call super's -didSelectPost, which will similarly complete the extension context.
    
    // Verify that we have a valid NSExtensionItem
    NSExtensionItem *imageItem = [self.extensionContext.inputItems firstObject];
    if(!imageItem){
        return;
    }
    
    // Verify that we have a valid NSItemProvider
    NSItemProvider *imageItemProvider = [[imageItem attachments] firstObject];
    if(!imageItemProvider){
        return;
    }
    
    // Look for an image inside the NSItemProvider
    if([imageItemProvider hasItemConformingToTypeIdentifier:(NSString *)kUTTypeImage]){
        [imageItemProvider loadItemForTypeIdentifier:(NSString *)kUTTypeImage options:nil completionHandler:^(UIImage *image, NSError *error) {
            if(image){
                NSLog(@"image %@", image);
                // do your stuff here...
    
                // complete and return
                [self.extensionContext completeRequestReturningItems:nil completionHandler:nil];       
            }
        }];
        }
    // this line should not be here. Cos it's called before the block finishes.
    // and this is why the console log or any other task won't work inside the block
    // [self.extensionContext completeRequestReturningItems:nil completionHandler:nil];
    
    }
    

    I hope it'll work for URL sharing as well.

    0 讨论(0)
提交回复
热议问题