为了方便程序调用,我们把XMPP的一些主要方法写在AppDelegate中

在AppDelegate.m下这几个方法为:

  1. -(void)setupStream{
  2. //初始化XMPPStream
  3. xmppStream = [[XMPPStream alloc] init];
  4. [xmppStream addDelegate:self delegateQueue:dispatch_get_current_queue()];
  5. }
  6. -(void)goOnline{
  7. //发送在线状态
  8. XMPPPresence *presence = [XMPPPresence presence];
  9. [[self xmppStream] sendElement:presence];
  10. }
  11. -(void)goOffline{
  12. //发送下线状态
  13. XMPPPresence *presence = [XMPPPresence presenceWithType:@"unavailable"];
  14. [[self xmppStream] sendElement:presence];
  15. }
  16. -(BOOL)connect{
  17. [self setupStream];
  18. //从本地取得用户名,密码和服务器地址
  19. NSUserDefaults *defaults = [NSUserDefaults standardUserDefaults];
  20. NSString *userId = [defaults stringForKey:USERID];
  21. NSString *pass = [defaults stringForKey:PASS];
  22. NSString *server = [defaults stringForKey:SERVER];
  23. if (![xmppStream isDisconnected]) {
  24. return YES;
  25. }
  26. if (userId == nil || pass == nil) {
  27. return NO;
  28. }
  29. //设置用户
  30. [xmppStream setMyJID:[XMPPJID jidWithString:userId]];
  31. //设置服务器
  32. [xmppStream setHostName:server];
  33. //密码
  34. password = pass;
  35. //连接服务器
  36. NSError *error = nil;
  37. if (![xmppStream connect:&error]) {
  38. NSLog(@"cant connect %@", server);
  39. return NO;
  40. }
  41. return YES;
  42. }
  43. -(void)disconnect{
  44. [self goOffline];
  45. [xmppStream disconnect];
  46. }

这几个是基础方法,接下来就是XMPPStreamDelegate中的方法,也是接受好友状态,接受消息的重要方法

  1. //连接服务器
  2. - (void)xmppStreamDidConnect:(XMPPStream *)sender{
  3. isOpen = YES;
  4. NSError *error = nil;
  5. //验证密码
  6. [[self xmppStream] authenticateWithPassword:password error:&error];
  7. }
  8. //验证通过
  9. - (void)xmppStreamDidAuthenticate:(XMPPStream *)sender{
  10. [self goOnline];
  11. }
  12. //收到消息
  13. - (void)xmppStream:(XMPPStream *)sender didReceiveMessage:(XMPPMessage *)message{
  14. //    NSLog(@"message = %@", message);
  15. NSString *msg = [[message elementForName:@"body"] stringValue];
  16. NSString *from = [[message attributeForName:@"from"] stringValue];
  17. NSMutableDictionary *dict = [NSMutableDictionary dictionary];
  18. [dict setObject:msg forKey:@"msg"];
  19. [dict setObject:from forKey:@"sender"];
  20. //消息委托(这个后面讲)
  21. [messageDelegate newMessageReceived:dict];
  22. }
  23. //收到好友状态
  24. - (void)xmppStream:(XMPPStream *)sender didReceivePresence:(XMPPPresence *)presence{
  25. //    NSLog(@"presence = %@", presence);
  26. //取得好友状态
  27. NSString *presenceType = [presence type]; //online/offline
  28. //当前用户
  29. NSString *userId = [[sender myJID] user];
  30. //在线用户
  31. NSString *presenceFromUser = [[presence from] user];
  32. if (![presenceFromUser isEqualToString:userId]) {
  33. //在线状态
  34. if ([presenceType isEqualToString:@"available"]) {
  35. //用户列表委托(后面讲)
  36. [chatDelegate newBuddyOnline:[NSString stringWithFormat:@"%@@%@", presenceFromUser, @"nqc1338a"]];
  37. }else if ([presenceType isEqualToString:@"unavailable"]) {
  38. //用户列表委托(后面讲)
  39. [chatDelegate buddyWentOffline:[NSString stringWithFormat:@"%@@%@", presenceFromUser, @"nqc1338a"]];
  40. }
  41. }
  42. }

这里面有两个委托方法,一个是用户列表委托,还有一个就是消息委托,用户列表委托主要就是取得在线用户,更新用户TableView,消息委托就是取得好友发送的消息,并更新消息TableView,当然这两个TableView是在不同的Controller中的

定义完两个委托,我们就要在不同的Controller中实现这两个委托了

在好友Controller中实现<KKChatDelegate>并写入如下方法

  1. //取得当前程序的委托
  2. -(KKAppDelegate *)appDelegate{
  3. return (KKAppDelegate *)[[UIApplication sharedApplication] delegate];
  4. }
  5. //取得当前的XMPPStream
  6. -(XMPPStream *)xmppStream{
  7. return [[self appDelegate] xmppStream];
  8. }
  9. //在线好友
  10. -(void)newBuddyOnline:(NSString *)buddyName{
  11. if (![onlineUsers containsObject:buddyName]) {
  12. [onlineUsers addObject:buddyName];
  13. [self.tView reloadData];
  14. }
  15. }
  16. //好友下线
  17. -(void)buddyWentOffline:(NSString *)buddyName{
  18. [onlineUsers removeObject:buddyName];
  19. [self.tView reloadData];
  20. }

