显示隐藏文件,看到git的文件夹 defaults write com.apple.finder AppleShowAllFiles Yes && killall Finder defaults write com.apple.finder AppleShowAllFiles No && killall Finder 设置git的(全局)用户与邮箱 git config --global user.name Fung git config --global user.email fung@qq.com 掌握的知识点 1.配置应用程序的iPad与iPhone的各个设备的AppIcon 2.配置应用程序的iPad与iPhone的各个设备的启动图片(LaunchImage) 一、设置AppIcon与LaunchImage 1 > 设置iPhone-iOS8. 0 /iOS7. 0 iPad- iOS7.0的AppIcon 2 > 设置LaunchImage时, a > Xcode6在TARGETS中去除Launch Sreen File里的值 b > 在Lauch Images Source中添加在assests中的启动图片 c > iPhone横屏与iPad横屏的启动图片不设置 d > iPad1024x768图片不设置 " 因为微信资源包没有找到对应尺寸的图片,说明有些启动图片可以不用设置 " " 微信iPad运行,如果是横屏启动时,是没有显示图片的 " 二、实现登录排版 1 > 创建一个Login.storyboard,使用导航控制器包装登录控制器 2 > 自定义导航控制器,设置导航条背景与状态栏样式 3 > 添加登录框容器,使用AutoLayout,固定大小,顶点间距,水平居中 4 > 使用Images.xcassets对登录按钮的背景图片进行slicing(切片操作) 5 > 设置密码的文本输入框样式 *设置borderStyle = UITextBorderStyleNone; " storyboard设置,提醒为什么不能设置高度 " * 设置textFiled的高度42 *设置backgroudImage为拉伸后的图片 " 代码实现,因为背景图片放在xcassets时,xcode会崩溃 " * 设置密码文件框左边的图片 设置图片控制的尺寸、图片内容模式 设置文本框的leftViewMode为总是显示(UITextFieldViewModeAlways) " 否则不显示左边的view " 5 > 对控制器view进行拖拽手势监听实现上,实现登录输入框的上下滑动及反弹 " 注:提交git版-微信目录结构分层+登录界面排版 " 三、注册界面排版 1 > 添加一个注册框容器,设置顶部,左边,右边距离固定,高度固定 2 > 添加手机与密码文本输入框、注册按钮,并设置背影图片,文字居中显示 3 > 设置按钮的可用状态 " " 4 > 判断当前设置是否为iphone,如果为iphone,注册框容器的左右两边间距为10 // [UIDevice currentDevice].userInterfaceIdiom == UIUserInterfaceIdiomPhone 5 > 设置导航条Item的样式- 白色字体 四、实现其它登录排版 0 >在登录界面的底部添加 " 其它方式登录> " 按钮 1 > 将 " 注册界面 " 的视图拷贝到新的控制器 2 > 去除横屏,这样在ipad上可以旋转,iphone上不可以 五、完成主界面排版[MainStoryboard] 1 > 微信、通讯录、发现、我[拖四个导航控制器到Tabbar控制器] 2 > 设置tabbarBtn的背景及文字,及导航栏的背景 微信 tabbar_mainframe / tabbar_mainframeHL 通讯录 tabbar_contacts / tabbar_contactsHL 发现 tabbar_discover / tabbar_discoverHL 我 tabbar_me / tabbar_meHL " 注:选设置普通状态的图片与文件,有时间再自定义tabbar " 3 > 设置全局的状态栏样式 *在info.plist中添加View controller- based status bar appearance=NO *在自定导航控制器是设置[UIApplication shareApplication].statusBarStyle = UIStatusBarStyleLightContent *设置导航栏标题 字体为粗体18/ 颜色为白色, iOS7或者iOS以上使用NSFontAttributeName / NSForegroundColorAttributeName属性 iOS7以下UITextAttributeFont /UITextAttributeColor a> 导入CocoaLumberjack 日志框架 " 无须依赖 " b > 导入CocoaAsyncSocket 底层网络框架,实现异步socket网络通讯 " 依赖:CFNetwork&Security框架 " c > 导入KisssXML XML解析框架 依赖libxml2.dylib " 还需要配置编译选项 "" " Other Linker Flags = -lxml2 " " Header Search Path = /usr/include/libxml2 " e > libidn 直接导入 f > 导入下面四个文件夹 Authentication Categories Utilities " 依赖libresolv.dylib " g > 添加Extensions h > 将Sample_XMPPFramework.h导入,并将其改名为XMPPFramework.h " 注在xcode6中系统有的Framework可以不用添加,但是dylib一定要添加 " 2 > 实现登录 *> 选分析登录的步骤 *> 初始化xmppStream核心类,并添加代理为全局队列dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0 ) *> 连接服务器时,设置主机,端口与登录账号[先写死代码] *> 在 " 代理中连接成功 " 时发送密码[先写死代码] *> 在 " 代理中授权成功 " 时, 打印 *> 在 " 代理中授权失败 " 时, 打印 *> 在 " 代理中授权成功 " 时,通知用户上线 *> 官方把登录数据放在沙盒里 " 我们放在单例对象 " *> 登录成功用block的方式通知 " 其它方式登录控制器 " *> 在其它方式登录控制器中,用MBProgessHUB 显示进度, " 在IPad中显示一定要用toView的显示方法,因为窗口没有横竖屏的概念,只有控制器的view才有横竖屏的概念 " *> 隐藏HUB时,引出block是在子线程调用的,不能直接刷新UI *> 接收block的成功登录时,切换到主界面, " 在切换到主界面前,要清除当前模态窗口 " 强调自己写的block引用控制器时,要用__weak 七、使用xmpp框架实现注册功能 *> 先分析注册的步骤,与登录形成一个对比 *> 在AppDelegate中添加用户注册的方法 *> 在AppDelegate中添加一个userRegister的属性 *> " 连接服务器,连接成功后判断userRegister在代理中做登录还是注册的操作 " *> 在代理中监听注册成功还是失败,再以block的方式通知 " 注册控制器 " *> " 注册控制器 " 再以 " 代理的方式 " 通知 " 登录控制器 " ,并设置label的注册后的帐号 *> 抽取 " 其它登录控制器 " 登录的代码 到一个父类,实现登录 0 > 用户注销[断开连接,回到登录界面] *> 通知用户下线、断开连接、显示登录界面 1 > 用户登录过,未注销,不管是程序在后台还是重新启动,直接来到主界面 2 > 用户登录过,未注销,退出台后时与服务器断开连接,获取焦点时获取自动 " 在UserInfo单例中添加一个login属性来记录登录状态 " 3 > 登录界面设置上次登录的帐号 九、自动登录提示 0 > 分析自动登录监听中的情况 1.开始连接服务器 2.连接失败 3.登录成功 4.登录失败 1 > 在历史控制器中添加通知监听 [[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(autoLoginStatus:) name:WXAutoLoginStatusNotification object :nil]; 2 > 在APPDelegate中发送通知 3 > " 自动登录时延迟两秒登录,好演示效果,连接的主机用域名比较慢 " 4 > 转圈圈指示器是否转还是停止,应该放在主线程中 十、socket后台支持 * 注 iOS8不用写支持也可以后台支持 * iOS7 和 iOS8 下都要演示 1 > " Required background modes " 的key,值设置为voip,来指定类型 2 > xmppStream.enableBackgroundingOnSocket = YES 3 > xmpp代理接收到消息时,如果不在前台实现本地通知 [UIApplication sharedApplication] applicationState] != UIApplicationStateActive 设置本地通知的内容(用户名\n内容前多少个字) 声音( default ) 4 > 在iOS8要注册通知类型registerUserNotificationSettings
//模块
一、自动连接模块
1> 当用户网络不稳定时,如果断线,自动连接
2> 打开自动连接模块在xmppframework.h文件中打开XMPPReconncet
3> 在设置xmppStream时,开始自动连接并"激活"
4> "引用官方示例程序的teardown方法" 释放资源
二、电子名片模块与头像模块
0> "查看电子名片的模型文件,数据结构"
0> 什么是电子名片模块/头像模块
1> 在xmppframework.h中打开电子名片模块/头像模块的头文件
2> 在设置xmppStream中,设置电子名片模块/头像模块并激活,
"获取用户的个人信息,里面是不是有头像,也一并获取"
3> 在teardownxmppStream做相应的资源释放
4> 在 "Me"中打印 doc,不要使用SimPhodlers
5> 在document看CoreData生成的数据库,用SQLiteManager查看
6> 在"" 中添加静态单元格,显示图片与账号
》XMPPvCardTemp *myvCard = xmppDelegate.vCardModule.myvCardTemp;
》warning 看PPT,查看有些数据是没有解析的
》取消头像,显示日志,使用XMLJSON格式化工具格式化,解释电子名片的电话未解析
三、电子名片详细页面进行设置
0> 进入详细页面进行静态表格的排版
1> 绑定数据
2> 如果识别不同类型cell的点击,设置每个cell的tag不同
3> 点击cell,取消选择中、进行头相选择
4> "编辑信息界面,传一个cell到编辑控制控制器,在调用代理时时调用
[self.profileCell layoutSubviews];//重新刷新cell就会显示detailTextLabel,因为没文件,默认是不会创建label"
5> 保存数据
四、获取好友列表(花名册)
0> "查看花名册的模型文件,数据结构"
0> 在xmppframework中打开花名册模块
1> setupXmppStream中添加花名册模块
2> teardown中释放花名删模块
3> 在联系人控制串获取好友列表
 * 获取好友列表时,一定要添加过滤条件?,因为多人登录,可能用同一个数据库?
 * 答案不会,演示数据,每次登录会清除以前的好友,做得不够好,实际开发自己创建数据库来管理
