上篇博客介绍了如何使用UITableView实现类似QQ的好友界面布局。这篇讲述如何利用自定义单元格来实现聊天界面的布局。

借助单元格实现聊天布局难度不大,主要要解决的问题有两个:

1.自己和其他人说话头像和气泡图像在不同的位置。

找了些类似的例子,有根据不同情况设置不同的自定义类的。这里使用根据说话人的属性来设置不同的位置,在一个单一的单元格类中。

2.像微博等根据说话的内容长短对说话图片进行拉伸,以及单元格自适应高度。

实现步骤:

搭建界面

数据属性字典

读取数据

- (void)loadData
{
const NSString *RMsgKey = @"msg";
const NSString *RMineKey = @"ismine"; NSString *path = [[NSBundle mainBundle] pathForResource:@"messages" ofType:@"plist"];
NSArray *dataArray = [NSArray arrayWithContentsOfFile:path];
if (!dataArray)
{
MyLog(@"读取文件失败");
return;
} _msgList = [NSMutableArray arrayWithCapacity:dataArray.count];
[dataArray enumerateObjectsUsingBlock:^(NSDictionary *dict, NSUInteger idx, BOOL *stop) {
Message *message = [[Message alloc] init];
message.msg = dict[RMsgKey];
message.mine = [dict[RMineKey] boolValue];
[_msgList addObject:message];
}];
}

将数据指定给Message类,存到一个数组中方便以后绑定到自定义单元格中

这次使用的是用NIB文件建的自定义,而非像前两篇博客使用手绘的方式。

awakeFromNib和layoutSubviews方法

- (void)awakeFromNib
{
_msgButton.titleLabel.numberOfLines = 0;
_msgButton.titleLabel.font = [UIFont systemFontOfSize:RChatFontSize];
} - (void)layoutSubviews
{
[super layoutSubviews]; CGRect rect = _msgButton.frame;
rect.size.height = self.bounds.size.height - 2*RMarginSize;
_msgButton.frame = rect;
}

然后是数据绑定方法,根据传入的数据,安排头像和会话按钮的位置

- (void)bindMessage:(Message *)message
{
UIImage *headerImage;
UIImage *normalImage;
UIImage *highlightedImage; CGRect iconRect = _headerView.frame;
CGRect btnRect = _msgButton.frame; UIEdgeInsets insets; if (message.isMine)
{
headerImage = [UIImage imageNamed:@"me"];
normalImage = [UIImage imageNamed:@"mychat_normal"];
highlightedImage = [UIImage imageNamed:@"mychat_focused"]; iconRect.origin.x = RMarginSize;
btnRect.origin.x = RBtnX; insets = UIEdgeInsetsMake(0, 20, 0, 30);
}
else
{
headerImage = [UIImage imageNamed:@"other"];
normalImage = [UIImage imageNamed:@"other_normal"];
highlightedImage = [UIImage imageNamed:@"other_focused"]; iconRect.origin.x = self.bounds.size.width - RMarginSize - RIconSide;
btnRect.origin.x = self.bounds.size.width - iconRect.origin.x - RMarginSize; insets = UIEdgeInsetsMake(0, 30, 0, 30);
}
_headerView.frame = iconRect;
_headerView.image = headerImage; //拉伸图片
normalImage = [normalImage stretchableImageWithLeftCapWidth:normalImage.size.width*0.5 topCapHeight:normalImage.size.height*0.6];
highlightedImage = [highlightedImage stretchableImageWithLeftCapWidth:highlightedImage.size.width*0.5 topCapHeight:highlightedImage.size.height*0.6]; [_msgButton setContentEdgeInsets:insets];
_msgButton.frame = btnRect;
[_msgButton setBackgroundImage:normalImage forState:UIControlStateNormal];
[_msgButton setBackgroundImage:highlightedImage forState:UIControlStateHighlighted];
[_msgButton setTitle:message.msg forState:UIControlStateNormal];
}

关于上面拉伸图片的方法,在IOS6以后可以使用另一个方法,resizableImageWithCapInsets:<#(UIEdgeInsets)#> resizingMode:<#(UIImageResizingMode)#>

传入一个Edgeinsets和一个拉伸模式就可以对图片完成编辑。

下面在tableview的代理方法中设置自适应高度的行高

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath
{
Message *msg = _msgList[indexPath.row];
UIFont *font = [UIFont systemFontOfSize:RChatFontSize];
CGFloat height = [msg.msg sizeWithFont:font constrainedToSize:CGSizeMake(150, 10000)].height;
CGFloat lineHeight = [font lineHeight]; return RCellHeight + height - lineHeight;
}

根据内容和字体设置合适的高度

最后绑定数据就可以完成显示了

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
HRChatCell *cell = [tableView dequeueReusableCellWithIdentifier:RCellIdentifier];
[cell bindMessage:_msgList[indexPath.row]];
return cell;
}

完成效果

Demo源码:点击打开链接

以上为本篇博客全部内容,欢迎指正和交流。转载请注明出处~

