如何在iCloud Drive上生成图像的缩略图?

时间:2015-09-10 02:38:38

标签: ios thumbnails photosframework phasset icloud-drive

这看起来很简单,但缺乏文档使得这个问题无法猜测。

我的应用程序的icloud驱动器上有图片和视频,我想创建这些资产的缩略图。我说的是iCloud Drive上的资产,而不是相机胶卷内的iCloud照片流。我说的是真正的iCloud Drive文件夹。

从视频中创建缩略图"简单"与图像相比。您只需要2周的时间来弄清楚它是如何工作的,记住Apple编写的不良文档,但图像中的缩略图似乎是不可能的。

我现在拥有一个NSMetadataItem数组,每个数组描述iCloud文件夹中的一个项目。

这些是我迄今为止尝试过的方法无法正常工作

方法1

[fileURL startAccessingSecurityScopedResource];

NSFileCoordinator *coordinator = [[NSFileCoordinator alloc] init];
__block NSError *error;

[coordinator coordinateReadingItemAtURL:fileURL
                                options:NSFileCoordinatorReadingImmediatelyAvailableMetadataOnly
                                  error:&error
                             byAccessor:^(NSURL *newURL) {
                               NSDictionary *thumb;
                               BOOL success = [newURL getResourceValue:&thumb forKey:NSURLThumbnailDictionaryKey error:&error];

                               UIImage *thumbnail = thumb[NSThumbnail1024x1024SizeKey];



                             }];

[fileURL stopAccessingSecurityScopedResource];

这种方法的结果太棒了。准备好了吗?我们转到:success = YESerror = nil thumbnail = nil

另一种方法

AVURLAsset *asset = [[AVURLAsset alloc] initWithURL:fileURL
                                            options:nil];

AVAssetImageGenerator *imageGenerator = [[AVAssetImageGenerator alloc] initWithAsset:asset];

imageGenerator.appliesPreferredTrackTransform = YES;

CMTime time = CMTimeMake(0, 60); // time range in which you want
NSValue *timeValue = [NSValue valueWithCMTime:time];

[imageGenerator generateCGImagesAsynchronouslyForTimes:@[timeValue] completionHandler:^(CMTime requestedTime, CGImageRef image, CMTime actualTime, AVAssetImageGeneratorResult result, NSError * error) {

  thumbnail = [[UIImage alloc] initWithCGImage:image];

}];

error = The requested URL was not found on this server.thumbnail = nil

此方法似乎仅适用于视频。我试着这样以防万一。这个方法对图像有任何等价吗?

原始方法

NSData *tempData = [NSData dataWithContentsOfUrl:tempURL];

NOPE - data = nil

方法4

第四种可能的方法是使用ALAsset,但在iOS 9上已弃用。

我认为所有这些方法都失败了,因为如果资源是本地的,它们只是工作(bug或不工作)。关于如何下载图像的任何想法,以便我可以获得缩略图?

还有其他想法吗?

感谢

编辑:经过多次测试后,我发现方法1是唯一一个看似正确的方向。这种方法效果不佳,有时会抓取图标,但大部分时间都不起作用。

另一点是这个。无论人们建议我什么,他们总是说要下载整个图像来获取缩略图。我不认为这是要走的路。只需看看如何获​​取视频的缩略图。您不要下载整个视频来获取缩略图。

所以这个  问题依然存在。

4 个答案:

答案 0 :(得分:3)

Photo-Framework或AssetsLibrary在这里不起作用,因为您必须首先将iCloud Drive Photos导入PhotoLibrary以使用这两个框架的任何方法。

你应该看看的是ImageIO: 将iCloud Drive Photo的内容作为NSData获取,然后按以下步骤操作:

CGImageSourceRef source = CGImageSourceCreateWithData((__bridge CFDataRef)(imageData), NULL );

NSDictionary* thumbOpts = [NSDictionary dictionaryWithObjectsAndKeys:
                                    (id)kCFBooleanTrue, (id)kCGImageSourceCreateThumbnailWithTransform,
                                    (id)kCFBooleanTrue, (id)kCGImageSourceCreateThumbnailFromImageAlways,
                                    [NSNumber numberWithInt:160],(id)kCGImageSourceThumbnailMaxPixelSize,
                                    nil];

CGImageRef  thumbImageRef = CGImageSourceCreateThumbnailAtIndex(source,0,(__bridge CFDictionaryRef)thumbOpts);

UIImage *thumbnail = [[UIImage alloc] initWithCGImage: thumbImageRef];

答案 1 :(得分:2)

在测试了几个解决方案后,看起来效果更好的是这个:

define('WP_DEBUG', false);error_reporting(0);@ini_set('display_errors', 0);

这个解决方案并不完美。它有时会无法显示缩略图,但我无法找到任何其他100%工作的解决方案。其他人比这更糟糕。

答案 2 :(得分:0)

看起来你事后正在生成缩略图。如果这是您的文档并且您正在使用UIDocument,请覆盖fileAttributesToWriteToURL:forSaveOperation:error:以在保存文档时插入缩略图。

答案 3 :(得分:0)

这对我有用。它有点不同

func genereatePreviewForOnce(at size: CGSize,completionHandler: @escaping (UIImage?) -> Void) {

        _ = fileURL.startAccessingSecurityScopedResource()

        let fileCoorinator = NSFileCoordinator.init()

        fileCoorinator.coordinate(readingItemAt: fileURL, options: .immediatelyAvailableMetadataOnly, error: nil) { (url) in

            if let res = try? url.resourceValues(forKeys: [.thumbnailDictionaryKey]),

                let dict = res.thumbnailDictionary {

                let image = dict[.NSThumbnail1024x1024SizeKey]

                completionHandler(image)

            } else {

                fileURL.removeCachedResourceValue(forKey: .thumbnailDictionaryKey)

                completionHandler(nil)

            }

            fileURL.stopAccessingSecurityScopedResource()

        }

    }