4> "使用CoreData的普通方式获取好友列表"
5> 使用NSFetchedResultsController 获取好友列表
6> "在cell中演示sectionNum的在线状态,然后在排序中多添加个sectionNum的升序",从而演示出NSFetchedResultsController的作用
7> 获取头像 自定联系人Cell
if (friend.photo) {
    cell.imageView.image = friend.photo;
}else{
    NSData *data = [xmppDelegate.vCardAvatarModule photoDataForJID:friend.jid];
    if (data) {
        cell.imageView.image = [UIImage imageWithData:data];
    }else{
        cell.imageView.image = [UIImage imageNamed:@"login_defaultAvatar"];
五、添加好友+删除好友
0> TextField左边的图片放在分类里
0> 是否添加为手机号友
/**注:写在分类UITextField里
 NSString *telRegex = @"^1[3578]\\d{9}$";
 NSPredicate *prediate = [NSPredicate predicateWithFormat:@"SELF MATCHES %@", telRegex];
 return [prediate evaluateWithObject:self.text];
1> 不能添加自己为好友
2> 添加的好友已经存在
[xmppDelegate.rosterStorage userExistsWithJID:friendJid xmppStream:xmppDelegate.xmppStream]
[xmppDelegate.roster subscribePresenceToUser:friendJid];
3> 删除好友
[xmppDelegate.roster removeUser:friend.jid]
六、聊天界面排版
0> 使用VFL排行聊天表格与输入框
tableViewHConst "H:|-0-[tableView]-0-|"
inputViewHConst "H:|-0-[inputView]-0-|"
vConst "V:|-0-[tableView]-0-[inputView(50)]-0-|"
1> 键盘监听,判断iphone还是ipad 再判断ipad的是横屏还是竖屏
UIInterfaceOrientationIsPortrait(self.interfaceOrientation)
#define iSiPhoneDevice ([UIDevice currentDevice].userInterfaceIdiom == UIUserInterfaceIdiomPhone)
"重点"
2> 返回按钮监听
*> 设置textfiel的代理 实现textViewDidChange
*> 判断是否有换行符 "\n"
*> 有就去除掉首尾的空白字符和换行字符
[textView.text stringByTrimmingCharactersInSet:[NSCharacterSet whitespaceAndNewlineCharacterSet]];  //
七、获取聊天消息+发送聊天消息
"聊天消息模块:什么是聊天消息模块,xmpp帮我们做数据存储 查看0135模块的数据结构"
0> xmppframework.h中打开消息归档模块与消息数据存储
1> 在setupXmppStream中消息归档模块与消息数据存储
2> 在teardown中释放消息归档模块与消息数据存储
3> 在通讯录控制器传递好友jid到聊天控制器
4> 从数据库加载聊天数据,一次性加载全部"使用NSFecthResultsController"
"强调:查询时要带条件,一定要属于当前登录用户与好友的聊天记录"
5> 发送消息
XMPPMessage *message = [XMPPMessage messageWithType:@"chat" to:self.friendJid];
[message addBody:msg];
[xmppDelegate.xmppStream sendElement:message];
6> 聊天数据滚动到底部
7> 讲解怎么判断自己发送的消息还是别人发的消息
八、实现输入框随文字的伸缩
九、实现图片的发送级语音发送的原理
#pragma mark 发送图片
-(void)sendImage:(UIImage *)image{
    XMPPMessage *msg = [XMPPMessage messageWithType:@"chat" to:self.friendJid];
    NSData *data = UIImagePNGRepresentation(image);
    [msg addBody:@"image"];
    XMPPElement *attachment = [XMPPElement elementWithName:@"attachment" stringValue:[data base64EncodedStringWithOptions:0]];
    [msg addChild:attachment];
    [[WXXMPPTools sharedWXXMPPTools].xmppStream sendElement:msg];
#parm mark cell显示
XMPPMessage *xmppMsg = msg.message;
chatCell.imageView.image = nil;
if ([msg.body isEqual:@"image"]) {
    NSArray *child = xmppMsg.children;
    for (XMPPElement *node in child) {
        if([[node name] isEqualToString:@"attachment"]){
            NSString *base64 = [node stringValue];
            NSData *imageData = [[NSData alloc] initWithBase64EncodedString:base64 options:0];
            chatCell.imageView.image = [UIImage imageWithData:imageData];
[message addAttributeWithName:@"bodyType" stringValue:@"image"];
[xmppMsg attributeStringValueForName:@"bodyType"]
NSArray *emailEles = [self elementsForName:@"EMAIL"];
if (emailEles.count != 0) {
    NSXMLElement *email = emailEles[0];
    NSXMLElement *userID = [email elementForName:@"USERID"];
    return @[[userID stringValue]];
NSArray *emailEles = [self elementsForName:@"EMAIL"];
if (emailEles.count != 0 && emails.count!=0) {
    NSXMLElement *email = emailEles[0];
    NSXMLElement *userID = [NSXMLElement elementWithName:@"USERID" stringValue:emails[0] ];
    [email removeElementForName:@"USERID"];
    [email addChild:userID];