Android 环信跳坑爬坑之路

众所周知。。环信巨坑哪~~~~不过最近找了条相对简单的捷径!!在这里简单做下小记,以便下次查阅。。

引入EaseUI

EaseUI作为一个模块引入到项目中,引入之后需要注意其build.gradle中依赖的第三方框架(比如glide)的版本是否与项目其他模块版本一致,需要统一版本。

加入辅助工具类

com.hyphenate.easeui下新建HxHelper辅助类,具体作用如后文。内容如下

public class HxHelper {
    //扩展消息-昵称
    public static final String MSG_EXT_NICKNAME = "hx_nickname";
    //扩展消息-头像
    public static final String MSG_EXT_AVATAR = "hx_avatar";

    private volatile static HxHelper instance;

    public Application app;
    public Opts mOpts;

    //所有的会话集合
    private Map<String, EMConversation> mConvMap;

    private HxHelper() {
        if (null != instance)
            throw new IllegalStateException("Can not instantiate singleton class.");
    }

    /**
     * 初始化
     *
     * @param application Application
     * @param opts        配置项
     */
    public void init(Application application, Opts opts) {
        app = application;
        mOpts = opts;
    }

    /**
     * 单例模式
     *
     * @return 单例实例
     */
    public static HxHelper getInstance() {
        if (null == instance) {
            synchronized (HxHelper.class) {
                if (null == instance) {
                    instance = new HxHelper();
                }
            }
        }
        return instance;
    }

    public EaseUser getUser(String username) {
        EaseUser user = new EaseUser(username);

        //获取到所有会话
        mConvMap = EMClient.getInstance().chatManager().getAllConversations();

        if (null != mConvMap) {
            List<EMMessage> msgList = null;
            for (Map.Entry<String, EMConversation> et : mConvMap.entrySet()) {
                msgList = et.getValue().getAllMessages();
                //遍历消息列表,从消息扩展中获取昵称和头像
                if (null != msgList && !msgList.isEmpty()) {
                    for (EMMessage msg : msgList) {
                        if (!TextUtils.equals(username, msg.getFrom())) {
                            //如果该条消息不是该用户的,就遍历下一条
                            continue;
                        }
                        //设置昵称和用户名
                        try {
                            user.setNickname(msg.getStringAttribute(MSG_EXT_NICKNAME));
                            user.setAvatar(msg.getStringAttribute(MSG_EXT_AVATAR));
                        } catch (Exception e) {
                            e.printStackTrace();
                        }
                    }
                }
            }
        }

        return user;
    }

    /**
     * 配置项
     */
    public static class Opts {
        public boolean showChatTitle;
    }
}

改变头像控件形状

若要将头像改为圆形或者圆角,就需要在ApplicationonCreate()方法中添加如下代码

//设置头像为圆形
EaseAvatarOptions avatarOpts = new EaseAvatarOptions();
//0:默认,1:圆形,2:矩形
avatarOpts.setAvatarShape(1);
//设置倒角
//avatarOpts.setAvatarRadius(radius);
EaseUI.getInstance().setAvatarOptions(avatarOpts);

设置聊天界面

直接使用EaseChatFragment,具体代码如下

//new出EaseChatFragment或其子类的实例
 EaseChatFragment chatFragment = new EaseChatFragment();
 //传入参数
 Bundle args = new Bundle();
 args.putInt(EaseConstant.EXTRA_CHAT_TYPE, EaseConstant.CHATTYPE_GROUP);
 args.putString(EaseConstant.EXTRA_USER_ID, "username");
 chatFragment.setArguments(args);
 getSupportFragmentManager().beginTransaction().add(R.id.container, chatFragment).commit();

其中R.id.container为显示该界面的容器,代码如下即可

<FrameLayout
    android:id="@+id/fl_chat_content"
    android:layout_width="match_parent"
    android:layout_height="match_parent"/>

隐藏自带的标题栏

