Objective-c——UI基础开发第八天(QQ聊天界面)

一、知识点:

  QQ聊天界面

  双模型的使用(dataModel和frameModel)

  UITextField的使用

  通知的使用

  拉伸图片的两种方法(slicing/image对象的resizeableImageWithCapInsets属性)

  枚举

  方法的抽取(相同的拿出,不同的部分作为参数)

二、设置tableview的基本格式

1)定义tableview基本

numberOfSectionsInTableView:设置块

numberOfRowsInSection:设置每块对应的行数

cellForRowAtIndexPath:设置cell样式

2)cell的重用

  设置cell

satic NSString *identifier [email protected]"QQCell";

  先从缓存池中寻找

QQCell *cell =[_tableview dequeuesableCellWithIdentifier:identifier];

  如果缓存池中找不到cell 就创建

if(cell==nil)

{cell =[[QQCell alloc]initWithStyle:UITableViewCellStyleDefault reuserIndentiier:identifier];}

3)设定自定义的cell格式(注意为什么消息框是button:因为既可以设置文本,又可以设置图片)

a、定义需要用到的属性及样式

@property(nonatomic,weak)UILabel *timeLabel;

@property(nonatomic,weak)UIButton *textButton;

@property(nonatomic,weak)UIImageView *iconImage;

b、定义初始化方法,注意与viewController中cell创建时调用的方法要一致

-(instancetype)initWithStyle:(UITableViewCellStyle)style reuseIdentifier:(NSString *)reuseIdentifier{

if(self =[super initWithStyle:style reuseIdentifier:reuseIdentifier]{

[self setupUI];//在内部设置button、label之类的样式

}}return self;

c、双模型(在懒加载中将数据模型传递给frame模型:原因:
  1、懒加载中,方法只需要执行一次

  2、行高rowheight 方法必须在cell方法调用前调用,所以要在cell中实现frame的计算,传递出来顺序是有问题的

传递方式: 在frameModel中定义QQModel这一属性

  a、定义model属性

  @class QQModel;

  @property (nonatomic ,strong) QQModel *qqModel;

  b、在懒加载中实现赋值

  QQModel *model =[QQModel QQmodelWithDictionary:dict];

  QQFrameModel *frameModel =[QQFrameModel alloc]init];

  frameModel.qqModel =model;//传递成功

  c、把frameModel的值传入dataArray数组中,因为这会frameModel中既有(数据model也包含各个控件的Frame)

  [_dataArray addObject:frameModel];

d、如何根据文本数据内容,设置frame

  1)、设置最大可变动区域(如果无限制则:MAXFLOAT)

  CGSize maxArea =CGSizeMake(textMaxA,MAXFLOAT);

  2)、设置属性字典(注意属性字体大小的设置一定要和原先控件中字体大小一致,否则,可能文本中以省略号带过)

  NSDictionary *dict [email protected]{NSFontAttributeName:[UIFont systemFontOfSize:17};

  3)、设置对象数据的真实大小boundingRectWithSize

  CGSize RealSize =[_qqModel.text boundingRectWithSize :maxArea options:NSStringDrawingUsesLineFragmentOrigin attributes:dict context :nil].size;

  

e、在cell的设置类中调用frameModel和数据模型并设置

  1)重写FrameModel的set方法:

@property(nonatomic,strong) QQFrameModel *frameModel;

-(void)setFrameModel:(QQFrameModel *)frameModel{

  2)必须对模型进行实例化

_frameModel =frameModel;

  3)设置模型中的数据和Frame

}

  怎么获取qqModel

  #import “QQModel.h”

  QQModel *qqModel =_frameModel.qqModel;

f、代码实现图片拉伸

/**

1、获取图片

2、设置拉伸 线条构成的中心区域(区域越小越好,所以将宽高都设成其中心线)

3、调用image对象的resizableImageWithCapInSets属性

*/

UIImage *image =[UIImage imageNamed:imageName];

CGFloat *halfWidth =image.size.width/2;

CGFloat *halfHeight =image.size.height/2;

UIImage *resizeImage =[image resizableImageWithCapInsets:UIEdgeInsetsMake(halfHeight,halfWidth,halfHeight,halfWidth) resizingMode:UIImageResizingModeStretch];

g、注意图片拉伸之后,文本可能不在图片内部,可以通过增加button的frame大小,同时设置文本的那边距(contentEdgeInsets UIEdgeInsetsMake)

textButton.contentEdgeInsets =UIEdgeInsetsMake(20,20,20,20);

h、设置行高

heightForRowAtIndexPath

QQ FrameModel *frameModel =self.dataArray[indexPath.row];

return frameModel.cellHeight

三、比较时间(两两比较,同一时间内的label只显示一遍,可以知道,只要比较一次,而且又要获得上一次的时间,最好在懒加载中实现)

/**

0、在qqModel中设置一个HiddenTimeLabel属性 存储比较完的结果

1、取出上一次的数据

2、进行比较,如果一样的话,就往qqModel 中传入HiddenTimeLabel=YES

*/