在viewDidLoad中加入

  1. //设定在线用户委托
  2. KKAppDelegate *del = [self appDelegate];
  3. del.chatDelegate = self;

这两行代码,让好友列表的委托实现方法在本程序中

在viewWillAppear中加入

  1. [super viewWillAppear:animated];
  2. NSString *login = [[NSUserDefaults standardUserDefaults] objectForKey:@"userId"];
  3. if (login) {
  4. if ([[self appDelegate] connect]) {
  5. NSLog(@"show buddy list");
  6. }
  7. }else {
  8. //设定用户
  9. [self Account:self];
  10. }

判断本地保存的数据中是否有userId,没有的话就跳转到登录页面

这里最重要的就是connect了,这一句话就是登录了,成功的话,页面就会显示好友列表了。

  1. #pragma mark UITableViewDelegate
  2. -(void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath{
  3. //start a Chat
  4. chatUserName = (NSString *)[onlineUsers objectAtIndex:indexPath.row];
  5. [self performSegueWithIdentifier:@"chat" sender:self];
  6. }
  7. -(void)prepareForSegue:(UIStoryboardSegue *)segue sender:(id)sender{
  8. if ([segue.identifier isEqualToString:@"chat"]) {
  9. KKChatController *chatController = segue.destinationViewController;
  10. chatController.chatWithUser = chatUserName;
  11. }
  12. }

当显示出好友列表,我们选择一个好友进行聊天

将当前好友名称发送给聊天页面

下面是聊天Controller了

在KKChatController.h中加入

  1. NSMutableArray *messages;

这是我们要显示的消息,每一条消息为一条字典

接下来就是每一条消息的显示了

  1. -(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
  2. static NSString *identifier = @"msgCell";
  3. UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:identifier];
  4. if (cell == nil) {
  5. cell = [[UITableViewCell alloc] initWithStyle:UITableViewCellStyleSubtitle reuseIdentifier:identifier];
  6. }
  7. NSMutableDictionary *dict = [messages objectAtIndex:indexPath.row];
  8. cell.textLabel.text = [dict objectForKey:@"msg"];
  9. cell.detailTextLabel.text = [dict objectForKey:@"sender"];
  10. cell.accessoryType = UITableViewCellAccessoryNone;
  11. return cell;
  12. }

跟上面好友Controller一样,这里我们也需要XMPPStream

  1. -(KKAppDelegate *)appDelegate{
  2. return (KKAppDelegate *)[[UIApplication sharedApplication] delegate];
  3. }
  4. -(XMPPStream *)xmppStream{
  5. return [[self appDelegate] xmppStream];
  6. }

在ViewDidLoad中加入

  1. KKAppDelegate *del = [self appDelegate];
  2. del.messageDelegate = self;

设定消息委托由自己来接收和处理

  1. #pragma mark KKMessageDelegate
  2. -(void)newMessageReceived:(NSDictionary *)messageCotent{
  3. [messages addObject:messageCotent];
  4. [self.tView reloadData];
  5. }

接下来最重要的就是发送消息了

  1. - (IBAction)sendButton:(id)sender {
  2. //本地输入框中的信息
  3. NSString *message = self.messageTextField.text;
  4. if (message.length > 0) {
  5. //XMPPFramework主要是通过KissXML来生成XML文件
  6. //生成<body>文档
  7. NSXMLElement *body = [NSXMLElement elementWithName:@"body"];
  8. [body setStringValue:message];
  9. //生成XML消息文档
  10. NSXMLElement *mes = [NSXMLElement elementWithName:@"message"];
  11. //消息类型
  12. [mes addAttributeWithName:@"type" stringValue:@"chat"];
  13. //发送给谁
  14. [mes addAttributeWithName:@"to" stringValue:chatWithUser];
  15. //由谁发送
  16. [mes addAttributeWithName:@"from" stringValue:[[NSUserDefaults standardUserDefaults] stringForKey:USERID]];
  17. //组合
  18. [mes addChild:body];
  19. //发送消息
  20. [[self xmppStream] sendElement:mes];
  21. self.messageTextField.text = @"";
  22. [self.messageTextField resignFirstResponder];
  23. NSMutableDictionary *dictionary = [NSMutableDictionary dictionary];
  24. [dictionary setObject:message forKey:@"msg"];
  25. [dictionary setObject:@"you" forKey:@"sender"];
  26. [messages addObject:dictionary];
  27. //重新刷新tableView
  28. [self.tView reloadData];
  29. }
  30. }

