融云聊天頁面長按消息后“翻譯”功能的實現(xiàn)方法

項目要求實現(xiàn)“翻譯”的功能,融云 SDK 本身沒這個功能飘蚯,所以只能曲線救國了析命,通過自定義消息來實現(xiàn),下面是功能實現(xiàn)相關(guān)內(nèi)容工育。

資源鏈接:

官網(wǎng):https://www.rongcloud.cn/
自定義消息文檔:https://docs.rongcloud.cn/v4/views/im/ui/guide/private/conversation/msgsend/ios.html#createcustom

實現(xiàn)思路

  1. 創(chuàng)建自定義 cell虾宇,與 SDK 內(nèi)置的文本消息進行綁定。因為他們內(nèi)置的文本消息 cell 不支持?jǐn)U展顯示翻譯的內(nèi)容如绸,所以需要使用自定義 cell嘱朽。
  2. 在聊天頁面將自定義 cell 與內(nèi)置的文本消息進行綁定。
  3. 重寫長按消息 cell 的方法怔接,判斷如果是文本消息燥翅,增加“翻譯”按鈕。
  4. 點擊“翻譯”按鈕蜕提,對文本內(nèi)容進行翻譯森书,并將翻譯好的內(nèi)容設(shè)置到數(shù)據(jù)源中。
  5. 刷新 UI,會觸發(fā)自定義 cell 中的回調(diào)方法凛膏,在回調(diào)方法中重新設(shè)置高度杨名,并添加 UI,對數(shù)據(jù)源中翻譯好的內(nèi)容進行展示猖毫。

