美文网首页三方
【融云总结】融云即时通讯的一些坑和解决方案

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

作者: Lucky丶夏日 | 来源:发表于2019-05-06 15:55 被阅读188次

    前言

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

    • 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

    相关文章

      网友评论

        本文标题:【融云总结】融云即时通讯的一些坑和解决方案

        本文链接:https://www.haomeiwen.com/subject/iqcdoqtx.html