多数情况下,我们都有自己的标题栏,而引入的EaseChatFragment已经自带了个毁三观的标题栏,此时就需要将它隐藏。虽然EaseChatFragment提供了隐藏和显示标题栏的方法,对这种直接使用EaseChatFragment的方法不适用(原因下边分析),此时前面的HxHelper就派上用场了。。

chatFragment.hideTitleBar();//无效,具体原因后面分析
  • 第一步,需要在ApplicationonCreate()方法中添加如下代码
//设置有关环信自定义的相关配置
HxHelper.Opts opts = new HxHelper.Opts();
opts.showChatTitle = false;
HxHelper.getInstance().init(this, opts);
  • 第二步,打开EaseChatFragment,找到setUpView()方法,在其末尾添加如下代码
//自定义代码,设置标题栏的显示与隐藏
if (HxHelper.getInstance().mOpts.showChatTitle) {
    showTitleBar();
} else {
    hideTitleBar();
}

到这里,头像形状就自定义成功了

  • 前面说的调用EaseChatFragmentshowTitleBar()hideTitleBar()无效,主要原因如下
//注意这里的titleBar,该控件是在EaseChatFragment父类(EaseBaseFragment)中的onActivityCreated()方法中才获取到的,所以提交了事务之后就调用hideTitleBar()的话,titleBar是null的,所以设置无效。

//hideTitleBar()方法
public void hideTitleBar() {
    if (titleBar != null) {
        titleBar.setVisibility(View.GONE);
    }
}

//获取到titleBar的地方(EaseBaseFragment中)
@Override
public void onActivityCreated(Bundle savedInstanceState) {
    ...
    titleBar = (EaseTitleBar) getView().findViewById(R.id.title_bar);
    ...
}

设置头像和昵称之类信息

最简单的方法是通过扩展消息发送这些字段,然后通过EaseUI.getInstance().setUserProfileProvider(EaseUserProfileProvider userProvider)方法设置用户信息提供者,再在其getUser(String username)方法中做手脚,不过这就需要借助其提供的接口EaseChatFragment.EaseChatFragmentHelper来实现发送消息前带上扩展消息。具体代码如下

  • 第一步,设置扩展消息辅助器
//定义辅助类实现EaseChatFragment.EaseChatFragmentHelper的onSetMessageAttributes(EMMessage message)方法
private class ChatHelper implements EaseChatFragment.EaseChatFragmentHelper {
    @Override
    public void onSetMessageAttributes(EMMessage message) {
        User user = App.getInstance().getUserClone();
        if (null == user) {
            return;
        }
        //设置自己的头像和昵称到消息扩展中
        message.setAttribute(HxHelper.MSG_EXT_NICKNAME, user.memberNickname);
        message.setAttribute(HxHelper.MSG_EXT_AVATAR, user.memberAvatar);
    }
}

//然后调用EaseChatFragment的setChatFragmentHelper(EaseChatFragment.EaseChatFragmentHelper helper)方法,设置辅助器即可
  • 第二步,在ApplicationonCreate()方法中设置EaseUserProfileProvider ,具体代码如下
EaseUI.getInstance().setUserProfileProvider(username -> {
    User user = getUserClone();
    //如果是当前用户,就设置自己的昵称和头像
    if (null != user && TextUtils.equals(user.hxUsername, username)) {
        EaseUser eu = new EaseUser(username);
        eu.setNickname(user.memberNickname);
        eu.setAvatar(user.memberAvatar);
        return eu;
    }
    //否则交给HxHelper处理,从消息中获取昵称和头像
    return HxHelper.getInstance().getUser(username);
});

到此,昵称和头像就能完美显示了~~

Android 7.0+ 拍照权限异常

众所周知,Android 7.0+拍照和图片裁剪要用FileProvider,但一年过去了,环信貌似并不想做这方面的兼容(TNND)。不过修改起来也不容易,在EaseChatFragmentselectPicFromCamera()方法中做修改。

  • 第一步,先在AndroidManifest.xmlapplication中注册FileProvider,这里就直接在EaseUI模块中做修改,在其AndroidManifest.xml