代碼部分

  1. 創(chuàng)建自定義 cell 繼承于 RCTextMessageCell台谍,m 文件中代碼如下,具體效果可自行調(diào)整

    #import "RCDTextMessageCell.h"
    
    #define RCDScreenWidth [UIScreen mainScreen].bounds.size.width
    #define Font_Size 16
    #define Extra_BackgroupView_CornerRadius 6.f
    
    @interface RCDTextMessageCell ()
    
    //翻譯內(nèi)容的 Label
    @property (strong, nonatomic) UILabel *extraLabel;
    
    //翻譯內(nèi)容的背景圖
    @property (strong, nonatomic) UIView *extraBackgroundView;
    
    @end
    
    @implementation RCDTextMessageCell
    
    + (CGSize)sizeForMessageModel:(RCMessageModel *)model
          withCollectionViewWidth:(CGFloat)collectionViewWidth
             referenceExtraHeight:(CGFloat)extraHeight {
        //翻譯好的內(nèi)容
        NSString *extra = model.extra;
        
        CGSize superSize = [super sizeForMessageModel:model withCollectionViewWidth:collectionViewWidth referenceExtraHeight:extraHeight];
        if (extra.length > 0) {
            CGSize extraSize = [RCDTextMessageCell getTextLabelSize:extra];
            CGFloat finalHeight = superSize.height + extraSize.height;
            return CGSizeMake(superSize.width, finalHeight);
        }else {
            return superSize;
        }
    }
    
    //計算文本 size 的方法
    + (CGSize)getTextLabelSize:(NSString *)content {
        if ([content length] > 0) {
            float maxWidth = RCDScreenWidth -
            (10 + [RCIM sharedRCIM].globalMessagePortraitSize.width + 10) * 2 - 5 - 35;
            CGRect textRect = [content
                               boundingRectWithSize:CGSizeMake(maxWidth, 8000)
                               options:(NSStringDrawingTruncatesLastVisibleLine | NSStringDrawingUsesLineFragmentOrigin |
                                        NSStringDrawingUsesFontLeading)
                               attributes:@{NSFontAttributeName : [UIFont systemFontOfSize:Font_Size]}
                               context:nil];
            textRect.size.height = ceilf(textRect.size.height);
            textRect.size.width = ceilf(textRect.size.width);
            return CGSizeMake(textRect.size.width + 5, textRect.size.height + 5);
        } else {
            return CGSizeZero;
        }
    }
    
    - (void)setDataModel:(RCMessageModel *)model {
        [super setDataModel:model];
        NSString *extra = model.extra;
        
        //為了防止復(fù)用問題的處理
        [self.extraLabel removeFromSuperview];
        [self.extraBackgroundView removeFromSuperview];
        
        //如果有翻譯的內(nèi)容吁断,添加 UI趁蕊,進行展示
        if (extra.length > 0) {
            CGSize extraSize = [[self class] getTextLabelSize:extra];
            CGRect superFrame = self.bubbleBackgroundView.frame;
            CGRect extraBackgroundViewFrame = CGRectZero;
            
            //判斷消息方向,設(shè)置翻譯背景圖的 frame
            if (model.messageDirection == MessageDirection_SEND) {
                extraBackgroundViewFrame = CGRectMake(superFrame.origin.x + superFrame.size.width - extraSize.width - 15, superFrame.size.height + 3, extraSize.width + 10, extraSize.height + 6);
            } else {
                extraBackgroundViewFrame = CGRectMake(superFrame.origin.x + 5, superFrame.origin.y + superFrame.size.height + 3, extraSize.width + 10, extraSize.height + 6);
            }
            
            self.extraBackgroundView = [[UIView alloc] initWithFrame:extraBackgroundViewFrame];
            self.extraBackgroundView.backgroundColor = [UIColor whiteColor];
            self.extraBackgroundView.layer.cornerRadius = Extra_BackgroupView_CornerRadius;
            [self.messageContentView addSubview:self.extraBackgroundView];
            
            CGRect extraLabelFrame = CGRectMake(8, 3, extraSize.width, extraSize.height);
            self.extraLabel = [[UILabel alloc] initWithFrame:extraLabelFrame];
            self.extraLabel.font = [UIFont systemFontOfSize:Font_Size];
            self.extraLabel.numberOfLines = 0;
            [self.extraLabel setLineBreakMode:NSLineBreakByWordWrapping];
            [self.extraLabel setTextAlignment:NSTextAlignmentLeft];
            [self.extraBackgroundView addSubview:self.extraLabel];
            self.extraLabel.text = extra;
        }
    }
    
    @end
    
  2. 在聊天頁面類導(dǎo)入并綁定自定義 cell :

    #import "RCDTextMessageCell.h"
    
    - (void)viewDidLoad {
        [super viewDidLoad];    
        [self registerClass:[RCDTextMessageCell class] forMessageClass:[RCTextMessage class]];
    }
    
  3. 在聊天頁面實現(xiàn)長按消息的“翻譯”方法:

    • 創(chuàng)建一個屬性用于暫存長按時候的 model仔役。
    @property (strong, nonatomic) RCMessageModel *longTouchModel;
    
    • 重寫長按消息 cell 的方法掷伙,判斷如果是文本消息,增加“翻譯”按鈕又兵,translate 是實現(xiàn)“翻譯”的方法任柜。

      - (NSArray<UIMenuItem *> *)getLongTouchMessageCellMenuList:(RCMessageModel *)model {
          NSMutableArray<UIMenuItem *> *menuList = [[super getLongTouchMessageCellMenuList:model] mutableCopy];
          if ([model.content isKindOfClass:[RCTextMessage class]]) {
              UIMenuItem *forwardItem = [[UIMenuItem alloc] initWithTitle:@"翻譯"
                                                                   action:@selector(translate)];
              self.longTouchModel = model;
              [menuList addObject:forwardItem];
          }
          return menuList;
      }
      
  4. 實現(xiàn)“翻譯”方法,并將翻譯好的內(nèi)容設(shè)置到數(shù)據(jù)源中:

    • 代碼僅供參考沛厨,翻譯的方法還需要自己來實現(xiàn)宙地,把最終翻譯好的結(jié)果賦值給 self.longTouchModel.extra
    • 這里必須要提的是 cellSize = CGSizeZero,否則刷新 UI 不會改變 cell 的高度
    • 刷新 self.conversationMessageCollectionView
    • 如果翻譯的是最后一條消息逆皮,需要滾動到底部宅粥,否則翻譯的內(nèi)容會被遮擋
    - (void)translate {
        if (self.longTouchModel) {
            NSString *result = @"翻譯后的結(jié)果。";
            RCTextMessage *txtMsg = (RCTextMessage *)self.longTouchModel.content;
            if ([txtMsg.content isEqualToString:@"How are you?"]) {
                result = @"你好嗎电谣?";
            } else if ([txtMsg.content isEqualToString:@"I’m fine."]) {
                result = @"我很好秽梅。";
            }
            self.longTouchModel.extra = result;
            self.longTouchModel.cellSize = CGSizeZero;
            [self.conversationMessageCollectionView reloadData];
            RCMessageModel *model = [self.conversationDataRepository lastObject];
            if (model.messageId == self.longTouchModel.messageId ) {
                [self scrollToBottomAnimated:YES];
            }
        }
    }
    
  5. 刷新 UI 時,顯示翻譯的內(nèi)容辰企,自定義 cell 會回調(diào)設(shè)置 CGSize 的方法风纠,改變 cell 高度况鸣,同時也會回調(diào) setDataModel 方法牢贸,在這個方法中添加翻譯的內(nèi)容,具體可以參考自定義 cell 的 m 文件中代碼镐捧,這里就不重復(fù)貼代碼了潜索。

