版權聲明
本文由陳懷哲首發自簡書:http://www.jianshu.com/users/9f2e536b78fd/latest_articles;
微信公衆號:陳懷哲(chenhuaizhe2016);
無需受權便可轉載,但請自覺保留以上版權聲明。css
在看這篇文章以前,你須要配置好服務器,以及完成註冊和登陸的基本功能,才能繼續好友和聊天的操做。python
下面兩篇文章是環境配置和註冊、登陸功能的詳細介紹:
XMPP的mysql和openfire環境配置
iOS中XMPP簡單聊天實現 註冊和登陸mysql
另外必須瞭解一些CoreData相關知識 sql
點擊登陸以後,驗證成功就會跳到好友頁面。這個時候須要顯示你已經有的好友。
那麼在tableViewCell中顯示好友姓名,須要數據源,數據源從服務器獲看你是否有好友,檢索到你的好友後把他顯示在列表上。
xmpp中管理好友的類是 XMPPRoster,而且使用coredata來儲存好友,達到數據持久化的效果。
那麼咱們能夠將獲取儲存好友的倉庫和xmppRoster對象的初始化封裝在XMPPManager中。
在.h文件中聲明:數組
//好友管理
@property(nonatomic,strong)XMPPRoster xmppRoster;
@interface XMPPManager : NSObject<XMPPStreamDelegate,XMPPRosterDelegate>
//2.好友管理//得到一個存儲好友的CoreData倉庫,用來數據持久化 XMPPRosterCoreDataStorage rosterCoreDataStorage = [XMPPRosterCoreDataStorage sharedInstance];//初始化xmppRoster self.xmppRoster = [[XMPPRoster alloc]initWithRosterStorage:rosterCoreDataStorage dispatchQueue:dispatch_get_main_queue()];//激活 [self.xmppRoster activate:self.xmppStream];//設置代理 [self.xmppRoster addDelegate:self delegateQueue:dispatch_get_main_queue()];
接收好友請求。
將接收到好友請求的方法也封裝在XMPPManager中:ruby
// 收到好友請求執行的方法-(void)xmppRoster:(XMPPRoster )sender didReceivePresenceSubscriptionRequest:(XMPPPresence )presence{ self.fromJid = presence.from; UIAlertView alert = [[UIAlertView alloc]initWithTitle:@"提示:有人添加你" message:presence.from.user delegate:self cancelButtonTitle:@"拒絕" otherButtonTitles:@"OK", nil]; [alert show];}
-(void)alertView:(UIAlertView )alertView clickedButtonAtIndex:(NSInteger)buttonIndex{ switch (buttonIndex) { case 0: [self.xmppRoster rejectPresenceSubscriptionRequestFrom:self.fromJid]; break; case 1: [self.xmppRoster acceptPresenceSubscriptionRequestFrom:self.fromJid andAddToRoster:YES]; break; default: break; }}
添加好友,添加的好友必須是服務器上存在的用戶,須要看對方是否贊成。對方贊成以後,刷新好友列表,顯示出來,同時在服務器上也要添加,這裏服務器上用的是coredata來存儲我的的好友信息。服務器
接下來是tableview數據源代理方法微信
這時候數組明顯是沒有jid對象的。獲取jid對象是在XMPPPRoster代理方法中實現的:優化
pragma mark 開始檢索好友列表的方法-(void)xmppRosterDidBeginPopulating:(XMPPRoster *)sender{ NSLog(@"開始檢索好友列表");}
pragma mark 正在檢索好友列表的方法-(void)xmppRoster:(XMPPRoster )sender didRecieveRosterItem:(DDXMLElement )item{ NSLog(@"每個好友都會走一次這個方法");//得到item的屬性裏的jid字符串,再經過它得到jid對象 NSString jidStr = [[item attributeForName:@"jid"] stringValue]; XMPPJID jid = [XMPPJID jidWithString:jidStr];//是否已經添加 if ([self.rosterJids containsObject:jid]) { return; }//將好友添加到數組中去 [self.rosterJids addObject:jid];//添加完數據要更新UI(表視圖更新) NSIndexPath *indexPath = [NSIndexPath indexPathForRow:self.rosterJids.count-1 inSection:0]; [self.tableView insertRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationAutomatic];}
pragma mark 好友列表檢索完畢的方法-(void)xmppRosterDidEndPopulating:(XMPPRoster )sender{ NSLog(@"好友列表檢索完畢");}
4. 刪除好友。列表刪除,數組刪除,服務器刪除。ui
pragma mark 刪除好友執行的方法-(void)tableView:(UITableView )tableView commitEditingStyle:(UITableViewCellEditingStyle)editingStyle forRowAtIndexPath:(NSIndexPath )indexPath{ if (editingStyle==UITableViewCellEditingStyleDelete) { //找到要刪除的人 XMPPJID jid = self.rosterJids[indexPath.row];//從數組中刪除 [self.rosterJids removeObjectAtIndex:indexPath.row];//從Ui單元格刪除 [tableView deleteRowsAtIndexPaths:@[indexPath] withRowAnimation:UITableViewRowAnimationAutomatic ];//從服務器刪除 [[XMPPManager defaultManager].xmppRoster removeUser:jid]; }}
5.進入聊天頁面
1.發送普通文本消息
一樣在XMPPManager中進行封裝;
//聊天信息歸檔
@property(nonatomic,strong)XMPPMessageArchiving xmppMessageArchiving;
//信息歸檔的上下文
@property(nonatomic,strong)NSManagedObjectContext messageArchivingContext;
在init初始化時:
//3.保存聊天記錄 //初始化一個倉庫 XMPPMessageArchivingCoreDataStorage *messageStorage = [XMPPMessageArchivingCoreDataStorage sharedInstance]; //建立一個消息歸檔對象 self.xmppMessageArchiving = [[XMPPMessageArchiving alloc]initWithMessageArchivingStorage:messageStorage dispatchQueue:dispatch_get_main_queue()]; //激活 [self.xmppMessageArchiving activate:self.xmppStream]; //上下文 self.messageArchivingContext = messageStorage.mainThreadManagedObjectContext;
在聊天頁面的viewDidload中:
發送普通消息:
-(void)doSend{ //建立一個消息對象,而且指明接收者 XMPPMessage *message = [XMPPMessage messageWithType:@"chat" to:self.chatToJid]; //設置消息內容 [message addBody:@"呵呵呵呵呵呵呵呵呵呵"]; //發送消息 [[XMPPManager defaultManager].xmppStream sendElement:message]; //發送成功或者失敗,有兩種對應的代理方法}
消息發送是否成功,會走下面的代理方法:
#pragma mark 刷新消息的方法-(void)reloadMessage{ //獲得上下文 NSManagedObjectContext context = [XMPPManager defaultManager].messageArchivingContext; //搜索對象 NSFetchRequest request = [[NSFetchRequest alloc]init]; //建立一個實體描述 NSEntityDescription entity = [NSEntityDescription entityForName:@"XMPPMessageArchiving_Message_CoreDataObject" inManagedObjectContext:context]; [request setEntity:entity]; //查詢條件 NSPredicate pre = [NSPredicate predicateWithFormat:@"streamBareJidStr = %@ AND bareJidStr = %@",[XMPPManager defaultManager].xmppStream.myJID.bare,self.chatToJid.bare]; request.predicate = pre; //排序方式 NSSortDescriptor sort = [[NSSortDescriptor alloc]initWithKey:@"timestamp" ascending:YES]; request.sortDescriptors = @[sort]; //執行查詢 NSError error = nil; NSArray array = [context executeFetchRequest:request error:&error]; if (self.messages.count != 0) { [self.messages removeAllObjects]; } [self.messages addObjectsFromArray:array]; [self.tableView reloadData];}
2.顯示聊天記錄
- (NSInteger)tableView:(UITableView )tableView numberOfRowsInSection:(NSInteger)section {
return self.messages.count;
}
- (UITableViewCell )tableView:(UITableView )tableView cellForRowAtIndexPath:(NSIndexPath )indexPath { static NSString cellIndentifier = @"cell"; UITableViewCell cell = [tableView dequeueReusableCellWithIdentifier:cellIndentifier]; if (cell==nil) { cell = [[UITableViewCell alloc]initWithStyle:UITableViewCellStyleValue1 reuseIdentifier:cellIndentifier]; } //將聊天信息放到cell上 //拿到一個聊天消息 XMPPMessageArchiving_Message_CoreDataObject message = self.messages[indexPath.row]; if (message.isOutgoing == YES) { cell.detailTextLabel.text = message.body; } }else{ cell.textLabel.text = message.body; } return cell;}
成功後就能夠聊天了:
3.發送圖片等消息(重點)
發送視頻等其餘文件也是同樣,xmpp中須要將圖片轉化成NSData,而後轉化成成base64的字符串進行傳輸,而後接收到以後再反轉化成圖片。
首先要訪問系統相冊。
遵循代理:
@interface ChatViewController ()<XMPPStreamDelegate,UIImagePickerControllerDelegate,UINavigationControllerDelegate>
這時候須要更改cellForRowAtIndexPath:方法,注意紅色部分。
我把圖片設置爲cell的imageView,因此圖片顯示了在左邊,說明圖片消息發送是成功的,視頻等其餘類型的消息,也是一樣的原理。
到這裏,簡單的聊天功能就實現了,瞭解了基本原理和操做,咱們本身還能夠加入更多的自定義,從而優化得更好。這裏僅僅講述了一些基本的方法,若是想了解更多,趕快本身動手實踐吧