IOS详解TableView——对话聊天布局的实现的更多相关文章

  1. IOS详解TableView——选项抽屉(天猫商品列表)

    在之前的有篇文章讲述了利用HeaderView来写类似QQ好友列表的表视图. 这里写的天猫抽屉其实也可以用该方法实现,具体到细节每个人也有所不同.这里采用的是点击cell对cell进行运动处理以展开“ ...

  2. IOS详解TableView——内置刷新,EGO,以及搜索显示控制器

    内置刷新 内置刷新是苹果IOS6以后才推出的一个API,主要是针对TableViewController增加了一个属性,refreshControl,所以如果想用这个内置下拉刷新的话,最好给你的Tab ...

  3. Android 布局学习之——Layout(布局)详解二(常见布局和布局参数)

    [Android布局学习系列]   1.Android 布局学习之——Layout(布局)详解一   2.Android 布局学习之——Layout(布局)详解二(常见布局和布局参数)   3.And ...

  4. iOS:详解MJRefresh刷新加载更多数据的第三方库

    原文链接:http://www.ios122.com/2015/08/mjrefresh/ 简介 MJRefresh这个第三方库是李明杰老师的杰作,这个框架帮助我们程序员减轻了超级多的麻烦,节约了开发 ...

  5. Xamarin+Prism开发详解五:页面布局基础知识

    说实在的研究Xamarin到现在,自己就没设计出一款好的UI,基本都在研究后台逻辑之类的!作为Xamarin爱好者,一些简单的页面布局知识还是必备的. 布局常见标签: StackLayout Abso ...

  6. QDockWidget嵌套布局详解-实现Visual Studio布局

    概述 许多工程软件,如Qt Creator,VS,matlab等,都是使用dock布局窗口,这样用户可以自定义界面,自由组合窗口. Qt的嵌套布局由QDockWidget完成,用Qt Creator拖 ...

  7. [iOS]详解调整UIButton的title和image的位置

    UIButton的默认布局是:title在右,image在左; 很多时候我们需要的是title在左边,或者title在下面,这时就需要调整UIButton的TitleLabel和ImageView的位 ...

  8. 详解CSS的Flex布局

    本文由云+社区发表 Flex是Flexible Box 的缩写,意为"弹性布局",是CSS3的一种布局模式.通过Flex布局,可以很优雅地解决很多CSS布局的问题.下面会分别介绍容 ...

  9. Android五大布局详解——TableLayout(表格布局)

    TableLayout 前面所学的LinearLayout和RelativeLayout两大布局已经完全适用于各种开发条件下,其他的布局仅供参考学习,毕竟知识就是力量,以后的开发过程中万一遇到也能游刃 ...

随机推荐

  1. Android 5.0五大安全特性

    全盘加密(Full Disk Encryption, FDE) 对所有闪存数据加密.性能下降较大 Nexus 6,Nexus 9无法关闭FDE 对于其它设备.Google推荐开启 多用户支持 4.2中 ...

  2. if语句之有房么?有钱么?有能力么?

    思路:1.如果有房,可以谈谈 2.如果没有房,问第二个条件有钱么,如果有,可以谈谈 3.如果没有房没有钱,则问第三个条件有能力么,如果有,可以谈谈 4.如果以上三个条件都没有,则拜拜 Console. ...

  3. ThinkPHP - 关联模型 - 一对多

    使用之前,先引入文件夹,否则相应的功能不能实现. 如果对thinkPHP不精通,使用或开发的时候,最好直接使用完成版本的ThinkPHP. 关系模型定义: <?php /** * 继承自 Rel ...

  4. sql执行疑问

    遇到一个奇怪的事情,学习执行计划的时候写了这么一个sql来看执行计划的执行顺序, SELECT COUNT(*) FROM SIC84 WHERE AAC001 IN (SELECT AAC001 F ...

  5. php下正则表达式整理

    一.正则表达式的历史背景 1,内容深厚的正则表达式 ^.+@.+\\..+$ 形式 字符串搜索与匹配的工具 2,应用范围 手机输入法 Windows文件搜索 linux 列出文件命令 网站用户注册,如 ...

  6. 安卓Launcher之获取手机安装的应用列表,安卓launcher

    Launcher中最主要的就是获取所有应用列表的入口以及图标,一般获取的方法有两种: PackageInfo ResolveInfo 运行获取所有APP的Launcher并且允许进行点击事件,进入到应 ...

  7. Flask web开发 简单介绍

    Flask是一个基于python的轻量级web框架.当安装好后Flask后 (pip install flask),就可以开始使用了. 一.最简单的例子 1.新建目录,作为web应用的目录,如: mk ...

  8. springmvc+mybatis+redis(转)

    最近在学习redis的使用方法,它的本地使用方法比较简单,只需要先启动Redis服务器,然后运行测试代码即可.但是现在我想要在网站上访问数据库的时候采用Redis缓存,问题就出来了.要么是缓存直接失效 ...

  9. 安装Tomcat指定JDK(转)

    一.应用实例 一般情况下一台服务器只跑一个业务,那么就直接配置一套环境,设置好Java环境变量即可.某些时候一台服务器上会安装多个业务,而且各个业务需要的JDK版本各不相同,或者为了使业务独立开来,需 ...

  10. [转]如何申请和管理一个sourceforge项目

    假如你没有贡献免费源代码的想法,就不用继续读本文了.:) 如果读者的e文不错,作者建议你直接阅读sourceforge的 howto文档,他们整理得非常周到详实.本文只是作者的使用经验,也许可以给与作 ...