基于openfire+smack开发Android即时聊天应用[四]-单人聊天、群聊、发送接收文件等

这篇文章主要介绍如何实现点对点单人聊天、多人的群聊、以及如何给对方发送文件,如何发送图片消息和语音消息等功能。

1.单人聊天

1.首先创建聊天对象

/**
  * 创建聊天窗口
  * @param jid   好友的JID
  * @return
  */
 public Chat createChat(String jid) {
    if(isConnected()) {
        ChatManager chatManager = ChatManager.getInstanceFor(connection);
        return chatManager.createChat(jid);
    }
    throw new NullPointerException("服务器连接失败,请先连接服务器");
 }

创建聊天对象时,参数JID记得传聊天JID(解释请参考我的系列文章之基于openfire+smack开发Android即时聊天应用[三]-账号信息、添加好友、JID理解等)给好友发送文本消息

chat.sendMessage(message);

2.获取聊天对象管理器

/**
  * 获取聊天对象管理器
  * @return
  */
 public ChatManager getChatManager() {
    if(isConnected()) {
        ChatManager chatManager = ChatManager.getInstanceFor(connection);
        return chatManager;
    }
    throw new NullPointerException("服务器连接失败,请先连接服务器");
 }

3.接收文本消息

//创建聊天对象管理器监听
private ChatManagerListener chatManagerListener = new ChatManagerListener() {
    @Override
    public void chatCreated(Chat chat, boolean createdLocally) {
        chat.addMessageListener(new ChatMessageListener() {
        @Override
        public void processMessage(Chat chat, Message message) {
            //接收到消息Message之后进行消息展示处理,这个地方可以处理所有人的消息
        }
    });
    }
};
//设置聊天对象管理器处理监听
getChatManager().addChatListener(chatManagerListener);

上述代码会在你创建聊天对象时对该聊天对象设置消息处理监听,当接收到消息之后,会自动调用processMessage方法进行处理,我们可以在该方法中对接收到的消息进行展示或其他处理,所有好友发送过来的消息都会通过该方法处理。所以该监听最好在登陆之后进行设置,同时在断开连接或是注销时移除该监听。

2.群聊

1.创建群聊聊天室

/**
    * 创建群聊聊天室
    * @param roomName       聊天室名字
    * @param nickName       创建者在聊天室中的昵称
    * @param password       聊天室密码
    * @return
    */
   public MultiUserChat createChatRoom(String roomName, String nickName, String password) {
    if(!isConnected()) {
    throw new NullPointerException("服务器连接失败,请先连接服务器");
    }
    MultiUserChat muc = null;  
try {  
    // 创建一个MultiUserChat  
    muc = MultiUserChatManager.getInstanceFor(connection).getMultiUserChat(roomName + "@conference." + connection.getServiceName());
    // 创建聊天室  
    boolean isCreated = muc.createOrJoin(nickName);
    if(isCreated) {
        // 获得聊天室的配置表单  
        Form form = muc.getConfigurationForm();  
        // 根据原始表单创建一个要提交的新表单。  
        Form submitForm = form.createAnswerForm();
        // 向要提交的表单添加默认答复  
        List fields = form.getFields();
        for(int i = 0; fields != null && i < fields.size(); i++) {
            if(FormField.Type.hidden != fields.get(i).getType() &&
                                          fields.get(i).getVariable() != null) {  
            // 设置默认值作为答复  
            submitForm.setDefaultAnswer(fields.get(i).getVariable());  
            }  
        }
        // 设置聊天室的新拥有者  
        List owners = new ArrayList();  
        owners.add(connection.getUser());// 用户JID  
        submitForm.setAnswer("muc#roomconfig_roomowners", owners);  
        // 设置聊天室是持久聊天室,即将要被保存下来  
        submitForm.setAnswer("muc#roomconfig_persistentroom", true);  
        // 房间仅对成员开放  
        submitForm.setAnswer("muc#roomconfig_membersonly", false);  
        // 允许占有者邀请其他人  
        submitForm.setAnswer("muc#roomconfig_allowinvites", true);  
        if(password != null && password.length() != 0) {  
        // 进入是否需要密码  
        submitForm.setAnswer("muc#roomconfig_passwordprotectedroom",  true);  
        // 设置进入密码  
        submitForm.setAnswer("muc#roomconfig_roomsecret", password);  
     }  
     // 能够发现占有者真实 JID 的角色  
     // submitForm.setAnswer("muc#roomconfig_whois", "anyone");  
    // 登录房间对话  
    submitForm.setAnswer("muc#roomconfig_enablelogging", true);  
    // 仅允许注册的昵称登录  
    submitForm.setAnswer("x-muc#roomconfig_reservednick", true);  
    // 允许使用者修改昵称  
    submitForm.setAnswer("x-muc#roomconfig_canchangenick", false);  
    // 允许用户注册房间  
    submitForm.setAnswer("x-muc#roomconfig_registration", false);  
    // 发送已完成的表单(有默认值)到服务器来配置聊天室  
    muc.sendConfigurationForm(submitForm);  
    }
} catch (XMPPException | SmackException e) {  
    e.printStackTrace();  
    return null;  
}  
       return muc; 
   }

