基于XMPP的聊天实现2 - iOS

2016.7.27补充小功能

小菜单

效果图
先看看项目结构图
结构图
本文主要分三部分
  • 数据存储 (coredata)
  • 聊天内容搭建与计算
  • 聊天底部tabbar工具栏
1.数据存储

数据存储这块也是图一个方便,直接用的xmpp框架内部数据存储coredata,所以也就仅仅用到那几个字段,大概的实现了主体部分。

而正由于这个原因,在进入聊天界面,加载聊天历史记录的时候,界面UI有些许卡顿,加载的有些慢,影响用户体验。

自己实现coredata的话,聊天控制器中查询聊天历史记录那部分,直接把放在异步线程去操作,保持主线程更新数据就行了。(以后有时间就把这块代码加在项目中)

2.聊天内容搭建与计算

内容这个也没多少说的,有个小属性可以注意下

  _tableView.keyboardDismissMode = UIScrollViewKeyboardDismissModeOnDrag;

效果就是这样子


drag

主要就说下HBChatModel

这个就是聊天内容的数据模型。里面就有设置文本、图片、语音、地图等一些HBBaseTableViewCell的frame。有了这个,就直接拿到HBChatViewController控制器设置对应的参数即可,不需要再去做额外多余的计算了。

项目中实际用到这块的话,这块最好计算一次后,就将HBBaseTableViewCell高度缓存到caredata中。第二次就直接取,不用费劲计算了。Demo中每次实例化HBChatModel对象,都需要计算。


- (void)setMessage:(XMPPMessageArchiving_Message_CoreDataObject *)message
{
    _message = message;
    
    CGSize getChatBgSize;

    if ([message.messageStr isEqualToString:HBTypeText]){//聊天文字内容

        //聊天,表情字符串
        self.chatContent = [message.body HB_StringToChatAttributeString];
        
        CGSize size = [HBHelp HB_attributeBoundsSize:CGSizeMake(HBChatBgMaxWidth,MAXFLOAT) attributeContentText:[self.chatContent mutableCopy]];
//        //1.文字内容大小
        self.textSize = size;
        
        getChatBgSize = CGSizeMake(size.width + 2 * padding, size.height);
    
    }else if ([message.messageStr isEqualToString:HBTypeImage]){//图片
            
        self.imageSize = CGSizeMake(120, 60);
        
        getChatBgSize = CGSizeMake(self.imageSize.width + 2 * padding, self.imageSize.height + 2 * padding);
        
    }else if ([message.messageStr isEqualToString:HBTypeVoice]){//语音
        
    }else if ([message.messageStr isEqualToString:HBTypeMap]){//地图
        
    }
    
    //2.聊天背景大小
    self.chatBgSize = getChatBgSize;
    //3.行高
    self.cellHeight = self.chatBgSize.height + HBUserIconImageToTop + HBUserIconImageWH + HBChatBgToUserIconImage + HBUserIconImageToTop;;
    
    
}

就一个 setter方法。

带表情聊天内容转换成文字内容

- (NSString *)HB_ChatAttributeStringToString
{
    __block NSMutableString *chatStr = [NSMutableString string];
    [self enumerateAttributesInRange:NSMakeRange(0, self.length)
                                                     options:NSAttributedStringEnumerationLongestEffectiveRangeNotRequired
                                                  usingBlock:^(NSDictionary<NSString *,id> * _Nonnull attrs, NSRange range, BOOL * _Nonnull stop) {
                                                      
              HBTextAttachment *hbAttachment = [attrs objectForKey:@"NSAttachment"];
              if (hbAttachment) {
                  
                  [chatStr appendString:[NSString stringWithFormat:@"[/%@]",hbAttachment.emjoysName]];
                  
              }else{
                  NSAttributedString *aStr = [self attributedSubstringFromRange:range];
                  
                  [chatStr appendString:aStr.string];
              }
    }];
    return [chatStr copy];
}

文字内容转换成带表情的内容