ios xmpp demo的更多相关文章

  1. iOS XMPP 通信协议实现 图形化直观感受

    第一次随笔,实在不知写点什么有用的东西,那就分享一下本人最近的研究所得吧! 是关于iOS-XMPP-通信协议的实现,具体代码比较复杂,三言两句也实在难表达清楚,网上已有很多关于iOS XMPP协议的讲 ...

  2. ios xmpp开发应用后台模式接收聊天信息处理方案

    ios xmpp开发应用后台模式接收聊天信息 最近在使用xmppframwork来实现一个聊天应用,碰到了一个问题,应用进入后台以后,就接收不到消息了: 怎么样才能使应用被切到后台时,应用中的网络连接 ...

  3. 91平台iOS接入demo

    源码:http://pan.baidu.com/s/1DuBl6 今天整理硬盘,找到了一个有趣的demo.一年前,91助手游戏联运呈爆棚趋势,但是许多使用FlashAir开发的优秀的游戏和应用都卡在了 ...

  4. iOS xmpp协议实现聊天之openfire的服务端配置(一)

    今天弄这个openfire服务端的配置直接苦了一逼,只是好在最后最终配置好了.首先感谢@月光的尽头的博客给了我莫大的帮助. 切入正题,首先说一下iOS xmpp协议实现聊天openfireserver ...

  5. ios xmpp 发送语音图片解决方案

    ios xmpp 发送语音,图片解决方案,有需要的朋友可以参考下. 目前做IM多是用的xmpp. 因为项目需求需要实现语音和图片的发送. 发送语音图片有三种方法. 1,xmpp smack.文件传输方 ...

  6. XMPP协议实现即时通讯底层书写 (二)-- IOS XMPPFramework Demo+分析

    我希望,This is a new day! 在看代码之前,我认为你还是应该先整理一下心情,来听我说几句: 首先,我希望你是在早上边看这篇blog,然后一边開始动手操作,假设你仅仅是看blog而不去自 ...

  7. iOS XMPP Framework 中文概述

    本篇文章翻译XMPP Framework中的Overview of the XMPP Framework部分 介绍 The framework is divided into 2 parts: 1. ...

  8. iOS - XMPP 的使用

    1.XMPP XMPP 是一个基于 Socket 通信的即时通讯的协议,它规范了即时通信在网络上数据的传输格式,比如登录,获取好友列表等等的格式.XMPP 在网络传输的数据是 XML 格式. 开发架构 ...

  9. iOS - XMPP Openfire 服务器的搭建

    前言 提前下载好相关软件,且安装目录最好安装在全英文路径下.如果路径有中文名,那么可能会出现一些莫名其妙的问题. 提前准备好的软件: jdk-8u91-macosx-x64.dmg mysql-5.7 ...

随机推荐

  1. P4827 [国家集训队] Crash 的文明世界(第二类斯特林数+树形dp)

    传送门 对于点\(u\),所求为\[\sum_{i=1}^ndis(i,u)^k\] 把后面那堆东西化成第二类斯特林数,有\[\sum_{i=1}^n\sum_{j=0}^kS(k,j)\times ...

  2. 洛谷P3723 [AH2017/HNOI2017]礼物(FFT)

    传送门 首先,两个数同时增加自然数值相当于只有其中一个数增加(此增加量可以小于0) 我们令$x$为当前的增加量,${a},{b}$分别为旋转后的两个数列,那么$$ans=\sum_{i=1}^n(a_ ...

  3. [Xcode 实际操作]八、网络与多线程-(9)使用异步Get方式获取网页源码

    目录:[Swift]Xcode实际操作 本文将演示如何通过Get请求方式,异步获取网页源码. 异步请求与同步请求相比,不会阻塞程序的主线程,而会建立一个新的线程. 在项目导航区,打开视图控制器的代码文 ...

  4. IT兄弟连 JavaWeb教程 Cookie和Session应用结合使用

    一般对于不要求安全的非敏感数据,建议存储在Cookie中! 对于敏感的数据,占用空间较小的,建议存储在Session中! 对于敏感的,较大的数据,存数据库!

  5. chmod 详解

    http://man.linuxde.net/chmod chmod u+x,g+w f01 //为文件f01设置自己可以执行,组员可以写入的权限 chmod u=rwx,g=rw,o=r f01 c ...

  6. Bootstrap Table 从新InsertRow 刷新表格 数据的问题处理方案

    1.第一步获取数据源 var rows = { //要插入的数据,这里要和table列名一致SkuCode: data.rows[i].SkuCode,BarCode: data.rows[i].Ba ...

  7. 一篇文章彻底弄懂CAS实现SSO单点登录原理

    1. CAS 简介 1.1. What is CAS ? CAS ( Central Authentication Service ) 是 Yale 大学发起的一个企业级的.开源的项目,旨在为 Web ...

  8. hdu 6299 Balanced Sequence( 2018 Multi-University Training Contest 1 )

    #include <stdio.h> #include <iostream> #include <cstdlib> #include <cmath> # ...

  9. AVL树(自平衡二叉查找树)

    了解AVL树之前要先了解二叉查找树(BST),BST查找元素的时间复杂度平均是O(logN),最坏的情况是O(N),所有的元素都接在左子树(或者右子树)就相当于一串链表了.而AVL树会对子树过高的情况 ...

  10. tensorflow:实战Google深度学习框架第三章

    tensorflow的计算模型:计算图–tf.Graph tensorflow的数据模型:张量–tf.Tensor tensorflow的运行模型:会话–tf.Session tensorflow可视 ...