<application
    android:allowBackup="true"
    android:hardwareAccelerated="true"
    android:supportsRtl="true">
    <!--百度地图页面-->
    <activity android:name="com.hyphenate.easeui.ui.EaseBaiduMapActivity"/>
    <!--显示大图页面-->
    <activity android:name="com.hyphenate.easeui.ui.EaseShowBigImageActivity"/>
    <provider
        android:name=".EsFileProvider"
        android:authorities="com.hyphenate.easeui.provider"
        android:exported="false"
        android:grantUriPermissions="true">
        <meta-data
            android:name="android.support.FILE_PROVIDER_PATHS"
            android:resource="@xml/file_path"/>
    </provider>
</application>
  • 第二步,再在res目录下新建xml文件夹,再新建file_path.xml,其内容为
<paths>
    <external-path
        name="external_files"
        path="."/>
</paths>
  • 第三步,在EaseChatFragmentselectPicFromCamera()方法中找到如下代码
startActivityForResult(new Intent(MediaStore.ACTION_IMAGE_CAPTURE).putExtra(MediaStore.EXTRA_OUTPUT, Uri.fromFile(cameraFile)), REQUEST_CODE_CAMERA);
  • 第四步, 将其修改为如下即可
//为了兼容Android 7.0+,使用FileProvider
Uri fileUri = null;
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.N) {
    //Android 7.0以下
    fileUri = Uri.fromFile(cameraFile);
} else {
    //Android 7.0及以上
    fileUri = EsFileProvider.getUriForFile(getActivity(), "com.hyphenate.easeui.provider", cameraFile);
}
startActivityForResult(new Intent(MediaStore.ACTION_IMAGE_CAPTURE).putExtra(MediaStore.EXTRA_OUTPUT, fileUri), REQUEST_CODE_CAMERA);

