项目下载地址:https://github.com/townkoim/Andorid_Goty_Im_
说明:项目为一个商业项目、有在各大市场上线.里面包含了,第三方登录,分享,推送、即时IM聊天功能
android加入即时聊天的功能.
对比了融云、环信(主要对比了价格,小公司以经济实惠为主),最终还是选择了亲加、
第一步:进入http://www.gotye.com.cn/ 亲加。注册后台并登陆,在注册应用的时候,我这里选择“开放注册”,开放注册的好处就是,客户端可以根据用户的唯一ID,首次登陆亲加后台会自创建一个新用户,而第二次登陆的时候就会使用上一次登陆创建之后的用户名。
注意:一定要是客户端提供的唯一ID
在创建成功之后..看一下应用,应用概况这一栏目,会显示你注册之后的APPKEY等相关信息,在IM用户选项,会显示,你已经注册过的IM的用户
下一步--http://www.gotye.com.cn/download.html,在这里下载即时通讯API的安卓开发包(因为没接入语音通信,所以不需要下载语音通信API)
解压之后,将相对应的Jar文件以及.so库文件放到工程中,想对应的文件夹下
下一步:在adt-eclipse的安卓项目中集成亲加(根据亲加提供的api文档:http://www.gotye.com.cn/docs/ime/android.html)
1、在Mainfest文件中加入以下权限
<uses-permission android:name ="android.permission.INTERNET" />
<uses-permission android:name= "android.permission.WRITE_EXTERNAL_STORAGE" />
<uses-permission android:name ="android.permission.RECORD_AUDIO" />
<uses-permission android:name= "android.permission.ACCESS_NETWORK_STATE" />
<uses-permission android:name ="android.permission.ACCESS_WIFI_STATE" />
<uses-permission android:name= "android.permission.CHANGE_NETWORK_STATE" />
<uses-permission android:name ="android.permission.READ_PHONE_STATE" />
<uses-permission android:name ="android.permission.READ_CONTACTS" />
2、在自定义的application中初始化亲加,就是注册亲加到我的APP当中
//使用您在亲加管理平台申请到的appkey初始化API,appkey如果为空会返回参数错误。
//context为android上下文环境
//下文提到的gotyeApi即为GotyeAPI,后续不再赘述。
GotyeAPI gotyeApi = GotyeAPI. getInstance();
gotyeApi.init(MainActivity. this, "28683ef9-6295-4939-895f-96fbffa81c25" );
注意:在application当中,应该将gotyApi设一个get方法或者用static修饰、以便全局都可以调用,如果需要开启语音识别功能,要添加这段代码--
apiist.initIflySpeechRecognition();
3、登录注册亲加
说明:亲加载进行改版之后,做了一个很大的变动,将所有的回调都放在了 GotyeDelegate 这个类当中,以前都是每个模块都有每个模块的接口(个人感觉,改版之后还没有之前设计的合理,不利于程序的扩展,但是这样又统一进行了封装,利于程序的优化)
3.1 首先要进行注册
注册的接口为:https://qplusapi.gotye.com.cn:8443/api/ImportUsers?email=登录亲加后台的账号&devpwd=登录亲加后台的账号&appkey=48377791-c431-4b0b-8372-26d0b95e7aa9&useraccount=?
PS:这一步应该由服务器来做批量导入用户到亲加,但是我这边服务器没有做,所以只能通过在APP端,用户登录服务器的时候,去亲加注册一个账号,如果返回 {"errcode":200}字段,表示注册成功,如果返回 {"failedUsers":["13798477145"],"errcode":403},表示已经注册过了。
gotyeApi.addListener( delegate);
//因为在注册的时候没有假如密码的参数,所以这里登录的时候密码必须为Null,不能为空字符串,如需要密码,在之前注册的接口的用户名后面加上&&userpwd=密码
gotyeApi.login( "13798477145", null);
GotyeDelegate delegate = new GotyeDelegate(){
public void onLogin( int code, com.gotye.api.GotyeUser user) {
};
};
根据重写Login方法来得到登录的回调,根据code来判断是否登录成功,可以参考api文档中的 GotyeStatusCode 类判断code的错误码
至此--亲加登录就算完成了.登录完成之后还有一部必须来进行实现:
根据login的方法,可以看到返回了一个GotyUser对象, 在这里我们就可以改变这个User的一些属性,比如设置用的头像,设置用户的昵称,性别等
聊天模块
聊天的唯一的身份就是通过此ID。
GotyeImageMessage(图片)、 GotyeRichTextMessage(声音)、 GotyeVoiceMessage(二进制文本)
,用来构成MVC模式中的Model体系、但是在这个版本中--就只有了一个 GotyeMessage 对象了。通过 GotyeMessageType枚举类 ,来判断发送或者接受的是什么类型的消息
讲解一下其中会用到的几个关键的类
GotyeChatTarget 类---聊天对象的父类,聊天肯得要有一个对象,一个用户是一个对象,一个聊天室也是一个对象、一个聊天群组也是一个对象,
所以在该类中有三个对象 GotyeGroup , GotyeRoom , GotyeUser,根据类的名字应该知道 这三个类是三个实体类,对应的 群组、聊天室、用户的实体类,
所以在不同的场合,需要GotyeChatTarget 对象的时候,根据业务需求来new 它的子类即可。
GotyeDelegate类:
所有的回调都在此类当中、所有的发送消息都会回调在 onSendMessage (int code, GotyeMessage message)方法中-在此处根据code的值来判断是否发送成功.
下面来讲解一下发送消息:
发送普通文本消息
//构建GotyeMessage对象,第一个参数为发送者(可以省略),第二个参数为接受者。
GotyeMessage msg = GotyeMessage. createTextMessage( new GotyeUser( "自己的唯一ID" ), new GotyeUser( "对方身份的唯一ID" ), "要发送的文本" );
//发送消息
gotyeApi .sendMessage(msg);
图片消息
创建图片消息:
String imagePath = “/mnt/sdcard /icon.jpg”; ///< 设定原始图片路径,目前仅支持jpg
GotyeMessage message = GotyeMessage.createImageMessage(receiver, imagePath);
自定义数据消息
自定义数据消息 API支持发送用户自定义数据:
byte[] userdata = “1234567890”.getBytes(); ///< 自定义数据内容不能超过4KB
GotyeMessage message = GotyeMessage.createUserDataMessage(receiver,userdata,len);
或者发送指定路径文件里的数据内容:
String path = “/mnt/sdcard /userdata.txt”; ///<文件内容不能超过4KB
GotyeMessage message = GotyeMessage.createUserDataMessage(receiver,path);
获取自定义数据(如果存在): byte[] data = message.getUserData();
语音消息
开始录音:在调用开始录音的时候,会触发GotyeDelegate类的onStartTalk(int code, boolean isrealtime, int targetType, GotyeChatTarget target)的回调方法,在这里我们记下当前的时间
gotyeApi.startTalk(user,WhineMode.Default,maxDuration); ///< 单聊api,user为聊天对象,WhineMode为语音变声参数,maxDuration为时长(单位ms)
gotyeApi.startTalk(group,WhineMode.Default,maxDuration); ///< 群聊语音消息
gotyeApi.startTalk(room,WhineMode.Default,realtime,maxDuration);
///<realtime如果为true,则开始实时语音,聊天室所有用户会收到onRealPlayStart的回调
gotyeApi.stopTalk();
GotyeDelegate类的 onStopTalk (int code, GotyeMessage message, boolean isrealtime)的犯法,在这里,我们可以得到当前时间,减去刚才在onstartTalk中记下的时间、判断时间是否大于1S。(因为我在实际开发中,必须大于1S才允许发送)
然后在这个回调的方法里面。调用:
gotyeApi.sendMessage(message);///< 不调用发送消息方法消息是不会发送给对方的。但是该条语音消息会保存在本地。
OK,下面来看看接受消息:
onReceiveMessage ( GotyeMessage message)方法中,在这里,我们可以通过message.getType(),(通过 枚举 GotyeMessageType来判断 )
得到消息之后,我们就可以放到我们自己的ListView的适配器的数据源里面去了
下面来讲一下我的项目中微信聊天的布局的实现:
我用一个ListView来显示聊天界面的所有消息, 聊天界面的消息 自己发送的消息在界面的右边,别人发送的消息在界面的左边.
于是我通过重写BaseAdapter的getViewTypeCount()方法,让他return 2; 返回两个布局。
以及重写getItemViewType(int position)方法, 在这个方法中,我通过ListView的数据源.getPosition(position) 方法,得到消息的实体类。 也就是GoteMessage对象。
在GotyeMessage对象中调用getSender()方法,得到这个用户,和我自己刚登陆亲加的时候的用户信息进行匹配,如果相同,说明是我自己是发送者,那么久return 右边的布局,反之...
接着就来讲一下语音,图片,文本的布局..这个比较简单。通过 GotyeMessage对象的 GotyeMessageType判断, 不同的类型加载不同的数据类型即可。
如果接受的消息是语音或者图片:
接收/下载 消息
下载消息:
gotyeApi.downloadMediaInMessage(message)///< 只有图片消息和语音消息需要下载
注意: 新收到的消息如果是图片或语音类型,图片和语音文件并不随消息一起到达接收方,需要手动调用该方法下载。
回调到GotyeDelegate 中的onDownloadMessage
void onDownloadMediaInMessage(int code, GotyeMessage message);
回调原型: void onDownloadMediaInMessage(int code, GotyeMessage message); /// <下载的消息,下载好的媒体文件在message.media.path或pathEx中