KSMediaManager.m 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454
  1. //
  2. // KSMediaManager.m
  3. // StudentDaya
  4. //
  5. // Created by Kyle on 2020/1/3.
  6. // Copyright © 2020 DayaMusic. All rights reserved.
  7. //
  8. #import "KSMediaManager.h"
  9. #import <AssetsLibrary/AssetsLibrary.h>
  10. #import <Photos/Photos.h>
  11. #import "TZVideoPlayerController.h"
  12. #import "TZImagePickerController.h"
  13. #import <MobileCoreServices/MobileCoreServices.h>
  14. #import "KSPremissionAlert.h"
  15. #define COLUMNNUMBER (3)
  16. @interface KSMediaManager ()<UIImagePickerControllerDelegate,UINavigationControllerDelegate,TZImagePickerControllerDelegate>
  17. {
  18. BOOL _isSelectOriginalPhoto;
  19. }
  20. @property (nonatomic, strong) UIImagePickerController *imagePickerVc;
  21. @property (nonatomic, copy) MediaCallback callback;
  22. @property (nonatomic, strong) NSString *presentName;
  23. @end
  24. @implementation KSMediaManager
  25. - (instancetype)init {
  26. if (self = [super init]) {
  27. [self configDefaultConfig];
  28. }
  29. return self;
  30. }
  31. - (void)configDefaultConfig {
  32. self.mediaType = MEDIATYPE_PHOTO;
  33. self.maxPhotoNumber = 9;
  34. self.videoQuality = UIImagePickerControllerQualityType640x480;
  35. }
  36. - (void)setMediaType:(MEDIATYPE)mediaType {
  37. _mediaType = mediaType;
  38. if (mediaType == MEDIATYPE_PHOTO) {
  39. self.maxPhotoNumber = 9;
  40. }
  41. else if (mediaType == MEDIATYPE_VIDEO) {
  42. self.maxPhotoNumber = 1;
  43. }
  44. else if (mediaType == MEDIATYPE_ALL) {
  45. self.maxPhotoNumber = 2;
  46. }
  47. }
  48. - (void)setVideoQuality:(UIImagePickerControllerQualityType)videoQuality {
  49. _videoQuality = videoQuality;
  50. if ([[NSString deviceVersion] containsString:@"iPhone 12"]) {
  51. if (videoQuality == UIImagePickerControllerQualityType640x480) {
  52. _presentName = AVAssetExportPreset640x480;
  53. }
  54. else if (videoQuality == UIImagePickerControllerQualityTypeIFrame1280x720) {
  55. _presentName = AVAssetExportPreset1280x720;
  56. }
  57. else if (videoQuality == UIImagePickerControllerQualityTypeIFrame960x540) {
  58. _presentName = AVAssetExportPreset960x540;
  59. }
  60. else {
  61. _presentName = AVAssetExportPreset640x480;
  62. }
  63. }
  64. else {
  65. _presentName = AVAssetExportPresetMediumQuality;
  66. }
  67. }
  68. - (void)showAlertCallbackWithBlock:(MediaCallback)callback {
  69. UIAlertController *alertVC = [UIAlertController alertControllerWithTitle:nil message:nil preferredStyle:IS_IPAD ? UIAlertControllerStyleAlert : UIAlertControllerStyleActionSheet];
  70. [alertVC addAction:[UIAlertAction actionWithTitle:@"相机拍摄" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
  71. // 调用相机
  72. [self takePhoto];
  73. }]];
  74. [alertVC addAction:[UIAlertAction actionWithTitle:@"从手机相册选择" style:UIAlertActionStyleDefault handler:^(UIAlertAction * _Nonnull action) {
  75. // 调用相册
  76. [self pushImagePickerController];
  77. }]];
  78. [alertVC addAction:[UIAlertAction actionWithTitle:@"取消" style:UIAlertActionStyleCancel handler:^(UIAlertAction * _Nonnull action) {
  79. }]];
  80. alertVC.modalPresentationStyle = UIModalPresentationFullScreen;
  81. [self.baseCtrl presentViewController:alertVC animated:true completion:nil];
  82. if (callback) {
  83. self.callback = callback;
  84. }
  85. }
  86. - (void)noAlertCallback:(MediaCallback)callback {
  87. if (callback) {
  88. self.callback = callback;
  89. }
  90. }
  91. #pragma mark - TZImagePickerController
  92. - (void)extracted:(TZImagePickerController *)imagePickerVc {
  93. [imagePickerVc setDidFinishPickingPhotosHandle:^(NSArray<UIImage *> *photos, NSArray *assets, BOOL isSelectOriginalPhoto) {
  94. // 赋值
  95. if (self.callback) {
  96. dispatch_main_sync_safe(^{
  97. self.imageArray = [NSMutableArray arrayWithArray:photos];
  98. self.imageAsset = [NSMutableArray arrayWithArray:assets];
  99. self.callback(nil, self.imageArray, self.imageAsset);
  100. })
  101. }
  102. }];
  103. }
  104. - (void)pushImagePickerController {
  105. TZImagePickerController *imagePickerVc = [[TZImagePickerController alloc] initWithMaxImagesCount:self.maxPhotoNumber columnNumber:COLUMNNUMBER delegate:self];
  106. #pragma mark - 四类个性化设置,这些参数都可以不传,此时会走默认设置
  107. // 1.设置目前已经选中的图片数组
  108. imagePickerVc.selectedAssets = self.imageAsset; // 目前已经选中的图片数组
  109. imagePickerVc.isSelectOriginalPhoto = NO;
  110. // 2. Set the appearance
  111. // 2. 在这里设置imagePickerVc的外观
  112. [imagePickerVc.navigationBar setBarTintColor:THEMECOLOR];
  113. // 3. 设置是否可以选择视频/图片/原图
  114. if (self.mediaType == MEDIATYPE_ALL) {
  115. imagePickerVc.allowTakePicture = YES;
  116. imagePickerVc.allowTakeVideo = YES;
  117. imagePickerVc.allowPickingVideo = YES;
  118. imagePickerVc.allowPickingImage = YES;
  119. }
  120. else if (self.mediaType == MEDIATYPE_PHOTO) {
  121. imagePickerVc.allowTakePicture = YES;
  122. imagePickerVc.allowTakeVideo = NO;
  123. imagePickerVc.allowPickingVideo = NO;
  124. imagePickerVc.allowPickingImage = YES;
  125. }
  126. else if (self.mediaType == MEDIATYPE_VIDEO) {
  127. imagePickerVc.allowTakePicture = NO;
  128. imagePickerVc.allowTakeVideo = YES;
  129. imagePickerVc.allowPickingVideo = YES;
  130. imagePickerVc.allowPickingImage = NO;
  131. }
  132. imagePickerVc.videoMaximumDuration = self.videoMaxDuration ? self.videoMaxDuration : 480;
  133. // 设置视频拍摄质量
  134. [imagePickerVc setUiImagePickerControllerSettingBlock:^(UIImagePickerController *imagePickerController) {
  135. imagePickerController.videoQuality = UIImagePickerControllerQualityType640x480;
  136. }];
  137. imagePickerVc.allowCrop = self.needCropImage; // 单张才需要裁剪
  138. imagePickerVc.needCircleCrop = NO;
  139. imagePickerVc.showSelectBtn = YES;
  140. NSInteger left = 30;
  141. NSInteger widthHeight = kScreenWidth - 2 * left;
  142. NSInteger top = (kScreenHeight - widthHeight) / 2;
  143. imagePickerVc.cropRect = CGRectMake(left, top, widthHeight, widthHeight);
  144. // 4. 照片排列按修改时间升序
  145. imagePickerVc.sortAscendingByModificationDate = NO;
  146. // You can get the photos by block, the same as by delegate.
  147. // 你可以通过block或者代理,来得到用户选择的照片.
  148. [self extracted:imagePickerVc];
  149. [imagePickerVc setDidFinishPickingVideoHandle:^(UIImage *coverImage, PHAsset *asset) {
  150. NSInteger maxDuration = self.videoMaxDuration == 0 ? 480 : self.videoMaxDuration;
  151. if (asset.duration >= maxDuration + 1) {
  152. NSString *tipsMessage = @"";
  153. // 如果不是整分钟,提示秒 如果是整数分钟,提示分钟
  154. if (maxDuration % 60 == 0) {
  155. NSInteger minuteNum = maxDuration / 60;
  156. tipsMessage = [NSString stringWithFormat:@"视频长度不能超过%zd分钟", minuteNum];
  157. }
  158. else {
  159. tipsMessage = [NSString stringWithFormat:@"视频长度不能超过%zd秒",self.videoMaxDuration];
  160. }
  161. dispatch_main_async_safe(^{
  162. [LOADING_MANAGER MBShowAUTOHidingInWindow:tipsMessage];
  163. });
  164. return;
  165. }
  166. // PHAssetResource *resource = [[PHAssetResource assetResourcesForAsset:asset] firstObject];
  167. // long long size = [[resource valueForKey:@"fileSize"] longLongValue];
  168. //
  169. // NSLog(@"原视频大小:%@",[NSString stringWithFormat:@"%.2fM",(CGFloat)size/(1024*1024)]);
  170. dispatch_main_sync_safe(^{
  171. [LOADING_MANAGER MBShowInWindow:@"视频导出中..."];
  172. });
  173. dispatch_after(dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0.3 * NSEC_PER_SEC)), dispatch_get_main_queue(), ^{
  174. [[TZImageManager manager] getVideoOutputPathWithAsset:asset presetName:self.presentName success:^(NSString *outputPath) {
  175. dispatch_main_sync_safe(^{
  176. [LOADING_MANAGER removeHUDNoDelay];
  177. });
  178. NSLog(@"视频导出到本地完成,沙盒路径为:%@",outputPath);
  179. NSData *outputData = [NSData dataWithContentsOfURL:[NSURL fileURLWithPath:outputPath]]; //压缩后的视频
  180. NSLog(@"导出后的视频:%@",[NSString stringWithFormat:@"%.2fM",(CGFloat)outputData.length/(1024*1024)]);
  181. // Export completed, send video here, send by outputPath or NSData
  182. // 导出完成,在这里写上传代码,通过路径或者通过NSData上传
  183. if (self.callback) {
  184. dispatch_main_sync_safe(^{
  185. self.callback(outputPath, self.imageArray, self.imageAsset);
  186. });
  187. }
  188. } failure:^(NSString *errorMessage, NSError *error) {
  189. dispatch_main_sync_safe(^{
  190. [LOADING_MANAGER removeHUDNoDelay];
  191. [LOADING_MANAGER MBShowAUTOHidingInWindow:@"视频导出失败"];
  192. });
  193. NSLog(@"视频导出失败:%@,error:%@",errorMessage, error);
  194. }];
  195. });
  196. }];
  197. imagePickerVc.modalPresentationStyle = UIModalPresentationFullScreen;
  198. [self.baseCtrl presentViewController:imagePickerVc animated:YES completion:nil];
  199. }
  200. #pragma mark - UIImagePickerController
  201. - (void)takePhoto {
  202. AVAuthorizationStatus authStatus = [AVCaptureDevice authorizationStatusForMediaType:AVMediaTypeVideo];
  203. if ((authStatus == AVAuthorizationStatusRestricted || authStatus == AVAuthorizationStatusDenied) && kiOS7Later) {
  204. [KSPremissionAlert shareInstanceDisplayImage:CHECKDEVICETYPE_CAMREA message:@"请开启相机访问权限" showInView:self.baseCtrl.view cancel:^{
  205. } confirm:^{
  206. [self openSettingView];
  207. }];
  208. }
  209. else if (authStatus == AVAuthorizationStatusNotDetermined) {
  210. // 防止用户首次拍照拒绝授权时相机页黑屏
  211. [AVCaptureDevice requestAccessForMediaType:AVMediaTypeVideo completionHandler:^(BOOL granted) {
  212. if (granted) {
  213. dispatch_async(dispatch_get_main_queue(), ^{
  214. [self takePhoto];
  215. });
  216. }
  217. }];
  218. }
  219. else if ([PHPhotoLibrary authorizationStatus] == 2) { // 已被拒绝,没有相册权限,将无法保存拍的照片
  220. [KSPremissionAlert shareInstanceDisplayImage:CHECKDEVICETYPE_CAMREA message:@"请开启相册访问权限" showInView:self.baseCtrl.view cancel:^{
  221. } confirm:^{
  222. [self openSettingView];
  223. }];
  224. }
  225. else if ([PHPhotoLibrary authorizationStatus] == 0) { // 未请求过相册权限
  226. [[TZImageManager manager] requestAuthorizationWithCompletion:^{
  227. [self takePhoto];
  228. }];
  229. }
  230. else { // 调用相机
  231. UIImagePickerControllerSourceType sourceType = UIImagePickerControllerSourceTypeCamera;
  232. if ([UIImagePickerController isSourceTypeAvailable: UIImagePickerControllerSourceTypeCamera]) {
  233. self.imagePickerVc.sourceType = sourceType;
  234. NSMutableArray *mediaTypes = [NSMutableArray array];
  235. if (self.mediaType == MEDIATYPE_ALL) {
  236. [mediaTypes addObject:(NSString *)kUTTypeMovie];
  237. [mediaTypes addObject:(NSString *)kUTTypeImage];
  238. }
  239. else if (self.mediaType == MEDIATYPE_VIDEO) {
  240. [mediaTypes addObject:(NSString *)kUTTypeMovie];
  241. }
  242. else {
  243. [mediaTypes addObject:(NSString *)kUTTypeImage];
  244. }
  245. self.imagePickerVc.mediaTypes = mediaTypes;
  246. if(kiOS8Later) {
  247. _imagePickerVc.modalPresentationStyle = UIModalPresentationOverCurrentContext;
  248. }
  249. _imagePickerVc.modalPresentationStyle = UIModalPresentationFullScreen;
  250. [self.baseCtrl presentViewController:_imagePickerVc animated:YES completion:nil];
  251. } else {
  252. NSLog(@"模拟器中无法打开照相机,请在真机中使用");
  253. }
  254. }
  255. }
  256. // 外部拍照进入的方法
  257. - (void)imagePickerController:(UIImagePickerController*)picker didFinishPickingMediaWithInfo:(NSDictionary *)info {
  258. [picker dismissViewControllerAnimated:YES completion:nil];
  259. NSString *type = [info objectForKey:UIImagePickerControllerMediaType];
  260. TZImagePickerController *tzImagePickerVc = [[TZImagePickerController alloc] initWithMaxImagesCount:self.maxPhotoNumber delegate:self];
  261. [tzImagePickerVc showProgressHUD];
  262. if ([type isEqualToString:@"public.image"]) {
  263. tzImagePickerVc.allowCrop = self.needCropImage;
  264. tzImagePickerVc.needCircleCrop = NO;
  265. tzImagePickerVc.showSelectBtn = YES;
  266. tzImagePickerVc.sortAscendingByModificationDate = NO;
  267. UIImage *image = [info objectForKey:UIImagePickerControllerOriginalImage];
  268. // save photo and get asset / 保存图片,获取到asset
  269. [[TZImageManager manager] savePhotoWithImage:image completion:^(PHAsset *asset,NSError *error){
  270. [tzImagePickerVc hideProgressHUD];
  271. if (error) { // 如果保存失败,基本是没有相册权限导致的...
  272. [tzImagePickerVc hideProgressHUD];
  273. [KSPremissionAlert shareInstanceDisplayImage:CHECKDEVICETYPE_CAMREA message:@"请开启相册访问权限" showInView:self.baseCtrl.view cancel:^{
  274. } confirm:^{
  275. [self openSettingView];
  276. }];
  277. } else {
  278. TZAssetModel *assetModel = [[TZImageManager manager] createModelWithAsset:asset];
  279. if (self.needCropImage) {
  280. TZImagePickerController *imagePicker = [[TZImagePickerController alloc] initCropTypeWithAsset:assetModel.asset photo:image completion:^(UIImage *cropImage, id asset) {
  281. // 回调
  282. if (self.callback) {
  283. self.imageArray = [NSMutableArray arrayWithObject:cropImage];
  284. self.imageAsset = [NSMutableArray arrayWithObject:asset];
  285. dispatch_async(dispatch_get_main_queue(), ^{
  286. self.callback(nil, self.imageArray, self.imageAsset);
  287. });
  288. }
  289. }];
  290. imagePicker.allowPickingImage = YES;
  291. imagePicker.allowCrop = self.needCropImage;
  292. imagePicker.needCircleCrop = NO;
  293. NSInteger left = 30;
  294. NSInteger widthHeight = kScreenWidth - 2 * left;
  295. NSInteger top = (kScreenHeight - widthHeight) / 2;
  296. imagePicker.cropRect = CGRectMake(left, top, widthHeight, widthHeight);
  297. imagePicker.modalPresentationStyle = UIModalPresentationFullScreen;
  298. [self.baseCtrl presentViewController:imagePicker animated:YES completion:nil];
  299. }
  300. else {
  301. if (self.callback) {
  302. [self.imageArray addObject:image];
  303. [self.imageAsset addObject:assetModel.asset];
  304. dispatch_async(dispatch_get_main_queue(), ^{
  305. self.callback(nil, self.imageArray, self.imageAsset);
  306. });
  307. }
  308. }
  309. }
  310. }];
  311. }
  312. else if ([type isEqualToString:@"public.movie"]) {
  313. tzImagePickerVc.videoMaximumDuration = self.videoMaxDuration ? self.videoMaxDuration : 480;
  314. // 设置视频拍摄质量
  315. [tzImagePickerVc setUiImagePickerControllerSettingBlock:^(UIImagePickerController *imagePickerController) {
  316. imagePickerController.videoQuality = UIImagePickerControllerQualityType640x480;
  317. }];
  318. NSURL *videoUrl = [info objectForKey:UIImagePickerControllerMediaURL];
  319. if (videoUrl) {
  320. [[TZImageManager manager] saveVideoWithUrl:videoUrl completion:^(PHAsset *asset, NSError *error) {
  321. // 删除文件
  322. [self removeVideoWithPath:videoUrl.path];
  323. [tzImagePickerVc hideProgressHUD];
  324. if (!error) {
  325. dispatch_main_async_safe(^{
  326. [LOADING_MANAGER MBShowInWindow:@"视频处理中..."];
  327. });
  328. [[TZImageManager manager] getVideoOutputPathWithAsset:asset presetName:self.presentName success:^(NSString *outputPath) {
  329. // NSData *data = [NSData dataWithContentsOfFile:outputPath];
  330. dispatch_main_async_safe(^{
  331. [LOADING_MANAGER removeHUD];
  332. });
  333. NSLog(@"视频导出到本地完成,沙盒路径为:%@",outputPath);
  334. // Export completed, send video here, send by outputPath or NSData
  335. // 导出完成,在这里写上传代码,通过路径或者通过NSData上传
  336. NSData *outputData = [NSData dataWithContentsOfURL:[NSURL fileURLWithPath:outputPath]]; //压缩后的视频
  337. NSLog(@"导出后的视频:%@",[NSString stringWithFormat:@"%.2fM",(CGFloat)outputData.length/(1024*1024)]);
  338. // Export completed, send video here, send by outputPath or NSData
  339. // 导出完成,在这里写上传代码,通过路径或者通过NSData上传
  340. if (self.callback) {
  341. self.callback(outputPath, self.imageArray, self.imageAsset);
  342. }
  343. } failure:^(NSString *errorMessage, NSError *error) {
  344. dispatch_main_async_safe(^{
  345. [LOADING_MANAGER removeHUD];
  346. [LOADING_MANAGER MBShowAUTOHidingInWindow:@"视频导出失败"];
  347. });
  348. NSLog(@"视频导出失败:%@,error:%@",errorMessage, error);
  349. }];
  350. }
  351. }];
  352. }
  353. }
  354. }
  355. - (void)removeVideoWithPath:(NSString *)videoUrl {
  356. NSFileManager *fileMamager = [NSFileManager defaultManager];
  357. if ([fileMamager fileExistsAtPath:videoUrl]) {
  358. [fileMamager removeItemAtPath:videoUrl error:nil];
  359. }
  360. }
  361. - (void)imagePickerControllerDidCancel:(UIImagePickerController *)picker {
  362. if ([picker isKindOfClass:[UIImagePickerController class]]) {
  363. [picker dismissViewControllerAnimated:YES completion:nil];
  364. }
  365. }
  366. #pragma mark - getter
  367. - (NSMutableArray *)imageArray {
  368. if (!_imageArray) {
  369. _imageArray = [NSMutableArray array];
  370. }
  371. return _imageArray;
  372. }
  373. - (NSMutableArray *)imageAsset {
  374. if (!_imageAsset) {
  375. _imageAsset = [NSMutableArray array];
  376. }
  377. return _imageAsset;
  378. }
  379. - (UIImagePickerController *)imagePickerVc {
  380. if (_imagePickerVc == nil) {
  381. _imagePickerVc = [[UIImagePickerController alloc] init];
  382. _imagePickerVc.delegate = self;
  383. // set appearance / 改变相册选择页的导航栏外观
  384. _imagePickerVc.navigationBar.barTintColor = self.baseCtrl.navigationController.navigationBar.barTintColor;
  385. _imagePickerVc.navigationBar.tintColor = self.baseCtrl.navigationController.navigationBar.tintColor;
  386. UIBarButtonItem *tzBarItem, *BarItem;
  387. if (kiOS9Later) {
  388. tzBarItem = [UIBarButtonItem appearanceWhenContainedInInstancesOfClasses:@[[TZImagePickerController class]]];
  389. BarItem = [UIBarButtonItem appearanceWhenContainedInInstancesOfClasses:@[[UIImagePickerController class]]];
  390. } else {
  391. #pragma clang diagnostic push
  392. #pragma clang diagnostic ignored "-Wdeprecated-declarations"
  393. tzBarItem = [UIBarButtonItem appearanceWhenContainedIn:[TZImagePickerController class], nil];
  394. BarItem = [UIBarButtonItem appearanceWhenContainedIn:[UIImagePickerController class], nil];
  395. #pragma clang diagnostic pop
  396. }
  397. NSDictionary *titleTextAttributes = [tzBarItem titleTextAttributesForState:UIControlStateNormal];
  398. [BarItem setTitleTextAttributes:titleTextAttributes forState:UIControlStateNormal];
  399. }
  400. return _imagePickerVc;
  401. }
  402. - (void)openSettingView {
  403. [[UIApplication sharedApplication] openURL:[NSURL URLWithString:UIApplicationOpenSettingsURLString] options:@{} completionHandler:nil];
  404. }
  405. @end