到此,你已经成功跳过环信的巨坑,也爬出了其不少小坑~~

  • 1
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
public class DemoHXSDKHelper extends HXSDKHelper{ private static final String TAG = "DemoHXSDKHelper"; /** * EMEventListener */ protected EMEventListener eventListener = null; /** * contact list in cache */ private Map<String, User> contactList; /** * robot list in cache */ private Map<String, RobotUser> robotList; private CallReceiver callReceiver; /** * 用来记录foreground Activity */ private List<Activity> activityList = new ArrayList<Activity>(); public void pushActivity(Activity activity){ if(!activityList.contains(activity)){ activityList.add(0,activity); } } public void popActivity(Activity activity){ activityList.remove(activity); } @Override protected void initHXOptions(){ super.initHXOptions(); // you can also get EMChatOptions to set related SDK options EMChatOptions options = EMChatManager.getInstance().getChatOptions(); options.allowChatroomOwnerLeave(getModel().isChatroomOwnerLeaveAllowed()); } @Override protected void initListener(){ super.initListener(); IntentFilter callFilter = new IntentFilter(EMChatManager.getInstance().getIncomingCallBroadcastAction()); if(callReceiver == null){ callReceiver = new CallReceiver(); } //注册通话广播接收者 appContext.registerReceiver(callReceiver, callFilter); //注册消息事件监听 initEventListener(); } /** * 全局事件监听 * 因为可能会有UI页面先处理到这个消息,所以一般如果UI页面已经处理,这里就不需要再次处理 * activityList.size() <= 0 意味着所有页面都已经在后台运行,或者已经离开Activity Stack */ protected void initEventListener() { eventListener = new EMEventListener() { private BroadcastReceiver broadCastReceiver = null; @Override public void onEvent(EMNotifierEvent event) { EMMessage message = null; if(event.getData() instanceof EMMessage){ message = (EMMessage)event.getData(); EMLog.d(TAG, "receive the event : " + event.getEvent() + ",id : " + message.getMsgId()); } switch (event.getEvent()) { case EventNewMessage: //应用在后台,不需要刷新UI,通知栏提示新消息 if(activityList.size() <= 0){ HXSDKHelper.getInstance().getNotifier().onNewMsg(message); } break; case EventOfflineMessage: if(activityList.size() <= 0){ EMLog.d(TAG, "received offline messages"); List<EMMessage> messages = (List<EMMessage>) event.getData(); HXSDKHelper.getInstance().getNotifier().onNewMesg(messages); } break; // below is just giving a example to show a cmd toast, the app should not follow this // so be careful of this case EventNewCMDMessage: { EMLog.d(TAG, "收到透传消息"); //获取消息body CmdMessageBody cmdMsgBody = (CmdMessageBody) message.getBody(); final String action = cmdMsgBody.action;//获取自定义action //获取扩展属性 此处省略 //message.getStringAttribute(""); EMLog.d(TAG, String.format("透传消息:action:%s,message:%s", action,message.toString())); final String str = appContext.getString(R.string.receive_the_passthrough); final String CMD_TOAST_BROADCAST = "easemob.demo.cmd.toast"; IntentFilter cmdFilter = new IntentFilter(CMD_TOAST_BROADCAST); if(broadCastReceiver == null){ broadCastReceiver = new BroadcastReceiver(){ @Override public void onReceive(Context context, Intent intent) { Toast.makeText(appContext, intent.getStringExtra("cmd_value"), Toast.LENGTH_SHORT).show(); } }; //注册广播接收者 appContext.registerReceiver(broadCastReceiver,cmdFilter); } Intent broadcastIntent = new Intent(CMD_TOAST_BROADCAST); broadcastIntent.putExtra("cmd_value", str+action); appContext.sendBroadcast(broadcastIntent, null); break; } case EventDeliveryAck: message.setDelivered(true); break; case EventReadAck: message.setAcked(true); break; // add other events in case you are interested in default: break; } } }; EMChatManager.getInstance().registerEventListener(eventListener); EMChatManager.getInstance().addChatRoomChangeListener(new EMChatRoomChangeListener(){ private final static String ROOM_CHANGE_BROADCAST = "easemob.demo.chatroom.changeevent.toast"; private final IntentFilter filter = new IntentFilter(ROOM_CHANGE_BROADCAST); private boolean registered = false; private void showToast(String value){ if(!registered){ //注册广播接收者 appContext.registerReceiver(new BroadcastReceiver(){ @Override public void onReceive(Context context, Intent intent) { Toast.makeText(appContext, intent.getStringExtra("value"), Toast.LENGTH_SHORT).show(); } }, filter); registered = true; } Intent broadcastIntent = new Intent(ROOM_CHANGE_BROADCAST); broadcastIntent.putExtra("value", value); appContext.sendBroadcast(broadcastIntent, null); } @Override public void onChatRoomDestroyed(String roomId, String roomName) { showToast(" room : " + roomId + " with room name : " + roomName + " was destroyed"); Log.i("info","onChatRoomDestroyed="+roomName); } @Override public void onMemberJoined(String roomId, String participant) { showToast("member : " + participant + " join the room : " + roomId); Log.i("info", "onmemberjoined="+participant); } @Override public void onMemberExited(String roomId, String roomName, String participant) { showToast("member : " + participant + " leave the room : " + roomId + " room name : " + roomName); Log.i("info", "onMemberExited="+participant); } @Override public void onMemberKicked(String roomId, String roomName, String participant) { showToast("member : " + participant + " was kicked from the room : " + roomId + " room name : " + roomName); Log.i("info", "onMemberKicked="+participant); } }); } /** * 自定义通知栏提示内容 * @return */ @Override protected HXNotificationInfoProvider getNotificationListener() { //可以覆盖默认的设置 return new HXNotificationInfoProvider() { @Override public String getTitle(EMMessage message) { //修改标题,这里使用默认 return null; } @Override public int getSmallIcon(EMMessage message) { //设置小图标,这里为默认 return 0; } @Override public String getDisplayedText(EMMessage message) { // 设置状态栏的消息提示,可以根据message的类型做相应提示 String ticker = CommonUtils.getMessageDigest(message, appContext); if(message.getType() == Type.TXT){ ticker = ticker.replaceAll("\\[.{2,3}\\]", "[表情]"); } Map<String,RobotUser> robotMap=((DemoHXSDKHelper)HXSDKHelper.getInstance()).getRobotList(); if(robotMap!=null&&robotMap.containsKey(message.getFrom())){ String nick = robotMap.get(message.getFrom()).getNick(); if(!TextUtils.isEmpty(nick)){ return nick + ": " + ticker; }else{ return message.getFrom() + ": " + ticker; } }else{ return message.getFrom() + ": " + ticker; } } @Override public String getLatestText(EMMessage message, int fromUsersNum, int messageNum) { return null; // return fromUsersNum + "个基友,发来了" + messageNum + "条消息"; } @Override public Intent getLaunchIntent(EMMessage message) { //设置点击通知栏转事件 Intent intent = new Intent(appContext, ChatActivity.class); //有电话时优先转到通话页面 if(isVideoCalling){ intent = new Intent(appContext, VideoCallActivity.class); }else if(isVoiceCalling){ intent = new Intent(appContext, VoiceCallActivity.class); }else{ ChatType chatType = message.getChatType(); if (chatType == ChatType.Chat) { // 单聊信息 intent.putExtra("userId", message.getFrom()); intent.putExtra("chatType", ChatActivity.CHATTYPE_SINGLE); } else { // 群聊信息 // message.getTo()为群聊id intent.putExtra("groupId", message.getTo()); if(chatType == ChatType.GroupChat){ intent.putExtra("chatType", ChatActivity.CHATTYPE_GROUP); }else{ intent.putExtra("chatType", ChatActivity.CHATTYPE_CHATROOM); } } } return intent; } }; } @Override protected void onConnectionConflict(){ Intent intent = new Intent(appContext, MainActivity.class); intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); intent.putExtra("conflict", true); appContext.startActivity(intent); } @Override protected void onCurrentAccountRemoved(){ Intent intent = new Intent(appContext, MainActivity.class); intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); intent.putExtra(Constant.ACCOUNT_REMOVED, true); appContext.startActivity(intent); } @Override protected HXSDKModel createModel() { return new DemoHXSDKModel(appContext); } @Override public HXNotifier createNotifier(){ return new HXNotifier(){ public synchronized void onNewMsg(final EMMessage message) { if(EMChatManager.getInstance().isSlientMessage(message)){ return; } String chatUsename = null; List<String> notNotifyIds = null; // 获取设置的不提示新消息的用户或者群组ids if (message.getChatType() == ChatType.Chat) { chatUsename = message.getFrom(); notNotifyIds = ((DemoHXSDKModel) hxModel).getDisabledGroups(); } else { chatUsename = message.getTo(); notNotifyIds = ((DemoHXSDKModel) hxModel).getDisabledIds(); } if (notNotifyIds == null || !notNotifyIds.contains(chatUsename)) { // 判断app是否在后台 if (!EasyUtils.isAppRunningForeground(appContext)) { EMLog.d(TAG, "app is running in backgroud"); sendNotification(message, false); } else { sendNotification(message, true); } viberateAndPlayTone(message); } } }; } /** * get demo HX SDK Model */ public DemoHXSDKModel getModel(){ return (DemoHXSDKModel) hxModel; } /** * 获取内存中好友user list * * @return */ public Map<String, User> getContactList() { if (getHXId() != null && contactList == null) { contactList = ((DemoHXSDKModel) getModel()).getContactList(); }

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值