@property (nonatomic,assign,getter = isHiddenTimeLabel) BOOL HiddenTimeLabel;

QQFrameModel *latFrameModel =self.dataArray.lastObject;

if([lastFrameModel.qqModel.time isEqualToString :frameModel.qqModel.time])

{

qqModel.HiddenTimeLabel=YES;

}

调用:

_timeLabel.hidden =qqModel.isHiddenTimeLabel;(在hidden属性中直接调用重写的get方法即可)

四、补充使用枚举(内部的值类别,枚举的值名称,默认从0开始)

typedef NS_ENUM(NSInteger,QQUserType)

{

QQUserTypeOther=0;

QQuserTypeMe,

};

枚举的好处:将无意义的数字转化成有意义的字符,增加代码可读性

五、设置底部的弹框(了解textField属性和通知概念)

五.1 TextField 的使用

[email protected]"占位符"

 UIView *left =[[UIView alloc]initWithFrame:CGRectMake(100, 0, 10, 10)];

left.backgroundColor=[UIColor orangeColor];

text.leftView=left;

text.leftViewMode = UITextFieldViewModeAlways;

(添加leftview的时候必须要设置UITextFieldViewMode否则无法显示

leftview/rightview/leftViewMode/rightViewMode

密码:
text.secureTextEntry=YES;

text.cleanButtonMode =UITextFieldViewModeWhileEditing;

五、2)通知的使用

/**

1、注册监听者

2、通知中心发布通知

3、将销毁的监听者移除监听

*/

1

/**

addObserver: 监听对象

selector:监听对象收到通知之后调用的方法

name:通知的名称

object: 通知的发布者

*/

[[NSNotificationCenter defaultCenter] addObserver:ZhangSan selector:@selector(recieveNotification:) name:@"HuoYing" object:souhuHuoYing];

2

/**

接受通知之后,要执行的方法

*/

-(void) recieveNotification:(NSNotification *)noti

{

NSDictionary *c = noti.userInfo;

Company *company = c[@"company"];

NSLog(@"%@ 订阅了 %@公司的 %@视频已经更新了",self.personname,company.companyname,company.moviename);

//  NSLog(@"%@",noti);

}

3

-(void) dealloc

{

[[NSNotificationCenter defaultCenter]removeObserver:self.personname];

}

五、3)设置键盘监听,通过监听传递出的通知,改变底部输入框的frame和tableview的frame

1、// 添加监听,  键盘即将隐藏的时候, 调用

[[NSNotificationCenter defaultCenter] addObserver:self

selector:@selector(keyboardWillDisAppear:)

name:UIKeyboardWillHideNotification

object:nil];

调用的监听通知名称包括:UIKeyboardWillHideNotification /键盘即将隐藏的通知名

name:UIKeyboardWillChangeFrameNotification/键盘隐藏或出现的通知名

UIKeyboardWillShowNotification/键盘出现的通知名

2、收到通知消息时要执行的动画:(UIView animateWithDuration)

[UIView animateWithDuration:interval animations:^{

self.view.transform = CGAffineTransformMakeTranslation(0, (keyboardEndY - keyboardBeginY));

}];

3、用户如果滚动tableview的时候,或者用户输入完成按return的时候,键盘都需要消失

