11

从照片库共享图像以从照片应用程序获取图像时,我使用了解决方案表单iOS 共享扩展问题。这在模拟器中效果很好,但在设备上我收到一个错误,我无法访问 itemProvider 提供的 NSURL:

2018-02-18 12:54:09.448148+0100 MyApp[6281:1653186] [default] [ERROR] 无法确定 URL /var/mobile/Media/PhotoData/OutgoingTemp/554581B2-950C-4CFD-AE67-A2342EDEA04D/ IMG_2784.JPG (s) 由文件提供者管理

声明引起:

[itemProvider loadItemForTypeIdentifier:itemProvider.registeredTypeIdentifiers.firstObject options:nil completionHandler:^(id<NSSecureCoding> item, NSError *error) {
}

在 PHAsset 中搜索项目名称并不是一个好的解决方案,因为用户必须再次访问照片库。

4

2 回答 2

2

在 didSelectPost 中,您必须在处理完 inputItems 数组中的所有项目之前关闭 ShareExtension ViewController。ViewController 使用 [super didSelectPost] 或通过调用扩展上下文的完成方法来解除。

这是我的代码解决方案:

- (void)didSelectPost {

__block NSInteger itemCount = ((NSExtensionItem*)self.extensionContext.inputItems[0]).attachments.count;
__block NSInteger processedCount = 0;
for (NSItemProvider* itemProvider in ((NSExtensionItem*)self.extensionContext.inputItems[0]).attachments ) {

    if([itemProvider hasItemConformingToTypeIdentifier:@"public.jpeg"]) {
        NSLog(@"itemprovider = %@", itemProvider);

        [itemProvider loadItemForTypeIdentifier:@"public.jpeg" options:nil completionHandler: ^(id<NSSecureCoding> item, NSError *error) {

            NSData *imgData;
            if([(NSObject*)item isKindOfClass:[NSURL class]]) {
                imgData = [NSData dataWithContentsOfURL:(NSURL*)item];
            }
            if([(NSObject*)item isKindOfClass:[UIImage class]]) {
                imgData = UIImagePNGRepresentation((UIImage*)item);
            }
            NSDictionary *dict = @{
                                   @"imgData" : imgData,
                                   @"name" : self.contentText
                                   };
            NSUserDefaults *defaults = [[NSUserDefaults alloc] initWithSuiteName:@"group.share.extension1"];
            [defaults setObject:dict forKey:@"img"];
            [defaults synchronize];
            processedCount += 1;
            if (processedCount == itemCount)
                [super didSelectPost];
        }];
    }
}
于 2019-01-05T18:33:26.370 回答
0

loadItemForTypeIdentifierSwift 中的orloadItem方法是异步的,因此关闭 UI 必须作为其completionHandler.

例如我有:

override func didSelectPost() {
    // This is called after the user selects Post. Do the upload of contentText and/or NSExtensionContext attachments.
    if let item = self.extensionContext?.inputItems[0] as? NSExtensionItem, let attachments = item.attachments {
        for provider in attachments {
            if provider.hasItemConformingToTypeIdentifier(kUTTypeImage as String) {
                provider.loadItem(forTypeIdentifier: kUTTypeImage as String, options: nil, completionHandler: {item, error in
                    // do all you need to do here, i.e.
                    if let tmpURL = item as? URL {
                        // etc. etc.
                    }
                    // and, at the end, inside the completionHandler, call
                    // 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.
                    self.extensionContext!.completeRequest(returningItems: [], completionHandler: nil)
                })
            }
        }
    }
}

我通过以下方式关闭 UI:

self.extensionContext!.completeRequest(returningItems: [], completionHandler: nil)

或通过:

super.didSelectPost()

completionHandlerafter the async方法之外,loadItem您将获得各种权限错误,此外,这些错误可能是随机的,有时会发生,有时不会,这是因为有时您的异步调用loadItem有机会在 UI 被关闭之前终止有时它不会。

把这个留在这里,希望它可以帮助某人。这个问题花了我几个小时。

于 2021-04-27T00:38:34.323 回答