- (NSAttributedString *)HB_StringToChatAttributeString
{
    static dispatch_once_t onceToken;
    static NSRegularExpression *regularExpression;
    dispatch_once(&onceToken, ^{
        regularExpression = [NSRegularExpression regularExpressionWithPattern:@"\\[[a-zA-Z0-9\\u4e00-\\u9fa5/]+\\]"
                                                                      options:NSRegularExpressionCaseInsensitive
                                                                        error:nil];
    });
   
    NSMutableAttributedString *AttributeString = [[NSMutableAttributedString alloc] initWithString:self];
    
    [AttributeString setAttributes:@{NSFontAttributeName : chatTextFont}
                             range:NSMakeRange(0, AttributeString.length)];

    NSArray *array = [regularExpression matchesInString:self options:NSMatchingReportCompletion range:NSMakeRange(0, self.length)];
    
    
    [array enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(NSTextCheckingResult *result, NSUInteger idx, BOOL * _Nonnull stop) {
        
        NSString *emjoyStr = [self substringWithRange:result.range];//[/001]
        
        NSRange range = [emjoyStr rangeOfString:@"[/"];
        NSUInteger loc = range.location + range.length;
        NSUInteger len = [emjoyStr rangeOfString:@"]"].location;
        
        NSString *emjoyName = [emjoyStr substringWithRange:NSMakeRange(loc, len - loc)];//001
        
        HBTextAttachment *textAtt = [HBTextAttachment new];
        textAtt.emjoysName = emjoyName;
        UIImage *image = [UIImage imageNamed:textAtt.emjoysName];
        textAtt.image = image;
        
        NSAttributedString *imageAttribute = [NSAttributedString attributedStringWithAttachment:textAtt];
        [AttributeString replaceCharactersInRange:result.range withAttributedString:imageAttribute];
        
    }];
    return AttributeString;
    
}

3.聊天底部tabbar工具栏

HBChatView
这个写的也就是最纠结的地方。按钮的各种逻辑,状态,调了好久。体力活可真是体力活

各种监听

#pragma mark - Custom
- (void)keyboardNotifacation{
   [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyBoardWillShow:) name:UIKeyboardWillShowNotification object:nil];
   [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(keyBoardWillHide:) name:UIKeyboardWillHideNotification object:nil];
   
   //5.监听文本变化
   [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(textViewTextChange:) name:UITextViewTextDidChangeNotification object:self.textView];
   
   [self.textView addObserver:self forKeyPath:@"attributedText" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:NULL];
   [self.textView addObserver:self forKeyPath:@"inputView" options:NSKeyValueObservingOptionNew | NSKeyValueObservingOptionOld context:NULL];
   //6.监听自身frame
   [self addObserver:self forKeyPath:@"frame" options:NSKeyValueObservingOptionNew context:nil];
   
}
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary<NSString *,id> *)change context:(void *)context{
    if ([keyPath isEqualToString:@"frame"]) {
        
        if ([self.delegate respondsToSelector:@selector(chatViewDidChangeFrame:)]) {
            [self.delegate chatViewDidChangeFrame:self];
        }
        _newFrame = [[change objectForKey:@"new"] CGRectValue];
        
    }else if ([keyPath isEqualToString:@"attributedText"]){
        
        [self caclulaterTextViewHeight];
        
    }else if ([keyPath isEqualToString:@"inputView"]) {
    
        HBEmjoyView *emjoyView = [change objectForKey:@"new"];

        [UIView animateWithDuration:0.25 animations:^{
                
            CGFloat moveHeight = ((NSNull *)emjoyView == [NSNull null]) ? 258 : emjoyView.HB_H;
            
            self.HB_Y = [UIScreen mainScreen].bounds.size.height - moveHeight - self.HB_H;
        }];
        
    }else{
        
        [super observeValueForKeyPath:keyPath ofObject:object change:change context:context];
        
    }
}

多纠结,你自己看图说话😂😂


纠结图

计算 textView高度

- (void)caclulaterTextViewHeight{
    
    NSString *context = [self.textView.attributedText HB_ChatAttributeStringToString];
    
    if (![context hasSuffix:@"\n"] && context.length > 0) {

//        CGFloat getSysHeight = [self.textView sizeThatFits:CGSizeMake(self.textView.HB_W, MAXFLOAT)].height;
        CGFloat getHeight = [HBHelp HB_attributeBoundsSize:CGSizeMake(self.textView.HB_W, MAXFLOAT)
                  attributeContentText:[[context HB_StringToChatAttributeString] mutableCopy]].height;

        if (getHeight <= _originaTextViewH) {
            
            self.textView.HB_H = _originaTextViewH;
            
        }else{
            //1.限制最大高度
            if (getHeight >= _originaButtomViewH * 2) {
                getHeight = _originaButtomViewH * 2;
            }
            self.textView.HB_H = getHeight;
        }
        
        self.HB_H = self.textView.HB_H + self.textView.HB_Y * 2;
        self.HB_Y -= self.HB_H - _lastH;
        //2.始终更新最后的状态
        [self layoutIfNeeded];
        
        _lastButtomFrame = self.frame;
        _lastTextViewFrame = self.textView.frame;
        
    }
    
}
结语

这个项目也就提供大概的思路,但是大概的方向都是有了的。
后期有时间,会继续完善项目中细节的。

如果这个文章帮到了你,一定给我Star哦!

GitHub 欢迎围观! ··```

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,084评论 6 503
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,623评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,450评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,322评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,370评论 6 390
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,274评论 1 300
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,126评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,980评论 0 275
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,414评论 1 313
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,599评论 3 334
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,773评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,470评论 5 344
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,080评论 3 327
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,713评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,852评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,865评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,689评论 2 354

推荐阅读更多精彩内容