- (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView {

// 撤销textField 的第一响应者身份

[_textField resignFirstResponder];

}

4、默认的时候,tableview都应该滚动到最后一行(UITableView scrollToRowAtIndexPath)

- (void)scrollToBottom {

NSIndexPath *indexPath = [NSIndexPath indexPathForRow:self.dataArray.count - 1 inSection:0];

[_tableView scrollToRowAtIndexPath:indexPath atScrollPosition:UITableViewScrollPositionBottom animated:YES];

}

5、既然是聊天框,那么cell就不应该有被选中时候的颜色(selectionStyle)

// 选中cell后不改变颜色

cell.selectionStyle = UITableViewCellSelectionStyleNone;

6、撤销监听

#warning 一定不要忘记移除监听者

- (void)dealloc {

[[NSNotificationCenter defaultCenter] removeObserver:self];

}

六、设置时间格式

/**

1、取出时间(NSDate date)

2、初始化格式NSDateFormatter (NASDateFormatter) formatter.dateformat(属性)

3、日期转字符 formatter stringFromDate:currentDate

*/

// 取出当前的时间

NSDate *currentDate = [NSDate date];

// 设置时间的格式

NSDateFormatter *formatter = [[NSDateFormatter alloc] init];

// 时间格式

// yyyy-MM-dd HH:mm:ss  时间格式

formatter.dateFormat = @"HH:mm";

NSString *dateString = [formatter stringFromDate:currentDate];

七、通知和代理的区别

共同:都可以传递消息

区别:代理是一对一传递,通知是多对多,可以有多个消息发布者和多个消息接收者

例如,代理可以知道textfiled什么时候开始输入值,什么时候结束输入值,但是像键盘这种弹入弹出它的时间都是不可控的,需要实时监测这个就需要通知中心发布消息。

时间: 2025-01-02 03:27:50

Objective-c——UI基础开发第八天(QQ聊天界面)的相关文章

UI基础之UITableView案例QQ聊天界面

数据模型: #import <Foundation/Foundation.h> typedef enum{ LLMessageTypeMe, LLMessageTypeOther }LLMessageType; @interface LLMessage : NSObject /** * time */ @property (nonatomic, copy) NSString *time; /** * text */ @property (nonatomic, copy) NSString *t

UI基础--UITableView实现仿QQ聊天页面

需求:类似于QQ聊天页面的展示,内容包括有头像.时间.聊天内容.相同时间发生的内容,只显示第一条内容的时间,并且点击输入框时,可以滚动到最后一条内容信息.具体效果图: 实例的文件结构: 实现的具体步骤: 1.布局界面,主要包括一个UIImageView.3个UIButton.1个UITextField: 2.自定义数据模型类,并测试数据是否能正常加载: 3.自定义cell,由于每行数据的高度都是不规则的,所以考虑先自定义好frame再来写自定义cell.属性包括frame模型以及生成可重用cel

【iOS开发-65】QQ聊天界面案例:自定义cell、图片拉伸处理、NSNotification通知、键盘与视图移动以及输入框左边缩进处理

(1)案例 (2)源代码于素材下载 http://pan.baidu.com/s/1bnpiBCz (3)总结 --还是代码封装.控制器.视图.模型分别独立.里面还有很多代码可以独立出来整一个类. --如果某一个值只有特定的几个数字,那么可以用枚举来定义,注意命名规范 typedef enum{ WPMessageTypeMe=0, WPMessageTypeOther=1 }WPMessageType; --依然是计算一段文字所占据的宽和高 CGSize textMaxSize=CGSizeM

亲身体验用Java写的仿qq聊天界面

Java开发工具有许多种,新手用记事本写Java程序,有些人用NetBean,jbuilder,高手用eclipse,下面介绍用eclipse开发qq聊天界面. 代码如下: package Myjava_QQ; import java.awt.*; import javax.swing.*; import Myjava_QQ.truess; import java.awt.event.*; import java.applet.*; import java.io.BufferedReader;

【3】QQ 聊天界面

1.说明 稍微修改了下QQ示例里面的聊天界面界面,然后把代码扣过来完成了QQ聊天界面部分,效果还可以. 2.代码部分 // QQTalk.h文件 #ifndef __QQ_TALK_H__ #define __QQ_TALK_H__ #include <DuiLib/DuiLibEnv.h> #include <DuiLib/UIlib.h> using namespace DuiLib; #define QQ_TALK_XML _T("chatbox.xml"

QQ聊天界面的输入法顶起界面底部输入框效果的实现

转载请注明:http://www.cnblogs.com/frank-zouxu/p/4127115.html 今天在公司做项目的时候遇到一个需求:需要做一个界面,效果类似QQ聊天界面,如图1,当我们点击内容输入框准备输入内容的时候,底部的表情框的那一栏会被输入法的软键盘给顶起来,默认状态下,输入法会覆盖掉我们的表情输入框.起初,百思不得解的我费尽了心思,未果,偶然看到此篇博客http://blog.csdn.net/twoicewoo/article/details/7384398.其实,欲达

简单模仿QQ聊天界面

首先看一下最终的效果,显示了消息时间,用户昵称,用户头像. 大致实现方法: 用最简单的ListView显示消息内容. 不同的用户使用不同的消息布局文件,例子有2个用户"Tony","Hill". 代码文件清单: 主布局文件activity_main.xml: 1 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" 2 xmlns:tools=&

IOS开发学习笔记043-QQ聊天界面实现

QQ聊天界面实现 效果如下: ? ?实现过程: ?1.首先实现基本界面 ? ? ? ? 头像使用 UIImageView : ? ? ? ? 文字消息使用 UIButton ? ? ? ? 标签使用 UILable :水平居中 ? ? ? ? 所有元素在一个cell中,在加载cell时进行判断显示和隐藏. ? ? ? ??合理设置各个控件之间的约束关系.主要是UIIimageVIew和UIButton顶部对齐,间距为10.UIButton的宽度设置一个约束范围,比如说 (>=60 &&

WPF和Expression Blend开发实例:模拟QQ登陆界面打开和关闭特效

不管在消费者的心中腾讯是一个怎么样的模仿者抄袭者的形象,但是腾讯在软件交互上的设计一直是一流的.正如某位已故的知名产品经理所说的:设计并非外观怎样,感觉如何.设计的是产品的工作原理.我觉得腾讯掌握了其精髓.在2013版的桌面版QQ中,腾讯的登陆界面在打开的时候有一个展开的过程,而关闭的时候有个收缩的过程.效果如图: 借助WPF和Expression Blend,我们可以轻易的实现这么一个效果,最终用比较慢的速率实现这个效果如下: 这个效果一共能够分成两个部分:展开和收缩,具体的代码如下: 收缩的