KSChatConversationViewController.m 31 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738
  1. //
  2. // KSChatConversationViewController.m
  3. // KulexiuForTeacher
  4. //
  5. // Created by Kyle on 2022/3/23.
  6. //
  7. #import "KSChatConversationViewController.h"
  8. #import "GroupSettingViewController.h"
  9. #import "KSTabBarViewController.h"
  10. #import "UIButton+EnlargeEdge.h"
  11. #import "CustomNavViewController.h"
  12. #import <AVFoundation/AVFoundation.h>
  13. #import "KSBaseWKWebViewController.h"
  14. #import "WMPlayer.h"
  15. #import "KSRCloudMediaManager.h"
  16. #import "KSSelectConversationViewController.h"
  17. #import "GroupMemberModel.h"
  18. #define SHARE_MUSIC_TAG (2001)
  19. #import "KSChatLiveMessage.h"
  20. #import "KSChatLiveShareCell.h"
  21. #import "KSChatMusicShareCell.h"
  22. #import "KSChatMusicMessage.h"
  23. #import "KSEnterLiveroomManager.h"
  24. #import "KSAccompanyWebViewController.h"
  25. #import "KSPublicAlertView.h"
  26. #import "KSGroupTagImageView.h"
  27. @interface RCNaviDataInfo : NSObject
  28. @property (nonatomic, assign) NSTimeInterval uploadVideoDurationLimit;
  29. @end
  30. @interface RCNaviDataManager : NSObject
  31. @property (nonatomic, strong) RCNaviDataInfo *naviData;
  32. + (instancetype)sharedInstance;
  33. @end
  34. @interface RCSightMessage ()
  35. @property (nonatomic, readonly) AVAsset *asset;
  36. @property (nonatomic, assign) long long size;
  37. + (instancetype)messageWithAsset:(AVAsset *)asset thumbnail:(UIImage *)image duration:(NSUInteger)duration;
  38. @end
  39. @interface KSChatConversationViewController ()<RCMessageCellDelegate,UIAlertViewDelegate,UIActionSheetDelegate,WMPlayerDelegate>
  40. {
  41. WMPlayer *_wmPlayer;
  42. CGRect _playerFrame;
  43. }
  44. @property (nonatomic, strong) UIView *bgView;
  45. @property (nonatomic, assign) BOOL isRatation;
  46. @property (nonatomic, assign) BOOL isFirstLoad;
  47. @property (nonatomic, strong) KSPublicAlertView *alertView;
  48. @end
  49. @implementation KSChatConversationViewController
  50. - (void)viewDidLoad {
  51. [super viewDidLoad];
  52. // Do any additional setup after loading the view.
  53. if (self.conversationType == ConversationType_PRIVATE) {
  54. self.displayUserNameInCell = NO;
  55. }
  56. else if (self.conversationType == ConversationType_SYSTEM) {
  57. self.displayUserNameInCell = NO;
  58. self.chatSessionInputBarControl.hidden = YES;
  59. self.conversationMessageCollectionView.frame = CGRectMake(0, kNaviBarHeight, kScreenWidth, kScreenHeight - iPhoneXSafeBottomMargin - kNaviBarHeight);
  60. }
  61. if (self.conversationType != ConversationType_APPSERVICE &&
  62. self.conversationType != ConversationType_PUBLICSERVICE) {
  63. //加号区域增加发送文件功能,Kit中已经默认实现了该功能,但是为了SDK向后兼容性,目前SDK默认不开启该入口,可以参考以下代码在加号区域中增加发送文件功能。
  64. UIImage *imageFile = [UIImage imageNamed:@"actionbar_file_icon"];
  65. RCPluginBoardView *pluginBoardView = self.chatSessionInputBarControl.pluginBoardView;
  66. [pluginBoardView insertItem:imageFile
  67. highlightedImage:imageFile
  68. title:@"文件"
  69. atIndex:3
  70. tag:PLUGIN_BOARD_ITEM_FILE_TAG];
  71. }
  72. // 注册自定义消息
  73. [self registerClass:[KSChatLiveShareCell class] forMessageClass:[KSChatLiveMessage class]];
  74. [self registerClass:[KSChatMusicShareCell class] forMessageClass:[KSChatMusicMessage class]];
  75. [self leftRightButton];
  76. [self refreshUserInfoOrGroupInfo];
  77. self.isFirstLoad = YES;
  78. }
  79. - (void)viewWillAppear:(BOOL)animated {
  80. [super viewWillAppear:animated];
  81. if (self.isFirstLoad == NO) {
  82. [self refreshUserInfoOrGroupInfo];
  83. }
  84. [self refreshTitle];
  85. }
  86. - (void)viewDidAppear:(BOOL)animated {
  87. [super viewDidAppear:animated];
  88. self.isFirstLoad = NO;
  89. [IQKeyboardManager sharedManager].enableAutoToolbar = NO;
  90. }
  91. - (void)viewWillDisappear:(BOOL)animated {
  92. [super viewWillDisappear:animated];
  93. int unreadMsgCount = [[RCIMClient sharedRCIMClient] getUnreadCount:@[
  94. @(ConversationType_PRIVATE), @(ConversationType_APPSERVICE), @(ConversationType_GROUP),@(ConversationType_SYSTEM)
  95. ]];
  96. if (unreadMsgCount >= 1) {
  97. [(KSTabBarViewController *)self.tabBarController noteNewsWithIndex:2 count:unreadMsgCount];
  98. } else {
  99. [(KSTabBarViewController *)self.tabBarController clearNewsWithIndex:2];
  100. }
  101. [IQKeyboardManager sharedManager].enableAutoToolbar = YES;
  102. }
  103. - (void)notifyUpdateUnreadMessageCount {
  104. if (self.allowsMessageCellSelection) {
  105. [super notifyUpdateUnreadMessageCount];
  106. return;
  107. }
  108. [self leftRightButton];
  109. }
  110. - (void)leftRightButton {
  111. dispatch_main_async_safe(^{
  112. UIBarButtonItem *leftButton = [[UIBarButtonItem alloc] initWithImage:[UIImage imageNamed:@"back_black"] style:UIBarButtonItemStylePlain target:self action:@selector(backAction)];
  113. leftButton.imageInsets = UIEdgeInsetsMake(0, -5, 0, 0);
  114. leftButton.tintColor = [UIColor blackColor];
  115. self.navigationItem.leftBarButtonItem = leftButton;
  116. if (self.conversationType == ConversationType_GROUP) {
  117. [self rightButtonWithImage:@"group_setting"];
  118. }
  119. else if (self.conversationType == ConversationType_SYSTEM || self.conversationType == ConversationType_PRIVATE) {
  120. self.navigationItem.rightBarButtonItem = nil;
  121. }
  122. });
  123. }
  124. - (void)rightButtonWithImage:(NSString *)imageName {
  125. UIButton *rightBt = [UIButton buttonWithType:UIButtonTypeCustom];
  126. rightBt.frame =CGRectMake(0, 0, 80, 40);
  127. rightBt.contentEdgeInsets = UIEdgeInsetsMake(0, 0, 0, -40);
  128. [rightBt setEnlargeEdgeWithTop:10 right:10 bottom:10 left:10];
  129. rightBt.titleLabel.font = [UIFont systemFontOfSize:16];
  130. [rightBt setTitleColor:[UIColor blackColor] forState:UIControlStateNormal];
  131. [rightBt setImage:[UIImage imageNamed:imageName] forState:UIControlStateNormal];
  132. [rightBt addTarget:self action:@selector(rightBtnClick) forControlEvents:UIControlEventTouchUpInside];
  133. UIBarButtonItem *rightItem = [[UIBarButtonItem alloc]initWithCustomView:rightBt];
  134. self.navigationItem.rightBarButtonItem = rightItem;
  135. }
  136. - (void)backAction {
  137. [self.navigationController popViewControllerAnimated:YES];
  138. }
  139. - (void)rightBtnClick {
  140. GroupSettingViewController *groupVC = [[GroupSettingViewController alloc] init];
  141. groupVC.groupId = self.targetId;
  142. [self.navigationController pushViewController:groupVC animated:YES];
  143. }
  144. - (NSArray<UIMenuItem *> *)getLongTouchMessageCellMenuList:(RCMessageModel *)model {
  145. if (self.conversationType == ConversationType_SYSTEM) {
  146. return [NSArray array];
  147. }
  148. return [super getLongTouchMessageCellMenuList:model];
  149. }
  150. // 转发
  151. - (void)forwardMessage:(NSInteger)index
  152. completed:(void (^)(NSArray<RCConversation *> *conversationList))completedBlock {
  153. KSSelectConversationViewController *forwardSelectedVC = [[KSSelectConversationViewController alloc]
  154. initSelectConversationViewControllerCompleted:^(NSArray<RCConversation *> *conversationList) {
  155. completedBlock(conversationList);
  156. }];
  157. [self.navigationController pushViewController:forwardSelectedVC animated:NO];
  158. }
  159. #pragma mark override
  160. // 点击消息的处理
  161. - (void)didTapMessageCell:(RCMessageModel *)model {
  162. if ([model.objectName isEqualToString:@"RC:CHATSHARE:LIVE"]) { // 直播消息
  163. KSChatLiveMessage *liveShareMsg = (KSChatLiveMessage *)model.content;
  164. [KSEnterLiveroomManager joinLiveWithRoomId:liveShareMsg.roomUID inController:(CustomNavViewController *)self.navigationController callback:^{
  165. }];
  166. }
  167. else if ([model.objectName isEqualToString:@"RC:CHATSHARE:MUSIC"]) { // 曲谱消息
  168. KSChatMusicMessage *musicShareMsg = (KSChatMusicMessage *)model.content;
  169. KSAccompanyWebViewController *detailCtrl = [[KSAccompanyWebViewController alloc] init];
  170. detailCtrl.url = [NSString stringWithFormat:@"%@/accompany?id=%@",hostURL, musicShareMsg.songId];
  171. detailCtrl.hiddenNavBar = YES;
  172. detailCtrl.parmDic = @{@"isOpenLight" : @(YES), @"orientation" : @(0),@"isHideTitle" : @(YES)};
  173. [self.navigationController pushViewController:detailCtrl animated:YES];
  174. }
  175. else {
  176. [super didTapMessageCell:model];
  177. }
  178. }
  179. - (void)playVideoWithUrl:(NSString *)fileUrl {
  180. fileUrl = [fileUrl stringByAddingPercentEncodingWithAllowedCharacters:[NSCharacterSet URLQueryAllowedCharacterSet]];
  181. _playerFrame = CGRectMake(0, iPhoneXSafeTopMargin, kScreenWidth, kScreenHeight - iPhoneXSafeTopMargin - iPhoneXSafeBottomMargin);
  182. _wmPlayer = [[WMPlayer alloc] initWithFrame:_playerFrame];
  183. WMPlayerModel *playModel = [[WMPlayerModel alloc] init];
  184. playModel.videoURL = [NSURL URLWithString:fileUrl];
  185. _wmPlayer.playerModel = playModel;
  186. _wmPlayer.delegate = self;
  187. _bgView = [[UIView alloc] initWithFrame:CGRectMake(0, 0, kScreenWidth, kScreenHeight)];
  188. _bgView.backgroundColor = [UIColor blackColor];
  189. [[UIApplication sharedApplication].keyWindow addSubview:_bgView];
  190. [[UIApplication sharedApplication].keyWindow addSubview:_wmPlayer];
  191. [[UIApplication sharedApplication].keyWindow bringSubviewToFront:_wmPlayer];
  192. [_wmPlayer play];
  193. }
  194. - (void)wmplayer:(WMPlayer *)wmplayer clickedCloseButton:(UIButton *)backBtn {
  195. [wmplayer removePlayer];
  196. [_bgView removeFromSuperview];
  197. [self setNeedsStatusBarAppearanceUpdate];
  198. }
  199. - (void)wmplayer:(WMPlayer *)wmplayer clickedFullScreenButton:(UIButton *)fullScreenBtn {
  200. self.isRatation = !self.isRatation;
  201. if (self.isRatation) {
  202. [wmplayer removeFromSuperview];
  203. [UIView animateWithDuration:1.0f animations:^{
  204. wmplayer.transform = CGAffineTransformMakeRotation(M_PI_2);
  205. } completion:^(BOOL finished) {
  206. wmplayer.frame = CGRectMake(0, iPhoneXSafeTopMargin, kScreenWidth, kScreenHeight - iPhoneXSafeTopMargin - iPhoneXSafeBottomMargin);
  207. [[UIApplication sharedApplication].keyWindow addSubview:wmplayer];
  208. [[UIApplication sharedApplication].keyWindow bringSubviewToFront:wmplayer];
  209. }];
  210. }
  211. else {
  212. [wmplayer removeFromSuperview];
  213. [UIView animateWithDuration:1.0f animations:^{
  214. // 复原
  215. wmplayer.transform = CGAffineTransformIdentity;
  216. } completion:^(BOOL finished) {
  217. wmplayer.frame = CGRectMake(0, iPhoneXSafeTopMargin, kScreenWidth, kScreenHeight - iPhoneXSafeTopMargin - iPhoneXSafeBottomMargin);
  218. [[UIApplication sharedApplication].keyWindow addSubview:wmplayer];
  219. [[UIApplication sharedApplication].keyWindow bringSubviewToFront:wmplayer];
  220. }];
  221. }
  222. }
  223. - (void)didTapUrlInMessageCell:(NSString *)url model:(RCMessageModel *)model {
  224. if ([url containsString:@"dayaedu"]) {
  225. KSBaseWKWebViewController *ctrl = [[KSBaseWKWebViewController alloc] init];
  226. ctrl.url = url;
  227. [self.navigationController pushViewController:ctrl animated:YES];
  228. }
  229. else if ([url containsString:@"daya"] && [url hasSuffix:@".mp4"]) {
  230. [self playVideoWithUrl:url];
  231. }
  232. else {
  233. [super didTapUrlInMessageCell:url model:model];
  234. }
  235. }
  236. - (RCMessageContent *)willSendMessage:(RCMessageContent *)messageContent {
  237. //可以在这里修改将要发送的消息
  238. if ([messageContent isMemberOfClass:[RCTextMessage class]]) {
  239. // RCTextMessage *textMsg = (RCTextMessage *)messageContent;
  240. // textMsg.extra = @"";
  241. }
  242. return messageContent;
  243. }
  244. - (void)resendMessage:(RCMessageContent *)messageContent {
  245. if ([messageContent isKindOfClass:[RCSightMessage class]]) {
  246. RCSightMessage *sightMsg = (RCSightMessage *)messageContent;
  247. [self sendMediaMessage:sightMsg pushContent:nil appUpload:YES];
  248. }
  249. else {
  250. [super resendMessage:messageContent];
  251. }
  252. }
  253. #pragma mark --- refresh
  254. - (void)refreshUserInfoOrGroupInfo {
  255. if (self.conversationType == ConversationType_PRIVATE) {
  256. if (![self.targetId isEqualToString:[RCIM sharedRCIM].currentUserInfo.userId]) {
  257. [KSNetworkingManager imUserFriendQueryDetail:KS_POST userId:self.targetId success:^(NSDictionary * _Nonnull dic) {
  258. if ([dic integerValueForKey:@"code"] == 200 && [dic boolValueForKey:@"status"]) {
  259. NSDictionary *userDic = [dic dictionaryValueForKey:@"data"];
  260. RCUserInfo *user = [[RCUserInfo alloc] initWithUserId:self.targetId name:[userDic stringValueForKey:@"friendNickname"] portrait:[userDic stringValueForKey:@"friendAvatar"]];
  261. [[RCIM sharedRCIM] refreshUserInfoCache:user withUserId:self.targetId];
  262. [self refreshTitle];
  263. }
  264. else {
  265. }
  266. } faliure:^(NSError * _Nonnull error) {
  267. }];
  268. }
  269. }
  270. else if (self.conversationType == ConversationType_GROUP) {
  271. // 获取群成员
  272. [KSNetworkingManager imGroupMemberAllRequest:KS_POST groupId:self.targetId success:^(NSDictionary * _Nonnull dic) {
  273. if ([dic integerValueForKey:@"code"] == 200 && [dic boolValueForKey:@"status"]) {
  274. NSArray *sourceArray = [dic arrayValueForKey:@"data"];
  275. for (NSDictionary *parm in sourceArray) {
  276. if ([parm isKindOfClass:[NSNull class]]) {
  277. continue;
  278. }
  279. GroupMemberModel *model = [[GroupMemberModel alloc] initWithDictionary:parm];
  280. // 刷新缓存
  281. RCUserInfo *user = [[RCUserInfo alloc] initWithUserId:model.userId name:model.nickname portrait:model.avatar];
  282. if (model.isAdmin) {
  283. user.extra = @"owner";
  284. }
  285. [[RCIM sharedRCIM] refreshGroupUserInfoCache:user withUserId:model.userId withGroupId:self.targetId];
  286. }
  287. }
  288. else {
  289. }
  290. } faliure:^(NSError * _Nonnull error) {
  291. }];
  292. // 获取群组信息
  293. [KSNetworkingManager queryGroupDetail:KS_POST groupId:self.targetId success:^(NSDictionary * _Nonnull dic) {
  294. if ([dic integerValueForKey:@"code"] == 200 && [dic boolValueForKey:@"status"]) {
  295. NSDictionary *result = [dic dictionaryValueForKey:@"data"];
  296. RCGroup *groupInfo = [[RCGroup alloc] initWithGroupId:self.targetId groupName:[result stringValueForKey:@"name"] portraitUri:[result stringValueForKey:@"img"]];
  297. [[RCIM sharedRCIM] refreshGroupInfoCache:groupInfo withGroupId:self.targetId];
  298. [self refreshTitle];
  299. }
  300. else if ([dic integerValueForKey:@"code"] == 204) {
  301. // 弹窗提示是否删除群组
  302. MJWeakSelf;
  303. self.alertView = [KSPublicAlertView shareInstanceWithTitle:@"提示" descMessage:@"当前群组已被解散,是否删除聊天记录?" leftTitle:@"取消" rightTitle:@"确定" cancelAction:^{
  304. } sureAction:^{
  305. [weakSelf removeNoExistGroup];
  306. }];
  307. }
  308. } faliure:^(NSError * _Nonnull error) {
  309. }];
  310. }
  311. }
  312. - (void)removeNoExistGroup {
  313. [[RCIMClient sharedRCIMClient] removeConversation:ConversationType_GROUP targetId:self.targetId];
  314. [self.navigationController popToRootViewControllerAnimated:YES];
  315. }
  316. - (void)refreshTitle {
  317. if (self.conversationType == ConversationType_GROUP) {
  318. RCGroup *group = [[RCIM sharedRCIM] getGroupInfoCache:self.targetId];
  319. if (![NSString isEmptyString:group.groupName]) {
  320. self.title = group.groupName;
  321. }
  322. }
  323. else {
  324. RCUserInfo *userInfo = [[RCIM sharedRCIM] getUserInfoCache:self.targetId];
  325. if (![NSString isEmptyString:userInfo.name]) {
  326. self.title = userInfo.name;
  327. }
  328. }
  329. }
  330. - (void)pluginBoardView:(RCPluginBoardView *)pluginBoardView clickedItemWithTag:(NSInteger)tag {
  331. switch (tag) {
  332. case SHARE_MUSIC_TAG: // 曲谱分享
  333. {
  334. }
  335. break;
  336. default:
  337. {
  338. [RCNaviDataManager sharedInstance].naviData.uploadVideoDurationLimit = 480.0f;
  339. [super pluginBoardView:pluginBoardView clickedItemWithTag:tag];
  340. }
  341. break;
  342. }
  343. }
  344. - (void)willDisplayMessageCell:(RCMessageBaseCell *)cell atIndexPath:(NSIndexPath *)indexPath {
  345. if ([cell isKindOfClass:[RCMessageCell class]]) {
  346. UIImageView *imageView = (UIImageView *)((RCMessageCell *)cell).portraitImageView;
  347. imageView.contentMode = UIViewContentModeScaleAspectFill;
  348. imageView.layer.masksToBounds = YES;
  349. if (self.conversationType == ConversationType_GROUP) {
  350. UIView *contentView = (UIView *)((RCMessageCell *)cell).messageContentView;
  351. RCMessageModel *model = self.conversationDataRepository[indexPath.row];
  352. for (UIView *subView in contentView.subviews) {
  353. if ([subView isKindOfClass:[KSGroupTagImageView class]]) {
  354. [subView removeFromSuperview];
  355. break;
  356. }
  357. }
  358. RCUserInfo *user = [[RCIM sharedRCIM] getGroupUserInfoCache:model.senderUserId withGroupId:self.targetId];
  359. if ([user.extra isEqualToString:@"owner"]) {
  360. KSGroupTagImageView *avatarImage = [[KSGroupTagImageView alloc] initWithImage:[UIImage imageNamed:@"group_owner"]];
  361. [contentView addSubview:avatarImage];
  362. [avatarImage mas_makeConstraints:^(MASConstraintMaker *make) {
  363. make.width.mas_equalTo(36);
  364. make.height.mas_equalTo(16);
  365. make.centerX.mas_equalTo(imageView.mas_centerX);
  366. make.centerY.mas_equalTo(imageView.mas_bottom);
  367. }];
  368. }
  369. }
  370. }
  371. }
  372. #pragma mark ----- chatSessionInputBarControl delegate
  373. - (void)sightDidFinishRecord:(NSString *)url thumbnail:(UIImage *)image duration:(NSUInteger)duration {
  374. RCSightMessage *sightMessage = [RCSightMessage messageWithLocalPath:url thumbnail:image duration:duration];
  375. [self sendMediaMessage:sightMessage pushContent:nil appUpload:YES];
  376. }
  377. - (void)imageDataDidSelect:(NSArray *)selectedImages fullImageRequired:(BOOL)full {
  378. [self becomeFirstResponder];
  379. __weak KSChatConversationViewController *weakSelf = self;
  380. dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{
  381. for (int i = 0; i < selectedImages.count; i++) {
  382. @autoreleasepool {
  383. id item = [selectedImages objectAtIndex:i];
  384. if ([item isKindOfClass:NSData.class]) {
  385. NSData *imageData = (NSData *)item;
  386. UIImage *image = [UIImage imageWithData:imageData];
  387. image = [RCKitUtility fixOrientation:image];
  388. // 保留原有逻辑并添加大图缩小的功能
  389. [[KSRCloudMediaManager sharedManager] downsizeImage:image completionBlock:^(UIImage * _Nullable outimage, BOOL doNothing) {
  390. RCImageMessage *imagemsg;
  391. if (doNothing || !outimage) {
  392. imagemsg = [RCImageMessage messageWithImage:image];
  393. imagemsg.full = full;
  394. } else if (outimage) {
  395. NSData *newImageData = UIImageJPEGRepresentation(outimage, 1);
  396. imagemsg = [RCImageMessage messageWithImageData:newImageData];
  397. imagemsg.full = full;
  398. }
  399. [weakSelf sendMessage:imagemsg pushContent:nil];
  400. } progressBlock:^(UIImage * _Nullable outimage, BOOL doNothing) {
  401. }];
  402. }
  403. else if ([item isKindOfClass:NSDictionary.class]) {
  404. NSDictionary *assertInfo = item;
  405. if ([assertInfo objectForKey:@"avAsset"]) {
  406. AVAsset *model = assertInfo[@"avAsset"];
  407. UIImage *image = assertInfo[@"thumbnail"];
  408. NSString *localPath = assertInfo[@"localPath"];
  409. if (![localPath isKindOfClass:[NSString class]]) {
  410. localPath = @"";
  411. }
  412. dispatch_sync(dispatch_get_main_queue(), ^{
  413. NSUInteger duration = round(CMTimeGetSeconds(model.duration));
  414. RCSightMessage *sightMsg =
  415. [RCSightMessage messageWithAsset:model thumbnail:image duration:duration];
  416. sightMsg.localPath = localPath;
  417. [weakSelf sendMediaMessage:sightMsg pushContent:nil appUpload:YES];
  418. });
  419. }
  420. }
  421. [NSThread sleepForTimeInterval:0.5];
  422. }
  423. }
  424. });
  425. }
  426. - (void)uploadMedia:(RCMessage *)message uploadListener:(RCUploadMediaStatusListener *)uploadListener {
  427. // 获取 asset
  428. if ([message.content isKindOfClass:[RCSightMessage class]]) {
  429. RCSightMessage *sightMessage = (RCSightMessage *)message.content;
  430. NSString *fileName = [NSString stringWithFormat:@"sight_%@", [[sightMessage.name componentsSeparatedByString:@"_"] lastObject]];
  431. NSString *savePath = [[self getSightSavePath] stringByAppendingPathComponent:fileName];
  432. // 判断文件是否存在于该文件夹
  433. if ([[NSFileManager defaultManager] fileExistsAtPath:savePath]) {
  434. sightMessage.localPath = savePath;
  435. [self submitFileWithMessage:sightMessage uploadListener:uploadListener];
  436. }
  437. else {
  438. if (([sightMessage.localPath containsString:@"var/mobile/Media/"])) {
  439. NSURL *fileUrl = [NSURL fileURLWithPath:sightMessage.localPath];
  440. AVURLAsset *avAsset = [AVURLAsset URLAssetWithURL:fileUrl options:nil];
  441. [self startExportVideoWithVideoAsset:avAsset presetName:AVAssetExportPreset960x540 savePath:savePath success:^(NSString *outputPath) {
  442. sightMessage.localPath = savePath;
  443. [self submitFileWithMessage:sightMessage uploadListener:uploadListener];
  444. } failure:^(NSString *errorMessage, NSError *error) {
  445. // 导出失败 直接拷贝上传
  446. [self copyFileToPath:savePath uploadListener:uploadListener];
  447. }];
  448. }
  449. else {
  450. // 复制文件到指定文件夹
  451. [self copyFileToPath:savePath uploadListener:uploadListener];
  452. }
  453. }
  454. }
  455. else {
  456. uploadListener.errorBlock(INVALID_PARAMETER);
  457. }
  458. }
  459. - (void)copyFileToPath:(NSString *)savePath uploadListener:(RCUploadMediaStatusListener *)uploadListener {
  460. RCMessage *message = uploadListener.currentMessage;
  461. if ([message.content isKindOfClass:[RCSightMessage class]]) {
  462. RCSightMessage *sightMessage = (RCSightMessage *)message.content;
  463. // 复制文件到指定文件夹
  464. BOOL isSuccess = [self copyMediaFile:sightMessage.localPath toPath:savePath];
  465. if (isSuccess) {
  466. sightMessage.localPath = savePath;
  467. [self submitFileWithMessage:sightMessage uploadListener:uploadListener];
  468. }
  469. else {
  470. uploadListener.errorBlock(INVALID_PARAMETER);
  471. }
  472. }
  473. }
  474. - (void)startExportVideoWithVideoAsset:(AVURLAsset *)videoAsset presetName:(NSString *)presetName savePath:(NSString *)savePath success:(void (^)(NSString *outputPath))success failure:(void (^)(NSString *errorMessage, NSError *error))failure {
  475. NSArray *presets = [AVAssetExportSession exportPresetsCompatibleWithAsset:videoAsset];
  476. if ([presets containsObject:presetName]) {
  477. NSError *error = nil;
  478. AVAssetTrack *assetVideoTrack = nil;
  479. AVAssetTrack *assetAudioTrack = nil;
  480. CMTime start = CMTimeMakeWithSeconds(0, videoAsset.duration.timescale);
  481. CMTimeRange range = CMTimeRangeMake(start, videoAsset.duration);
  482. // Check if the asset contains video and audio tracks
  483. if ([[videoAsset tracksWithMediaType:AVMediaTypeVideo] count] != 0) {
  484. assetVideoTrack = [videoAsset tracksWithMediaType:AVMediaTypeVideo][0];
  485. }
  486. if ([[videoAsset tracksWithMediaType:AVMediaTypeAudio] count] != 0) {
  487. assetAudioTrack = [videoAsset tracksWithMediaType:AVMediaTypeAudio][0];
  488. }
  489. CMTime insertionPoint = kCMTimeZero;
  490. AVMutableComposition *composition = [[AVMutableComposition alloc] init];
  491. // Insert the video and audio tracks from AVAsset
  492. if (assetVideoTrack != nil) {
  493. // 视频通道 工程文件中的轨道,有音频轨、视频轨等,里面可以插入各种对应的素材
  494. AVMutableCompositionTrack *compositionVideoTrack = [composition addMutableTrackWithMediaType:AVMediaTypeVideo preferredTrackID:kCMPersistentTrackID_Invalid];
  495. // 视频方向
  496. [compositionVideoTrack setPreferredTransform:assetVideoTrack.preferredTransform];
  497. // 把视频轨道数据加入到可变轨道中 这部分可以做视频裁剪TimeRange
  498. [compositionVideoTrack insertTimeRange:range ofTrack:assetVideoTrack atTime:insertionPoint error:&error];
  499. }
  500. if (assetAudioTrack != nil) {
  501. AVMutableCompositionTrack *compositionAudioTrack = [composition addMutableTrackWithMediaType:AVMediaTypeAudio preferredTrackID:kCMPersistentTrackID_Invalid];
  502. compositionAudioTrack.preferredTransform = assetAudioTrack.preferredTransform;
  503. [compositionAudioTrack insertTimeRange:range ofTrack:assetAudioTrack atTime:insertionPoint error:&error];
  504. }
  505. AVAssetExportSession *session = [[AVAssetExportSession alloc] initWithAsset:composition presetName:presetName];
  506. // Optimize for network use.
  507. session.shouldOptimizeForNetworkUse = YES;
  508. NSArray *supportedTypeArray = session.supportedFileTypes;
  509. if ([supportedTypeArray containsObject:AVFileTypeMPEG4]) {
  510. session.outputFileType = AVFileTypeMPEG4;
  511. } else if (supportedTypeArray.count == 0) {
  512. if (failure) {
  513. failure(@"该视频类型暂不支持导出", nil);
  514. }
  515. NSLog(@"No supported file types 视频类型暂不支持导出");
  516. return;
  517. } else {
  518. session.outputFileType = [supportedTypeArray objectAtIndex:0];
  519. if (videoAsset.URL && videoAsset.URL.lastPathComponent) {
  520. savePath = [savePath stringByReplacingOccurrencesOfString:@".mp4" withString:[NSString stringWithFormat:@"-%@", videoAsset.URL.lastPathComponent]];
  521. }
  522. }
  523. session.outputURL = [NSURL fileURLWithPath:savePath];
  524. [session exportAsynchronouslyWithCompletionHandler:^{
  525. dispatch_async(dispatch_get_main_queue(), ^{
  526. switch (session.status) {
  527. case AVAssetExportSessionStatusUnknown: {
  528. NSLog(@"AVAssetExportSessionStatusUnknown");
  529. } break;
  530. case AVAssetExportSessionStatusWaiting: {
  531. NSLog(@"AVAssetExportSessionStatusWaiting");
  532. } break;
  533. case AVAssetExportSessionStatusExporting: {
  534. NSLog(@"AVAssetExportSessionStatusExporting");
  535. } break;
  536. case AVAssetExportSessionStatusCompleted: {
  537. NSLog(@"AVAssetExportSessionStatusCompleted");
  538. if (success) {
  539. success(savePath);
  540. }
  541. } break;
  542. case AVAssetExportSessionStatusFailed: {
  543. NSLog(@"AVAssetExportSessionStatusFailed");
  544. if (failure) {
  545. failure(@"视频导出失败", session.error);
  546. }
  547. } break;
  548. case AVAssetExportSessionStatusCancelled: {
  549. NSLog(@"AVAssetExportSessionStatusCancelled");
  550. if (failure) {
  551. failure(@"导出任务已被取消", nil);
  552. }
  553. } break;
  554. default: break;
  555. }
  556. });
  557. }];
  558. }
  559. else {
  560. if (failure) {
  561. NSString *errorMessage = [NSString stringWithFormat:@"当前设备不支持该预设:%@", presetName];
  562. failure(errorMessage, nil);
  563. }
  564. }
  565. }
  566. - (void)submitFileWithMessage:(RCSightMessage *)sightMessage uploadListener:(RCUploadMediaStatusListener *)uploadListener {
  567. NSURL *fileUrl = [NSURL fileURLWithPath:sightMessage.localPath];
  568. NSData *fileData = [NSData dataWithContentsOfURL:fileUrl];
  569. if (fileData.length) {
  570. NSString *suffix = [NSString stringWithFormat:@".%@",[fileUrl pathExtension]];
  571. [[KSUploadManager shareInstance] configBucketName:@"i-m"];
  572. [[KSUploadManager shareInstance] videoUpload:fileData fileName:@"sightVideo" fileSuffix:suffix progress:^(int64_t bytesWritten, int64_t totalBytes) {
  573. int progress = (int)(bytesWritten / totalBytes * 100);
  574. uploadListener.updateBlock(progress);
  575. } successCallback:^(NSMutableArray * _Nonnull fileUrlArray) {
  576. NSString *videoUrl = [fileUrlArray lastObject];
  577. sightMessage.remoteUrl = videoUrl;
  578. sightMessage.size = fileData.length;
  579. uploadListener.successBlock(sightMessage);
  580. } faliure:^(NSError * _Nullable error, NSString * _Nullable descMessaeg) {
  581. uploadListener.errorBlock(ERRORCODE_TIMEOUT);
  582. }];
  583. }
  584. else {
  585. uploadListener.errorBlock(INVALID_PARAMETER);
  586. }
  587. }
  588. - (BOOL)copyMediaFile:(NSString *)sourcePath toPath:(NSString *)toPath
  589. {
  590. BOOL retVal = YES;
  591. if ([[NSFileManager defaultManager] fileExistsAtPath:sourcePath]) { // 如果原文件存在
  592. if (![[NSFileManager defaultManager] fileExistsAtPath:toPath])
  593. {
  594. retVal = [[NSFileManager defaultManager] copyItemAtPath:sourcePath toPath:toPath error:NULL];
  595. }
  596. }
  597. else {
  598. retVal = NO;
  599. }
  600. return retVal;
  601. }
  602. - (NSString *)getSightSavePath {
  603. NSString *componentString = [NSString stringWithFormat:@"RongCloud/%@/RCSightCache", self.targetId];
  604. NSString *path = [[NSSearchPathForDirectoriesInDomains(NSCachesDirectory, NSUserDomainMask, YES)lastObject] stringByAppendingPathComponent:componentString];
  605. NSFileManager *fileManager = [NSFileManager defaultManager];
  606. BOOL isDir = FALSE;
  607. BOOL isDirExist = [fileManager fileExistsAtPath:path isDirectory:&isDir];
  608. if(!(isDirExist && isDir)) {
  609. BOOL bCreateDir = [fileManager createDirectoryAtPath:path withIntermediateDirectories:YES attributes:nil error:nil];
  610. if(!bCreateDir){
  611. NSLog(@"创建文件夹失败!");
  612. }
  613. NSLog(@"创建文件夹成功,文件路径%@",path);
  614. }
  615. NSLog(@"文件路径%@",path);
  616. return path;
  617. }
  618. /*
  619. #pragma mark - Navigation
  620. // In a storyboard-based application, you will often want to do a little preparation before navigation
  621. - (void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender {
  622. // Get the new view controller using [segue destinationViewController].
  623. // Pass the selected object to the new view controller.
  624. }
  625. */
  626. @end