方法一:UIKit提供的便利方法
首先保存图片到系统相册UIKit提供了一种很简单的方法,一句话就搞定。但如果想要取得刚刚保存的地址就没办法了。
UIImageWriteToSavedPhotosAlbum(
UIImage * _Nonnull image,
id _Nullable completionTarget,
SEL _Nullable completionSelector,
void * _Nullable contextInfo
);
- 各参数说明:
image 为需要保存的图片对象,id是target对象,sel是selector,即target对象上的方法名,contextInfo是任意指针,会传递到selector定义的方法上。一般是当完成后调用方法时使用,或者在完成时出错的处理。
方法二:photoKit来保存
假如现在场景是:调用系统相机拍下一张照片,准备保存到系统相机胶卷,并获取到图片信息
。此时我们需要通过photoKit来保存,
// 保存相片到相机胶卷
NSError *error1 = nil;
__block PHObjectPlaceholder *createdAsset = nil;
[[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
createdAsset = [PHAssetCreationRequest creationRequestForAssetFromImage:image].placeholderForCreatedAsset;
} error:&error1];
接下来我们需要去获取刚刚保存到的图片,本人的基本思路是先拿到相机胶卷的所有照片
,并拿到相机胶卷这个相册中的所有PHAsset对象
,又由于我们刚刚保存的图片是存放在最后一张的,所以只需要拿数组内的最后一张照片的PHAsset对象
。最后再通过最后一个PHAsset对象(刚刚保存的照片)获取图片信息
。
/** 最后一张照片为刚刚拍照传上去的 */
PHAsset *PHasset = [allPhots lastObject];
这里仔细讲一下根据PHAsset获取图片信息中的[PHCachingImageManager defaultManager] requestImageForAsset
,本人在这里遇到一点坑。详细请参照iOS获取图像的方式和坑点
#pragma mark - < 根据PHAsset获取图片信息 >
- (void)accessToImageAccordingToTheAsset:(PHAsset *)asset size:(CGSize)size resizeMode:(PHImageRequestOptionsResizeMode)resizeMode completion:(void(^)(UIImage *image,NSDictionary *info))completion
{
static PHImageRequestID requestID = -1;
CGFloat scale = [UIScreen mainScreen].scale;
CGFloat width = MIN([UIScreen mainScreen].bounds.size.width, 500);
if (requestID >= 1 && size.width / width == scale) {
[[PHCachingImageManager defaultManager] cancelImageRequest:requestID];
}
PHImageRequestOptions *option = [[PHImageRequestOptions alloc] init];
option.deliveryMode = PHImageRequestOptionsDeliveryModeOpportunistic;
// option.resizeMode = PHImageRequestOptionsResizeModeFast;
option.resizeMode = resizeMode;
requestID = [[PHCachingImageManager defaultManager] requestImageForAsset:asset targetSize:size contentMode:PHImageContentModeAspectFill options:option resultHandler:^(UIImage * _Nullable result, NSDictionary * _Nullable info) {
/** 必须做判断,要不然将走多次完成的completion的block */
if (![[info objectForKey:PHImageResultIsDegradedKey] boolValue]) {
dispatch_async(dispatch_get_main_queue(), ^{
completion(result,info);
});
}
}];
}
由于本人,只需要从相册拿到我设置需要的对应尺寸的照片,但是实际上是会多次走completion的block,在完成这里主要解释一下下面这个判断:
需要注意的是,该方法在默认情况下是异步执行的,而且 Photos 库可能会多次执行 resultHandler 块,因为这种情况就是图像需要从 iCloud 中下载的情况。在 requestImageForAsset 返回的内容中,一开始的那一次请求中会返回一个小尺寸的图像版本,当高清图像还在下载时,开发者可以首先给用户展示这个低清的图像版本,然后 block 在多次调用后,最终会返回高清的原图。至于当前返回的图像是哪个版本的图像,可以通过 block 返回的 NSDictionary info 中获知,PHImageResultIsDegradedKey 表示当前返回的 UIImage 是低清图。如果需要判断是否已经获得高清图,可以这样判断:
if (![[info objectForKey:PHImageResultIsDegradedKey] boolValue]) {
//这样只会走一次获取到高清图时
dispatch_async(dispatch_get_main_queue(), ^{
completion(result,info);
});
最后将本人在拍完照后走完成代理时的详细操作附上,至于其中获取相册所有图片的方法可见本人上一篇文章:获取相册所有图片
#pragma mark--uiimagepickercontroller delegate
-(void)imagePickerController:(UIImagePickerController *)picker didFinishPickingMediaWithInfo:(NSDictionary<NSString *,id> *)info
{
CGFloat scale = [UIScreen mainScreen].scale;
if (picker.sourceType == UIImagePickerControllerSourceTypeCamera)
{
[picker dismissViewControllerAnimated:YES completion:^{
}];
UIImage *image = [info valueForKey:UIImagePickerControllerOriginalImage];
// 保存相片到相机胶卷
NSError *error1 = nil;
__block PHObjectPlaceholder *createdAsset = nil;
[[PHPhotoLibrary sharedPhotoLibrary] performChangesAndWait:^{
createdAsset = [PHAssetCreationRequest creationRequestForAssetFromImage:image].placeholderForCreatedAsset;
} error:&error1];
if (error1) {
}
else
{
NSMutableArray *allPhots = [NSMutableArray array];
// 相机胶卷
PHFetchResult *smartAlbums = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeSmartAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
for (NSInteger i = 0; i < smartAlbums.count; i++) {
// 是否按创建时间排序
PHFetchOptions *option = [[PHFetchOptions alloc] init];
option.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"creationDate" ascending:YES]];
option.predicate = [NSPredicate predicateWithFormat:@"mediaType == %ld", PHAssetMediaTypeImage];
PHCollection *collection = smartAlbums[i];
//遍历获取相册
if ([collection isKindOfClass:[PHAssetCollection class]]) {
if ([collection.localizedTitle isEqualToString:@"相机胶卷"]) {
PHAssetCollection *assetCollection = (PHAssetCollection *)collection;
PHFetchResult *fetchResult = [PHAsset fetchAssetsInAssetCollection:assetCollection options:nil];
NSArray *assets;
if (fetchResult.count > 0) {
// 某个相册里面的所有PHAsset对象
assets = [self getAllPhotosAssetInAblumCollection:assetCollection ascending:YES ];
[allPhots addObjectsFromArray:assets];
}
}
}
}
/** 最后一张照片为刚刚拍照传上去的 */
PHAsset *PHasset = [allPhots lastObject];
NSString *date = [NSString stringWithFormat:@"%@",PHasset.creationDate];
PhotoModel *model = [[PhotoModel alloc]init];
model.date= date;
model.type = [NSString stringWithFormat:@"%ld",PHasset.mediaType];
/** 自带异步 */
[self accessToImageAccordingToTheAsset:PHasset size:CGSizeMake(kMyimageSize * scale,kMyimageSize * scale) resizeMode:PHImageRequestOptionsResizeModeFast completion:^(UIImage *image, NSDictionary *info) {
model.thumbNailImage = image;
if (self.dataArray.count < 3) {
[self.dataArray addObject:model];
[self getimageView];
}else
{
if (_myHud.hidden || _myHud == nil) {
_myHud = [JGProgressHUD onlyShowMessage:@"抱歉!暂时最多发布三张。" withHUDStyle:JGProgressHUDStyleDark inVC:self];
}
}
}];
}
}
}