上面这段创建群聊聊天室设置表单属性的那段代码引用于网上的代码段。
2.加入群聊聊天室

/**
  * 加入一个群聊聊天室
  * @param roomName     聊天室名字
  * @param nickName     用户在聊天室中的昵称
  * @param password     聊天室密码
  * @return
  */
 public MultiUserChat joinChatRoom(String roomName,  String nickName, String password) {  
    if(!isConnected()) {
  throw new NullPointerException("服务器连接失败,请先连接服务器");
    }  
     try {
         // 使用XMPPConnection创建一个MultiUserChat窗口  
         MultiUserChat muc = MultiUserChatManager.getInstanceFor(connection).
                          getMultiUserChat(roomName + "@conference." + connection.getServiceName());  
         // 聊天室服务将会决定要接受的历史记录数量  
         DiscussionHistory history = new DiscussionHistory();  
         history.setMaxChars(0);  
         // history.setSince(new Date());  
         // 用户加入聊天室  
         muc.join(nickName, password);  
         return muc;  
     } catch (XMPPException | SmackException e) {  
         e.printStackTrace();  
         return null;  
     }  
 }

在实现加入群聊聊天室的这段代码中有这么一段代码:

getMultiUserChat(roomName + "@conference." + connection.getServiceName());

在@与ServiceName中间必须加上conference这个字符串,我也不知道为什么,我最开始时不知道没有加,然后无论如何都加入失败,后来在网上查资料查了半天,有人说是要加上这个,然后我加上就成功了,暂时没搞明白为什么,先把程序跑通会用了再研究其他的原因。

3.群聊发送消息
当你创建或是加入群聊聊天室后,即可获得群聊对象MultiUserChat,通过该对象即可发送群聊消息:

multiUserChat.sendMessage(msg);//发送群聊消息

4.接收群聊消息

//聊天室消息监听
private MessageListener messageListener = new MessageListener() {
    @Override
    public void processMessage(Message message) {
        //与单聊接收处理消息类似,聊天室里所有人(包括发送人自己)发送的消息都会通过此方法进行回调处理
    }
};
//设置聊天室消息监听
multiUserChat.addMessageListener(messageListener);

群聊接收消息与单聊接收消息还是很像的,只是监听对象,监听方式稍稍有点区别,整个来说,消息接收还是很简单的。

3.文件传输

1.获取文件传输对象

/**
  * 获取发送文件的发送器
  * @param jid  一个完整的jid(如:laohu@192.168.0.108/Smack
  *                         后面的Smack应该客户端类型,不加这个会出错)
  * @return
  */
 public OutgoingFileTransfer getSendFileTransfer(String jid) {
    if(isConnected()) {
  return FileTransferManager.getInstanceFor(connection).createOutgoingFileTransfer(jid);
    }
    throw new NullPointerException("服务器连接失败,请先连接服务器");
 }

获取文件传输对象时的参数JID记得为文件传输JID:解释请参考我的系列文章之基于openfire+smack开发Android即时聊天应用[三]-账号信息、添加好友、JID理解等
2.发送文件

//获取文件传输对象
OutgoingFileTransfer transfer = getSendFileTransfer(jid);
//发送文件
transfer.sendFile(File file, String description);
//此处执行文件发送状态监听
以上代码为发送文件file,参数description为对这次文件传输的描述

3.文件传输(包括文件发送与接收)过程监听(传输开始、完成、进度百分比)

//文件传输过程中的状态监听分析
if(transfer.getProgress() < 1) {//开始传输
//传输进度,值为0~1
}

