- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
现在我正在使用以下代码。但是发生了一些奇怪的事情。
PHLivePhoto *livePhoto = .....;
NSString *fileName = [[NSProcessInfo processInfo] globallyUniqueString];
NSURL *url = [NSURL fileURLWithPath: NSTemporaryDirectory()];
url = [url URLByAppendingPathComponent: fileName];
NSData *data = [NSKeyedArchiver archivedDataWithRootObject: livePhoto];
[data writeToURL: url atomically: YES];
CKAsset *asset = [[CKAsset alloc] initWithFileURL: url];
然后使用CKModifyRecordsOperation
将该 Assets 保存到iCloud,然后取回。
NSData *data = [NSData dataWithContentsOfURL: fetchedAsset.fileURL];
PHLivePhoto *thePhoto = [NSKeyedUnarchiver unarchiveObjectWithData: data];
PHLivePhotoView *photoView = ......;
photoView.livePhoto = thePhoto;
大部分都有效,除了当 photoView 停止播放时,photoView 的图像就消失了。如果我再次长按它,它就会正常播放。
为什么会这样?
最佳答案
NSKeyedArchiver
似乎没有按预期保存实时照片。作为解决方案之一,您应该拆解PHLivePhoto
并分别检索视频和静态图像,然后将它们上传到iCloud:
#import <Photos/Photos.h>
#import <CloudKit/CloudKit.h>
#import <MobileCoreServices/MobileCoreServices.h>
+ (void) disassembleLivePhoto:(nonnull PHLivePhoto*)livePhoto completion:(void (^__nonnull)(UIImage * _Nullable stillImage, AVURLAsset * _Nullable video, NSURL* _Nullable imageURL, NSURL* _Nullable videoURL))block
{
NSArray<PHAssetResource*>* resources = [PHAssetResource assetResourcesForLivePhoto:livePhoto];
__block PHAssetResource *resImage = nil, *resVideo = nil;
NSString *fileName = [[NSUUID UUID] UUIDString];
NSURL *urlMov = [NSURL fileURLWithPath:[NSTemporaryDirectory() stringByAppendingPathComponent:[fileName stringByAppendingPathExtension:@"mov"]]];
NSURL *urlImg = [NSURL fileURLWithPath:[NSTemporaryDirectory() stringByAppendingPathComponent:[fileName stringByAppendingPathExtension:@"jpg"]]];
[resources enumerateObjectsUsingBlock:^(PHAssetResource * _Nonnull res, NSUInteger idx, BOOL * _Nonnull stop) {
if (res.type == PHAssetResourceTypePairedVideo){
resVideo = res;
} else if (res.type == PHAssetResourceTypePhoto){
resImage = res;
}
}];
[[PHAssetResourceManager defaultManager] writeDataForAssetResource:resVideo toFile:urlMov options:nil completionHandler:^(NSError * _Nullable error) {
[[PHAssetResourceManager defaultManager] writeDataForAssetResource:resImage toFile:urlImg options:nil completionHandler:^(NSError * _Nullable error) {
block([UIImage imageWithData:[NSData dataWithContentsOfURL:urlImg]], [AVURLAsset assetWithURL:urlMov], urlImg, urlMov);
}];
}];
}
- (void) sendLivePhotoComponentsWithImageURL:(nonnull NSURL*)urlImage videoURL:(nonnull NSURL*)urlVideo completionBlock:(void(^ __nonnull)(CKRecord* __nullable recordLivePhoto))block
{
CKAsset *assetVideo = [[CKAsset alloc] initWithFileURL:urlVideo];
CKAsset *assetImage = [[CKAsset alloc] initWithFileURL:urlImage];
CKContainer *ckcContainer = [CKContainer defaultContainer];
CKDatabase *ckdbPublic = [ckcContainer publicCloudDatabase]; // in this example I use public DB
[ckcContainer fetchUserRecordIDWithCompletionHandler:^(CKRecordID * _Nullable ownerRecordID, NSError * _Nullable error) {
CKRecordID *recordID = [[CKRecordID alloc] initWithRecordName:@"your_record_name_e.g._UUID" zoneID:ownerRecordID.zoneID];
CKRecord *record = [[CKRecord alloc] initWithRecordType:@"your_record_type" recordID:recordID];
record[@"your_video_asset_CK_key"] = assetVideo;
record[@"your_image_asset_CK_key"] = assetImage;
CKModifyRecordsOperation * op = [[CKModifyRecordsOperation alloc] initWithRecordsToSave:@[record] recordIDsToDelete:nil];
op.modifyRecordsCompletionBlock = ^void(NSArray<CKRecord *> * _Nullable savedRecords, NSArray<CKRecordID *> * _Nullable deletedRecordIDs, NSError * _Nullable operationError){
block(savedRecords.firstObject); // Done.
};
op.qualityOfService = NSQualityOfServiceUserInitiated;
[ckdbPublic addOperation:op];
}];
}
第二部分(从 iCloud 检索)有一个小“技巧”——您应该确保图像和视频在元数据中具有相同的 Assets ID,否则 iOS 将不知道这两部分(视频和图像)属于到一个复合 Assets - 实时照片 - 并且无法将它们组合成单个正确 PHLivePhoto
对象(但是,在这种情况下,您很可能会 获取 PHLivePhoto
,但它将被构造为静态照片,没有动画)。
这里最简单的方法是从视频 Assets 中提取 Assets ID,然后通过为其分配相同的 ID 来修改图像部分:
- (void) assembleLivePhotoWithCKRecord:(nonnull CKRecord*)record completion:(void (^__nullable)(PHLivePhoto* _Nullable livePhoto))block
{
// Operational data
CKAsset *assetVideo = record[@"your_video_asset_CK_key"];
CKAsset *assetImage = record[@"your_image_asset_CK_key"];
// Get video and prepare local URLs
NSString *fileName = [[NSUUID UUID] UUIDString];
NSString *pathVideo = [NSTemporaryDirectory() stringByAppendingPathComponent:[fileName stringByAppendingPathExtension:@"mov"]];
NSString *pathImage = [NSTemporaryDirectory() stringByAppendingPathComponent:[fileName stringByAppendingPathExtension:@"jpg"]];
NSURL *urlVideo = [NSURL fileURLWithPath:pathVideo];
NSURL *urlImage = [NSURL fileURLWithPath:pathImage];
NSData *dataVideo = [NSData dataWithContentsOfURL:assetVideo.fileURL];
[[NSFileManager defaultManager] createFileAtPath:pathVideo contents:dataVideo attributes:nil];
// Getting video asset ID from metadata
NSString *metaID = nil;
NSArray<AVMetadataItem*>* metadata = [[AVURLAsset assetWithURL:urlVideo] metadata];
for (AVMetadataItem *md in metadata){
if ([md.identifier containsString:@"com.apple.quicktime.content.identifier"]){
metaID = (NSString*)(md.value);
break;
}
}
// Get image
NSData *dataImage = [NSData dataWithContentsOfURL:assetImage.fileURL];
UIImage *image = [UIImage imageWithData:dataImage];
CGImageRef ref = [image CGImage];
// Update image's metadata to make it conform video metadata
NSDictionary *imgMetadata = @{@"{MakerApple}": @{@"17": metaID}};
NSMutableData *imageData = [NSMutableData new];
CGImageDestinationRef dest = CGImageDestinationCreateWithData((CFMutableDataRef)imageData, kUTTypeJPEG, 1, nil);
CGImageDestinationAddImage(dest, ref, (CFDictionaryRef)imgMetadata);
CGImageDestinationFinalize(dest);
[imageData writeToFile:pathImage atomically:YES];
[PHLivePhoto requestLivePhotoWithResourceFileURLs:@[urlImage, urlVideo] placeholderImage:nil targetSize:CGSizeZero contentMode:PHImageContentModeAspectFit resultHandler:^(PHLivePhoto * _Nullable livePhoto, NSDictionary * _Nonnull info) {
block(livePhoto); // NOTE: this block may be called several times
}];
}
根据 Apple 的文档,可能会多次调用带有实时照片的结果 block (有关详细信息,请参阅 PHLivePhoto.h
):
The result handler will be called multiple times to deliver new PHLivePhoto instances with increasingly more content.
此外,请记住,您应该添加所有必要的检查(实际上有很多)和错误处理程序等。
关于ios - 如何使用 CloudKit API 将 PHLivePhoto 保存到 iCloud,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38106104/
我是 icloud 新手。我正在开发一个应用程序。在我的应用程序中,我想将图片备份到 iCloud 服务器。但我不想将这些图像传回我的 iPhone。 这可能吗?怎么可能呢。我是 iCloud 新手。
我对 iCloud 很陌生——我制作了一个容器,然后在其中创建了一个记录类型。然后部署到生产。 好吧,我不再想要这种记录类型,所以我尝试删除它。 Invalid attempt to delete a
我已经在 icloud 中复制了我的 sqlite 数据库,我想使用该 icloud sqlite 数据库作为插入/更新和删除。我为它编写了代码,但它在其他设备上不起作用(我有 2 个设备)。意味着当
当本地商店更改为 iCloud 商店时,我正在尝试接收消息。 这是一个关键事件。所以我的用例是一个新设备在从一个空的商店开始后接收 iCloud 商店。我想通知 View 更新接收到的内容。 我像这样
我试图让 iCloud 中的键值存储正常工作,但我遇到了一个奇怪的问题。 我使用以下方式存储数据: [[NSUbiquitousKeyValueStore defaultStore] setObjec
我遇到了有关 iCloud 同步过程的问题。 当用户决定激活 iCloud 时,本地存储在应用程序沙箱中的文档将移动到本地 iCloud 目录,然后删除沙箱中的文件。这是通过 setUbiquitou
我对 iCloud 的想法有点纠结,并发布了一个更一般的问题 here .我最大的问题是决定是否应该停止将用户数据放在应用程序沙箱中的旧文档文件夹中。为了说明我的问题: docs据我所知,不要给出答案
嗨,我开发了一个 iPhone 应用程序,我将 NSMutableArray 存储到 NSUserDefaults 中。但是现在我想将相同的 NSMutableArray/NSUserDefaults
有没有办法访问特定的 PHAsset在多个设备上唯一的 iCloud 相册中? 我编写了一个简单的应用程序,其中列出了缩略图和 .localIdentifier s 来自 iCloud 共享相册,并且
我正在尝试将应用程序转移到另一个 Apple ID。当我这样做时,我收到一条错误消息,指出该应用程序无法传输,因为该应用程序启用了 iCloud。 由于这不是故意的,我尝试禁用 iCloud 并向应用
我的 iPhone 应用程序的文档文件夹与文件应用程序上的 iCloud 位置存在问题。我的应用程序的文档文件夹出现在文件应用程序的设备端,但没有出现在文件应用程序的 iCloud 位置。iCloud
我一直在为客户运行一个流程,该流程涉及通过在 ruby 脚本中进行 https GET 调用,从 me.com 获取他们公开可用的日历文件,然后将 .ics 文件中的数据转换为 html,然后将其
我在 App Store 中有一个简单的应用程序,它是 UITableView由用户填写一些 UITextField 填充s 在模态调用的 View 中。用户按下保存,所有内容都保存到 Core Da
我有点困惑。每次用户单击“备份”时,我都使用 iCloud Drive 存储内部数据的备份。这不像我想的那样工作,因为有时它只是不会在设备和单击恢复的用户之间同步得到他们的旧备份。 现在我已经阅读了有
我有一个 iOS 应用程序,其中一半用户希望使用 iCloud 在设备之间同步一些数据,而另一半则不需要。出于这个原因,我将核心数据(启用 iCloud)的 sqlite 文件放在应用程序文档目录中。
我正在为 friend 的企业构建一个 i0S 8 iPad 应用程序,以帮助他和他的员工在设备之间同步评估数据。换句话说,如果一名员工创建了评估,则运行该应用程序的所有其他员工都应该可以访问该评估。
我正在创建一个应用程序(目前仅适用于 iPhone),它需要一种自动备份数据的方法,并让用户能够从不同的设备访问他们的数据。数据可能会在一段时间后达到一致的大小,因为它还将包括照片等。 我目前正在考虑
我非常接近在我的应用程序中完成 iCloud 和核心数据的实现。这是一款仅限 iOS 7 的应用程序。 我在应用程序中为用户提供一个选项,以 UISwitch 的形式说出打开或关闭 iCloud。该应
我正在开发一个使用核心数据的应用程序,因为我认为我可以使用核心数据为 iCloud 提供的同步功能在 iPad 和 iPhone 之间同步数据,现在它已被弃用。 我遇到了 CloudKit,但它主要用
应用程序在 iCloud 中存储数据似乎有几种不同的方式(iCloud Drive 文档、带 iCloud 同步的核心数据、CloudKit 私有(private)数据库、键值对等),但我不确定是否仅
我是一名优秀的程序员,十分优秀!