工作之余,研究了一下相机与相册的自定义,在这里整理成篇仅供参考学习,希望可以给大家带来些许帮助,也期待大家的批评指正。
自定义相册
GIF 示例:
ccpcamera.gif
DEMO下载地址:https://github.com/IMCCP/CCPCustomCamera
在 iOS 设备中,照片是相当重要的一部分。在 iOS8.0之前,开发者只能使用 AssetsLibrary 框架来访问设备的照片库。而在 iOS8 之后,苹果提供了一个名为 PhotoKit 的框架,一个可以让应用更好地与设备照片库对接的框架.由于市面上有一部分应用还支持iOS7,同时为了更加全面的学习,在这里将整理AssetsLibrary 框架与 PhotoKit 框架的相关知识,供大家参考学习.
一、AssetsLibrary 基本介绍
AssetsLibrary: 代表整个设备中的资源库(照片库),通过 AssetsLibrary 可以获取和包括设备中的照片和视频
ALAssetsGroup: 映射照片库中的一个相册,通过 ALAssetsGroup 可以获取某个相册的信息,相册下的资源,同时也可以对某个相册添加资源。
ALAsset: 映射照片库中的一个照片或视频,通过 ALAsset 可以获取某个照片或视频的详细信息,或者保存照片和视频。
ALAssetRepresentation: ALAssetRepresentation 是对 ALAsset 的封装(但不是其子类),可以更方便地获取 ALAsset 中的资源信息,每个 ALAsset 都有至少有一个 ALAssetRepresentation 对象,可以通过 defaultRepresentation 获取。
二、PhotoKit 基本介绍
PhotoKit 是一套比 AssetsLibrary 更完整也更高效的库,对资源的处理跟 AssetsLibrary 也有很大的不同。
PhotoKit 基本构成的介绍:
PHAsset: 代表照片库中的一个资源,跟 ALAsset 类似,通过 PHAsset 可以获取和保存资源
PHFetchOptions: 获取资源时的参数,可以传 nil,即使用系统默认值
PHAssetCollection: PHCollection 的子类,表示一个相册或者一个时刻,或者是一个「智能相册(系统提供的特定的一系列相册,例如:最近删除,视频列表,收藏等等,如下图所示)
PHFetchResult: 表示一系列的资源结果集合,也可以是相册的集合,从PHCollection 的类方法中获得
PHImageManager: 用于处理资源的加载,加载图片的过程带有缓存处理,可以通过传入一个 PHImageRequestOptions 控制资源的输出尺寸等规格
PHImageRequestOptions: 如上面所说,控制加载图片时的一系列参数
三、主要功能
1.获取相册图片资源;
2.自定义相册功能;
3.图片浏览器功能;
4.主要 工具类 介绍:
a.CCPPhotoAlbumViewController 控制器 主要要来进行相册资源的读取;
- (void)viewDidLoad {
[super viewDidLoad];
[self makeUI];
//进行系统型号的判断,调用不同的图库处理框架
if ([self iOSIsbefore_iOS8]) {
[self iOSBefore_iOS8];
} else {
[self iOSAfter_iOS8];
}
}
//系统版本小于8.0
- (void) iOSBefore_iOS8 {
//提示
NSString *tipTitle = nil;
//相册的访问状态
ALAuthorizationStatus authorizationStatus = [ALAssetsLibrary authorizationStatus];
/* 获取当前应用对照片的访问授权状态
ALAuthorizationStatusNotDetermined = 0, // 用户还没有做出选择这个应用程序的问候
ALAuthorizationStatusRestricted, // 这个应用程序没有被授权访问照片数据。当前用户不能改变应用程序的状态,是受限制的。如家长控制权限
ALAuthorizationStatusDenied, // 用户已拒绝该应用程序访问照片数据
ALAuthorizationStatusAuthorized // 用户已授权该应用可以访问
*/
// 如果没有获取访问授权,或者访问授权状态已经被明确禁止,则显示提示语,引导用户开启授权
if (authorizationStatus == ALAuthorizationStatusRestricted || authorizationStatus == ALAuthorizationStatusDenied) {
NSDictionary *mainInfoDictionary = [[NSBundle mainBundle] infoDictionary];
NSString *myAppName = [mainInfoDictionary objectForKey:@"CFBundleDisplayName"];
tipTitle = [NSString stringWithFormat:@"请在设备的\"设置-隐私-照片\"选项中,允许%@访问你的手机相册", myAppName];
} else {
[self.assetsLibrary enumerateGroupsWithTypes:ALAssetsGroupAll usingBlock:^(ALAssetsGroup *group, BOOL *stop) {
if (group) {
[group setAssetsFilter:[ALAssetsFilter allPhotos]];
if (group.numberOfAssets > 0) {
// 把相册储存到数组中,方便后面展示相册时使用
[self.assetsArray addObject:group];
[self.nameArray addObject:[group valueForProperty:ALAssetsGroupPropertyName]];
//获取相册封面图
UIImage *posterImage = [UIImage imageWithCGImage:[group posterImage]];
[self.posterImageArray addObject:posterImage];
}
} else {
if ([self.assetsArray count] > 0) {
// 把所有的相册储存完毕,可以展示相册列表
} else {
// 没有任何有资源的相册,输出提示
}
}
} failureBlock:^(NSError *error) {
NSLog(@"Asset group not found!\n");
}];
dispatch_async(dispatch_get_main_queue(), ^{
[self.showTableView reloadData];
});
}
}
//系统版本号大于等于8.0
- (void) iOSAfter_iOS8 {
/*
PHAssetCollectionTypeAlbum 自建相册
PHAssetCollectionTypeSmartAlbum 智能相册
PHAssetCollectionTypeMoment 时刻相册
智能相册子类型
PHAssetCollectionSubtypeSmartAlbumGeneric 通用的
PHAssetCollectionSubtypeSmartAlbumPanoramas 全景
PHAssetCollectionSubtypeSmartAlbumVideos 视屏
PHAssetCollectionSubtypeSmartAlbumFavorites 收藏
PHAssetCollectionSubtypeSmartAlbumTimelapses 延时视屏,也会在PHAssetCollectionSubtypeSmartAlbumVideos在出现
PHAssetCollectionSubtypeSmartAlbumAllHidden 隐藏的
PHAssetCollectionSubtypeSmartAlbumRecentlyAdded 最近添加
PHAssetCollectionSubtypeSmartAlbumBursts 连拍
PHAssetCollectionSubtypeSmartAlbumSlomoVideos Slomo是slow motion的缩写,高速摄影慢动作解析
PHAssetCollectionSubtypeSmartAlbumUserLibrary 用户所有的资源
PHAssetCollectionSubtypeSmartAlbumSelfPortraits 所有前置摄像头拍的照片和视屏
PHAssetCollectionSubtypeSmartAlbumScreenshots 所有的截屏图
*/
//权限
PHAuthorizationStatus status = [PHPhotoLibrary authorizationStatus];
if (status == PHAuthorizationStatusDenied || status == PHAuthorizationStatusRestricted) {
[self noticeAlerPhotos];
}else{
[PHPhotoLibrary requestAuthorization:^(PHAuthorizationStatus status) {
if (status == PHAuthorizationStatusAuthorized) {
PHFetchOptions *option = [[PHFetchOptions alloc] init];
//排序方式
option.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"modificationDate" ascending:NO]];
// 列出所有相册智能相册
PHFetchResult *smartAlbums = [PHAssetCollection fetchAssetCollectionsWithType:PHAssetCollectionTypeSmartAlbum subtype:PHAssetCollectionSubtypeAlbumRegular options:nil];
for (NSInteger i = 0; i < smartAlbums.count; i++) {
// 获取一个相册(PHAssetCollection)
PHCollection *collection = smartAlbums[i];
if ([collection isKindOfClass:[PHAssetCollection class]]) {
PHAssetCollection *assetCollection = (PHAssetCollection *)collection;
// 从每一个智能相册中获取到的 PHFetchResult 中包含的才是真正的资源(PHAsset)
PHFetchResult *fetchResult = [PHAsset fetchAssetsInAssetCollection:assetCollection options:nil];
if (fetchResult.count > 0) {
[self.assetsArray addObject:fetchResult];
[self.nameArray addObject:assetCollection.localizedTitle];
//获取封面图片,就是第一张图片
PHAsset *asset = (PHAsset *)fetchResult.firstObject;
PHImageRequestOptions *options = [[PHImageRequestOptions alloc] init];
//默认的是异步加载,这里选择了同步
options.synchronous = YES;
//PHImageManagerMaximumSize:获取原图,占用很大内存 建议不要使用
[[PHImageManager defaultManager] requestImageForAsset:asset targetSize:PHImageManagerMaximumSize contentMode:PHImageContentModeAspectFill options:options resultHandler:^(UIImage * _Nullable result, NSDictionary * _Nullable info)
{
[self.posterImageArray addObject:result];
}];
}
}else {
NSAssert(NO, @"Fetch collection not PHCollection: %@", collection);
}
dispatch_async(dispatch_get_main_queue(), ^{
[self.showTableView reloadData];
});
}
}
}];
}
// 获取所有资源的集合,并按资源的创建时间排序
// PHFetchOptions *options = [[PHFetchOptions alloc] init];
// options.sortDescriptors = @[[NSSortDescriptor sortDescriptorWithKey:@"creationDate" ascending:YES]];
// PHFetchResult *assetsFetchResults = [PHAsset fetchAssetsWithOptions:options];
//
// // 这时 assetsFetchResults 中包含的,应该就是各个资源(PHAsset)
// for (NSInteger i = 0; i < assetsFetchResults.count; i++) {
// // 获取一个资源(PHAsset)
// PHAsset *asset = assetsFetchResults[i];
// }
//
// NSLog(@"%@",assetsFetchResults);
}
b.CCPShowPhotoVC 控制器 主要用于获取到的对应照片的展示
如图:
c.XLPhotoBrowser 主要用于图片的浏览
本着不重复造轮子的原则,demo中图片浏览器使用了 XLPhotoBrowser
XLPhotoBrowser下载地址:https://github.com/Shannoon/XLPhotoBrowser
在这里对框架作者表示感谢!
四、参考:
a.http://kayosite.com/ios-development-and-detail-of-photo-framework.html
b.[http://www.jianshu.com/p/535bfe3c328f](http://www.jianshu.com/p/535bfe3c3
c.http://www.jianshu.com/p/cc85282fac5e
d. http://www.cnblogs.com/Jenaral/p/5580497.html
在这里对blog作者表示感谢!
由于篇幅有限,在这里不再详细的展开,demo中对一些坑点都做了详细的注释,如果您有需要可下载demo进行查看.
感谢您的阅读,期待您的 Star,如果在使用中您有任何问题,可以在 github issue,我会尽自己能力给您答复 。
网友评论