while(!transfer.isDone()) {//判断传输是否完成,传输取消、传输完成、传输发生错误都会返回true
    try {
        Thread.sleep(200);
    } catch (InterruptedException e) {
    e.printStackTrace();
   }
}

if(FileTransfer.Status.complete.equals(transfer.getStatus())) {
    //传输完成  
} else if(FileTransfer.Status.cancelled.equals(transfer.getStatus())) {
    //传输取消
} else if(FileTransfer.Status.error.equals(transfer.getStatus())) {
    //传输错误
} else if(FileTransfer.Status.refused.equals(transfer.getStatus())) {
    //传输拒绝
}

以上代码需在子线程执行,可以在文件传输(发送、接收)开始时设置进度条,传输完成时去掉进度条,同时可以通过getProgress()方法获得文件传输的具体进度百分比。
4.接收文件

/**
     * 添加文件接收的监听
     * @param fileTransferListener
     */
    public void addFileTransferListener(FileTransferListener fileTransferListener) {
        if(isConnected()) {
        FileTransferManager.getInstanceFor(connection).addFileTransferListener(fileTransferListener);
        return;
        }
        throw new NullPointerException("服务器连接失败,请先连接服务器");
    }
addFileTransferListener(new FileTransferListener() {
    @Override
    public void fileTransferRequest(FileTransferRequest request) {
        // Accept it
    IncomingFileTransfer transfer = request.accept();
    try {
        String description = request.getDescription();
        //在目录fileDir目录下新建一个名字为request.getFileName()的文件
            File file = new File(fileDir ,request.getFileName());
            //开始接收文件(将传输过来的文件内容输出到file中)
        transfer.recieveFile(file);
        //此处执行文件传输监听
    } catch (SmackException | IOException e) {
        e.printStackTrace();
    }
    }
});

上面代码为设置文件接收监听

4.发送语音、图片消息

我查看了半天的Smack的API,但是没有找到直接发送语音、图片消息的API,我说说我的实现思路。

  • 其实图片、语音都是文件,我们可以把它们当做文件发送给好友。

  • 在发送文件的同时,用描述字段进行标记传输过来的是图片还是语音。

  • 然后在接收到该文件后通过描述字段进行区分当前接收的是图片文件还是语音文件,然后进行区分展示即可,这样就可以达到发送图片消息和语音消息。

  • 但是我的这种实现方式还是有问题的,因为这种方式对于单聊还是可以实现的。但是如果是群聊的话,我就必须给每个人都发一个相同的文件,这样的话一条语音或图片消息,其实是要发送N次的,对于发送人来说流量就多消耗了N-1倍,所以这种方式对于实现群聊是行不通的。

对于群聊发送语音和图片消息,我的思路是这样的:

  • 自己写一个上传文件的服务。
  • 发送语音或图片消息时,将图片或语音通过上述上传服务上传到服务器上。
  • 在上传完语音或图片后,再向聊天室里发送一个文本消息,发送内容为文件的类似下载地址这样的信息,同时还要告诉群成员这个文件是图片还是语音。
  • 群成员接收到这样的特殊文本消息后去自动下载这个文件然后进行展示或是其他处理。
  • 群聊天里发送图片或语音消息的这个实现方式我没有验证,但我觉得应该是可行的。至于单聊发送语音或图片消息的思路我是实现验证成功了的,是可行的。
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容

  • 点击查看原文 Web SDK 开发手册 SDK 概述 网易云信 SDK 为 Web 应用提供一个完善的 IM 系统...
    layjoy阅读 13,711评论 0 15
  • https://github.com/coderMyy/MYCoreTextLabel 图文混排 , 实现图片文...
    saman0阅读 2,934评论 1 19
  • 一、iOS 直播聊天室 Demo 说明 1、源码结构 2、AppDelegate 在 AppDelegate 中初...
    ajiao焦阅读 2,829评论 2 7
  • 一、简历准备 1、个人技能 (1)自定义控件、UI设计、常用动画特效 自定义控件 ①为什么要自定义控件? Andr...
    lucas777阅读 5,190评论 2 54
  • 要学习基于XMPP协议的IM开发,首先要熟悉XMPP协议本身。 XMPP协议的组成主要的XMPP 协议范本及当今应...
    RichieQ阅读 1,892评论 0 6