【融云总结】融云即时通讯的一些坑和解决方案

前言

融云即时通讯接入比较快,但是开发过程中会遇到各种各样的坑:

  • UserInfo问题
  • 切换登录后会话列表数据错乱问题
  • 聊天界面消息字体颜色问题

UserInfo问题

融云开发文档中提到,融云服务器不会帮我们缓存用户信息(一般指头像链接、昵称、用户id),如果要实现用户信息的及时刷新与获取,可以用2种方式:
1.在消息中携带用户信息
优点是省事方便,缺点是消息体会稍大一丢丢。

2.使用用户信息提供者,说白了就是在需要刷新用户信息的时候,调后台接口或者从App本地缓存中提供给融云SDK来刷新。
优点是消息体不会变大,缺点是如果是对方给你发消息,那么很抱歉你只能调用后台接口去刷新用户信息(前提是你之前没缓存过发送者的用户信息)。

显而易见,使用第1种方式会好一点,问题也会少一些。我们需要做的是:

  • 在和别人聊天前先持久化对方的用户信息(你都能用他的id来进行聊天了,相信头像链接和昵称应该都有吧)
  • 开启消息中携带用户信息
    RongIM.getInstance().setMessageAttachedUserInfo(true);
  • 在收到对方消息的时候,及时刷新对方用户信息
//监听消息接收事件
RongIM.setOnReceiveMessageListener((message, i) -> {
                UserInfo userInfo = message.getContent().getUserInfo();
                if (userInfo == null)
                {
                    LogUtils.getInstance().e("消息中的UserInfo为空......");
                    return false;
                }
                RongIM.getInstance().refreshUserInfoCache(userInfo);
                //返回false则依旧走融云默认的通知和铃声
                return false;
            });
  • 在适当时机刷新用户信息
    比如登录融云后设置自己的用户信息
    RongIM.getInstance().setCurrentUserInfo(userInfo);
    并且批量刷新本地持久化的融云用户信息(我写了个方法,批量刷新之前缓存过的用户信息,在进入应用首页和登录后会调用)

切换登录后会话列表数据错乱问题

当你是用融云官方会话列表组件(ConversationListFragment)来承载会话列表数据,并且没做任何处理,那么切换账户后肯定会出现数据错乱问题,比如会话列表居然残留上个用户的聊天记录,这是因为ConversationListFragment的数据没被刷新所致,我们切换用户后,必须刷新ConversationListFragment。

LogUtils.getInstance().e("尝试刷新消息Fg的会话列表");
            Uri uri = Uri.parse("rong://" + 你的包名).buildUpon()
                    .appendPath("conversationlist")
//这2行配置不需要一样,只需要和你刚初始化ConversationListFragment时的配置一样即可
                    .appendQueryParameter(Conversation.ConversationType.PRIVATE.getName(), "false") 
                    .appendQueryParameter(Conversation.ConversationType.DISCUSSION.getName(), "true")//讨论组
                    .build();
//ConversationListFragment的实例
            conversationListFragment.setUri(uri);

聊天界面消息字体颜色问题

