这看起来很简单,但缺乏文档使得这个问题无法猜测。
我的应用程序的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 = YES
,error = 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是唯一一个看似正确的方向。这种方法效果不佳,有时会抓取图标,但大部分时间都不起作用。
另一点是这个。无论人们建议我什么,他们总是说要下载整个图像来获取缩略图。我不认为这是要走的路。只需看看如何获取视频的缩略图。您不要下载整个视频来获取缩略图。
所以这个 问题依然存在。
答案 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()
}
}