到此為止,該功能已經(jīng)算是搞定了懂酱,同樣這個思路和處理方法也適用于“語音轉(zhuǎn)文字”竹习,把翻譯功能換成語音轉(zhuǎn)文字,cell 繼承于 RCVoiceMessageCell 就完了列牺,希望這篇分享能幫助到大家整陌。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子泌辫,更是在濱河造成了極大的恐慌随夸,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,496評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件震放,死亡現(xiàn)場離奇詭異宾毒,居然都是意外死亡,警方通過查閱死者的電腦和手機殿遂,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,407評論 3 392
  • 文/潘曉璐 我一進店門诈铛,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人墨礁,你說我怎么就攤上這事幢竹。” “怎么了饵溅?”我有些...
    開封第一講書人閱讀 162,632評論 0 353
  • 文/不壞的土叔 我叫張陵妨退,是天一觀的道長。 經(jīng)常有香客問我蜕企,道長咬荷,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,180評論 1 292
  • 正文 為了忘掉前任轻掩,我火速辦了婚禮幸乒,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘唇牧。我一直安慰自己罕扎,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,198評論 6 388
  • 文/花漫 我一把揭開白布丐重。 她就那樣靜靜地躺著腔召,像睡著了一般。 火紅的嫁衣襯著肌膚如雪扮惦。 梳的紋絲不亂的頭發(fā)上臀蛛,一...
    開封第一講書人閱讀 51,165評論 1 299
  • 那天,我揣著相機與錄音崖蜜,去河邊找鬼浊仆。 笑死,一個胖子當(dāng)著我的面吹牛豫领,可吹牛的內(nèi)容都是我干的抡柿。 我是一名探鬼主播,決...
    沈念sama閱讀 40,052評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼等恐,長吁一口氣:“原來是場噩夢啊……” “哼洲劣!你這毒婦竟也來了备蚓?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,910評論 0 274
  • 序言:老撾萬榮一對情侶失蹤囱稽,失蹤者是張志新(化名)和其女友劉穎星著,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體粗悯,經(jīng)...
    沈念sama閱讀 45,324評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡虚循,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,542評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了样傍。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片横缔。...
    茶點故事閱讀 39,711評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖衫哥,靈堂內(nèi)的尸體忽然破棺而出茎刚,到底是詐尸還是另有隱情,我是刑警寧澤撤逢,帶...
    沈念sama閱讀 35,424評論 5 343
  • 正文 年R本政府宣布膛锭,位于F島的核電站,受9級特大地震影響蚊荣,放射性物質(zhì)發(fā)生泄漏初狰。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,017評論 3 326
  • 文/蒙蒙 一互例、第九天 我趴在偏房一處隱蔽的房頂上張望奢入。 院中可真熱鬧,春花似錦媳叨、人聲如沸腥光。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,668評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽武福。三九已至,卻和暖如春痘番,著一層夾襖步出監(jiān)牢的瞬間捉片,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,823評論 1 269
  • 我被黑心中介騙來泰國打工夫偶, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留界睁,地道東北人觉增。 一個月前我還...
    沈念sama閱讀 47,722評論 2 368
  • 正文 我出身青樓兵拢,卻偏偏與公主長得像,于是被迫代替她去往敵國和親逾礁。 傳聞我的和親對象是個殘疾皇子说铃,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,611評論 2 353