融云默认的文本消息的颜色,不论对方还是自己的都是黑色(#262626),是不支持直接分开设置字体颜色的,要修改的话只能使用自定义的TextMessageItemProvider(小提示:文本消息是一种消息类型,控制其UI布局的是TextMessageItemProvider),然后在初始化融云SDK的位置加上:

        //注册自定义文本消息提供者(因为TextMessage这种消息类型已经被融云注册了,所以此处只需要重新注册一下Provider就好,不需要像自定义消息一样注册XxxMessage和Provider)
        RongIM.getInstance().registerMessageTemplate(new MyTextMessageProvider());

接着是自定义的TextMessageItemProvider,其实就是继承自官方的TextMessageItemProvider,并把里面的所有方法复制进去(怕出问题和方便快捷),然后修改关键方法 newView()bindView()
整个自定义的TextMessageItemProvider代码如下:

/**
 * 作者:HK
 * 日期:2019-04-19
 * 描述:自定义文本消息提供者(为了实现双方文本消息颜色不一样和更换气泡)
 */

@ProviderTag( messageContent = TextMessage.class , showReadState = true)
public class MyTextMessageProvider extends TextMessageItemProvider {

    public MyTextMessageProvider() {
    }

    public View newView(Context context, ViewGroup group) {
        View view = LayoutInflater.from(context).inflate(R.layout.my_rc_item_text_message, (ViewGroup)null);
        ViewHolder holder = new ViewHolder();
        holder.message = view.findViewById(R.id.tv_text1);
        view.setTag(holder);
        return view;
    }

    public Spannable getContentSummary(TextMessage data) {
        return null;
    }

    public Spannable getContentSummary(Context context, TextMessage data) {
        if (data == null) {
            return null;
        } else {
            String content = data.getContent();
            if (content != null) {
                if (content.length() > 100) {
                    content = content.substring(0, 100);
                }

                return new SpannableString(AndroidEmoji.ensure(content));
            } else {
                return null;
            }
        }
    }

    public void onItemClick(View view, int position, TextMessage content, UIMessage message) {
    }

    public void bindView(final View v, int position, TextMessage content, final UIMessage data) {
        ViewHolder holder = (ViewHolder)v.getTag();
        if (data.getMessageDirection() == Message.MessageDirection.SEND) {
            holder.message.setBackgroundResource(R.mipmap.my);
            //布局中message这个TextView设置了一个selector,selected设置成true和false会切换到相应字体颜色
            holder.message.setSelected(true);
        } else {
            holder.message.setBackgroundResource(R.mipmap.others);
            holder.message.setSelected(false);
        }

        final AutoLinkTextView textView = holder.message;
        if (data.getTextMessageContent() != null) {
            int len = data.getTextMessageContent().length();
            if (v.getHandler() != null && len > 500) {
                v.getHandler().postDelayed(new Runnable() {
                    public void run() {
                        textView.setText(data.getTextMessageContent());
                    }
                }, 50L);
            } else {
                textView.setText(data.getTextMessageContent());
            }
        }

        holder.message.setMovementMethod(new LinkTextViewMovementMethod(new ILinkClickListener() {
            public boolean onLinkClick(String link) {
                RongIM.ConversationBehaviorListener listener = RongContext.getInstance().getConversationBehaviorListener();
                RongIM.ConversationClickListener clickListener = RongContext.getInstance().getConversationClickListener();
                boolean result = false;
                if (listener != null) {
                    result = listener.onMessageLinkClick(v.getContext(), link);
                } else if (clickListener != null) {
                    result = clickListener.onMessageLinkClick(v.getContext(), link, data.getMessage());
                }

                if (listener == null && clickListener == null || !result) {
                    String str = link.toLowerCase();
                    if (str.startsWith("http") || str.startsWith("https"))
                    {
                        Context context = (v == null?null:v.getContext());
                        if (context != null)
                        {
                            context.startActivity(new Intent(context, MostWebViewActivity.class)
                                    .addFlags(Intent.FLAG_ACTIVITY_NEW_TASK)
                                    .putExtra(SPkey.url, link)
                            );
//                        Intent intent = new Intent("io.rong.imkit.intent.action.webview");
//                        intent.setPackage(v.getContext().getPackageName());
//                        intent.putExtra("url", link);
//                        v.getContext().startActivity(intent);
                        }
                        result = true;
                    }
                }
                return result;
            }
        }));
        textView.stripUnderlines();
    }

    private static class ViewHolder {
        AutoLinkTextView message;
        boolean longClick;

        private ViewHolder() {
        }
    }
}

附录:融云自定义帮助:

布局(IMKit下):
私聊布局:rc_fr_conversation.xml
输入框布局:rc_ext_extension_bar.xml
输入框-语音输入布局:rc_ext_voice_input.xml
输入框-文字输入布局:rc_ext_input_edit_text.xml
扩展栏布局:rc_ext_plugin_pager.xml
官方会话列表布局:rc_fr_conversationlist.xml

布局或者代码里引用的固定值:rc_ext_dimens.xml

加号(更多)控件:<io.rong.imkit.RongExtension

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