iOS开发之多表视图滑动切换示例(仿"头条"客户端)
【大咖・来了 第7期】10月24日晚8点观看《智能导购对话机器人实践》
好长时间没为大家带来iOS开发干货的东西了,今天给大家分享一个头条新闻客户端各个类别进行切换的一个示例。在Demo中对所需的组件进行的简单封装,在封装的组件中使用的是纯代码的形式,如果想要在项目中进行使用,稍微进行修改即可。
废话少说,先介绍一下功能点,下图是整个Demo的功能点,最上面左边的TabBarButtonItem是用来减少条目的,比如下图有三个按钮,点击减号会减少一个条目。右边的为增加一个条目。点击相应的按钮是切换到对应的表视图上,下方红色的是滑动的指示器,同时支持手势滑动。运行具体效果如下图所示。
一:实现方案
最上方是一个View, View上面实例化了一些按钮,平分屏幕的宽度,下方是一个ScrollView, ScrollView上面放了一些表视图,点击不同的Button, 滑动到对应的表示图上。除了点击按钮,还可以进行滑动切换,切换时,红色的指示器也会随之滑动。
主要的技术点就是通过ScrollView的回调,通过事件的响应来改变ScrollView的ContentOffset的值。在回调中根据ContentOffset的值来计算红色指示器的偏移量。
二:核心代码
1.组件中的主要属性
把上面整个视图进行了封装,命名为SlideTabBarView,下面的代码是主要属性:
- @interface SlideTabBarView()///@brife 整个视图的大小
- @property (assign) CGRect mViewFrame;
- ///@brife 下方的ScrollView
- @property (strong, nonatomic) UIScrollView *scrollView;
- ///@brife 上方的按钮数组
- @property (strong, nonatomic) NSMutableArray *topViews;
- ///@brife 下方的表格数组
- @property (strong, nonatomic) NSMutableArray *scrollTableViews;
- ///@brife TableViews的数据源
- @property (strong, nonatomic) NSMutableArray *dataSource;
- ///@brife 当前选中页数
- @property (assign) NSInteger currentPage;
- ///@brife 下面滑动的View
- @property (strong, nonatomic) UIView *slideView;
- @end
2.初始化方法如下,在调用初始化方法时需要传入SlideTabBarView的frame和选项卡的个数,初始化函数会调用一系列的初始化方法对组件进行初始化,代码如下:
- -(instancetype)initWithFrame:(CGRect)frame WithCount: (NSInteger) count{
- self = [super initWithFrame:frame];
- if (self) {
- _mViewFrame = frame;
- _tabCount = count;
- _topViews = [[NSMutableArray alloc] init];
- _scrollTableViews = [[NSMutableArray alloc] init];
- [self initDataSource];
- [self initScrollView];
- [self initTopTabs];
- [self initDownTables];
- [self initDataSource];
- [self initSlideView];
- }
- return self;
- }
3.initDataSource方法主要负责模拟生成下方TableView要显示的数据。代码如下:
- #pragma mark -- 初始化表格的数据源
- -(void) initDataSource{
- _dataSource = [[NSMutableArray alloc] initWithCapacity:_tabCount];
- for (int i = 1; i <= _tabCount; i ++) {
- NSMutableArray *tempArray = [[NSMutableArray alloc] initWithCapacity:20];
- for (int j = 1; j <= 20; j ++) {
- NSString *tempStr = [NSString stringWithFormat:@"我是第%d个TableView的第%d条数据。", i, j];
- [tempArray addObject:tempStr];
- }
- [_dataSource addObject:tempArray];
- }
- }
4.红色滑动指示器的初始化代码如下所示:
- #pragma mark -- 初始化滑动的指示View
- -(void) initSlideView{
- CGFloat width = _mViewFrame.size.width / _tabCount;
- _slideView = [[UIView alloc] initWithFrame:CGRectMake(0, TOPHEIGHT - 5, width, 5)];
- [_slideView setBackgroundColor:[UIColor redColor]];
- [self addSubview:_slideView];
- }
5.ScrollView的初始化代码如下, 指定ScrollView的大小位置以及背景颜色,并且设置分页可用并添加代理。
- #pragma mark -- 实例化ScrollView
- -(void) initScrollView{
- _scrollView = [[UIScrollView alloc] initWithFrame:CGRectMake(0, _mViewFrame.origin.y, _mViewFrame.size.width, _mViewFrame.size.height - TOPHEIGHT)];
- _scrollView.contentSize = CGSizeMake(_mViewFrame.size.width * _tabCount, _mViewFrame.size.height - 60);
- _scrollView.backgroundColor = [UIColor grayColor];
- _scrollView.pagingEnabled = YES;
- _scrollView.delegate = self;
- [self addSubview:_scrollView];
- }
6.添加上方的按钮,根据传入的个数来实例化多个按钮。
- #pragma mark -- 实例化顶部的tab
- -(void) initTopTabs{
- CGFloat width = _mViewFrame.size.width / _tabCount;
- for (int i = 0; i < _tabCount; i ++) {
- UIView *view = [[UIView alloc] initWithFrame:CGRectMake(i * width, 0, width, TOPHEIGHT)];
- view.backgroundColor = [UIColor lightGrayColor];
- if (i % 2) {
- view.backgroundColor = [UIColor grayColor];
- }
- UIButton *button = [[UIButton alloc] initWithFrame:CGRectMake(0, 0, width, TOPHEIGHT)];
- button.tag = i;
- [button setTitle:[NSString stringWithFormat:@"按钮%d", i+1] forState:UIControlStateNormal];
- [button addTarget:self action:@selector(tabButton:) forControlEvents:UIControlEventTouchUpInside];
- [view addSubview:button];
- [_topViews addObject:view];
- [self addSubview:view];
- }
- }
7.点击按钮触发的方法如下:
- #pragma mark --点击顶部的按钮所触发的方法
- -(void) tabButton: (id) sender{
- UIButton *button = sender;
- [_scrollView setContentOffset:CGPointMake(button.tag * _mViewFrame.size.width, 0) animated:YES];
- }
8.初始化下方的多个表视图:实例化表视图,并指定委托回调。
- #pragma mark --初始化下方的TableViews
- -(void) initDownTables{
- for (int i = 0; i < _tabCount; i ++) {
- UITableView *tableView = [[UITableView alloc] initWithFrame:CGRectMake(i * _mViewFrame.size.width, 0, _mViewFrame.size.width, _mViewFrame.size.height - TOPHEIGHT)];
- tableView.delegate = self;
- tableView.dataSource = self;
- [_scrollTableViews addObject:tableView];
- [_scrollView addSubview:tableView];
- }
- }
9.ScrollView的回调方法如下,下面***一个代理方法是根据ScrollView的偏移量来计算红色指示器的偏移量,第二个是滑动到哪个tableView,然后进行哪个TableView的数据加载。
- #pragma mark -- scrollView的代理方法
- -(void)scrollViewDidEndScrollingAnimation:(UIScrollView *)scrollView{
- [self scrollViewDidEndDecelerating:scrollView];
- }
- - (void)scrollViewDidEndDecelerating:(UIScrollView *)scrollView
- {
- _currentPage = _scrollView.contentOffset.x/_mViewFrame.size.width;
- UITableView *currentTable = _scrollTableViews[_currentPage];
- [currentTable reloadData];
- }
- -(void)scrollViewDidScroll:(UIScrollView *)scrollView{
- if ([_scrollView isEqual:scrollView]) {
- CGRect frame = _slideView.frame;
- frame.origin.x = scrollView.contentOffset.x/_tabCount;
- _slideView.frame = frame;
- }
- }
10.TableView的代理方法如下,数据源就是我们刚才做的假数据,Cell是由Xib实现的,使用的时候注册一下就可用了。
- pragma mark -- talbeView的代理方法
- -(NSInteger)numberOfSectionsInTableView:(UITableView *)tableView{
- return 1;
- }
- -(NSInteger) tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section{
- NSMutableArray *tempArray = _dataSource[_currentPage];
- return tempArray.count;
- }
- -(CGFloat) tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath{
- return 60;
- }
- -(UITableViewCell *)tableView:tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
- BOOL nibsRegistered=NO;
- if (!nibsRegistered) {
- UINib *nib=[UINib nibWithNibName:@"SlideBarCell" bundle:nil];
- [tableView registerNib:nib forCellReuseIdentifier:@"SlideBarCell"];
- nibsRegistered=YES;
- }
- SlideBarCell *cell = [tableView dequeueReusableCellWithIdentifier:@"SlideBarCell"];
- if ([tableView isEqual:_scrollTableViews[_currentPage]]) {
- cell.tipTitle.text = _dataSource[_currentPage][indexPath.row];
- }
- return cell;
- }
- Demo在GitHub上的分享地址:https://github.com/lizelu/SliderTabBar
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
全新移动端页面检查工具:Mobile Checker
【大咖・来了 第7期】10月24日晚8点观看《智能导购对话机器人实践》 Mobile Checker(官网)是一个移动端页面检查工具,移动Web开发者可通过该工具发现网站在移动端存在的问题。调试Web页面或者Web App在移动设备上运行更加流畅。 Mobile Checker为移动Web开发者提供一个全新的并且有帮助的移动端页面开发指南。目前我们已经开发了基本功能,现在加入我们,一起开发出更好的功能。 如何使用? 这个工具是基于Node.js和Selenium WebDriver创建的一个纯JavaScript Web应用。基于Mobile Web browser emulator API。 Mobile Checker拥有强大的技术可在移动设备上模拟一个Web浏览器。这就是为什么,Mobile Checker与目前市场上大部分的在线移动模拟器不同,它可以提供一个模拟,使得您的Web App看上去像是在不同的移动设备上(包括平板电脑和智能手机)运行。 安装 Mobile Checker是一个Node应用,最终将通过npm进行分贝。但与此同时你也可以克隆这个仓库: git clone:...
- 下一篇
成也教程,败也教程 ——浅析游戏新手教程设计
【大咖・来了 第7期】10月24日晚8点观看《智能导购对话机器人实践》 你是否有过这样的体验?当你兴致勃勃地打开一款期待已久的新游戏准备一展身手时,却被小儿科的教程关拖住手脚兴致大减;或者在充满期待地准备开始一款从未接触过的全新类型游戏时,却被各种晦涩的隐藏功能搞得摸不到头脑。 新手教程的存在令玩家又爱又恨,但不可否认的是,新手教程是绝大多数游戏——尤其是有着创新体验的游戏——不可或缺的一部分。如何设计合理的新手教程,不仅关系到玩家是否能顺利玩懂游戏,更是玩家对游戏重要的***印象。 游戏教程与交互设计 生活中的各种体验会在玩家脑中留下习惯,利用这些习惯便可以在游戏交互中达到事半功倍的效果,红色令人紧张、蓝色令人平静、绿色令人感到安全这些心理学的研究被广泛应用在UI交互设计上,交互设计中仅利用这些恰当的视觉提示,便可以传达出足以让玩家理解游戏设定的信息。 所以有人说,真正好的交互体验是不需要教程的。苹果的产品之所以受到追捧,正是因为其优秀的交互设计,即便是没有任何电子产品操作经验的老人和儿童,也能凭本能认知学会基础操作。 《水果忍者》的教程和UI交互***融合 游 戏产品中也不乏这样的...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- Mario游戏-低调大师作品
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- CentOS7,8上快速安装Gitea,搭建Git服务器
- CentOS7编译安装Cmake3.16.3,解决mysql等软件编译问题
- Jdk安装(Linux,MacOS,Windows),包含三大操作系统的最全安装
- CentOS关闭SELinux安全模块
- CentOS8编译安装MySQL8.0.19
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- Windows10,CentOS7,CentOS8安装MongoDB4.0.16
- CentOS7安装Docker,走上虚拟化容器引擎之路