iOS-仿微信相册: 根据显示内容显示并切换当前内容所在年份

最近公司项目中有个需求: 基本就是copy的微信-我-相册。其中根据不同的内容(比如: 纯文字、纯图片、文字+图片、文字+链接等)有不同的显示样式,这里就不详细写代码了,如果有类似需求的,完全可以根据自身需求自定义不同的cell来满足。本Demo中自定义的cell是自适应高度的。

微信-我-相册:

mine.png
year.png

感觉需要了解下滑动视图的contentOffset属性的含义,如果不太了解的可以看下contentSize、contentOffset和contentInset的图解辨别;

采用懒加载方式创建控件

  • 新建一个工程,创建tableView,这个基本属于最经常使用的视图了,这里采用Group样式。
- (UITableView *)tableView
{
    if (!_tableView) {
        _tableView = [[UITableView alloc] initWithFrame:CGRectZero style:UITableViewStyleGrouped];
        _tableView.backgroundColor = [UIColor whiteColor];
        _tableView.separatorStyle = UITableViewCellSeparatorStyleNone;
        _tableView.delegate = self;
        _tableView.dataSource = self;
        _tableView.estimatedRowHeight = 100;
        _tableView.rowHeight = UITableViewAutomaticDimension;
    }
    return _tableView;
}

cell自适应高度,需要如下代码: 估算高度原则上大于0的数就可以;第二句可写可不写,只是个人习惯于加上。

_tableView.estimatedRowHeight = 100;
_tableView.rowHeight = UITableViewAutomaticDimension;
  • 工程中导航栏的设置不同,则在添加滑动系列控件时方式也有所区别,可以稍微了解下这个; 默认情况下,导航栏透明,对于第一个添加到控制器视图上的滑动系列视图会将其显示内容自动向下偏移64个像素,即: contentInset 为(64,0,0,0)。
    一般我都在所在控制器取消自动偏移属性: 将automaticallyAdjustsScrollViewInsets属性设置为NO(false)。
self.automaticallyAdjustsScrollViewInsets = NO;
  • 添加tableView到控制器视图
[self.view addSubview:self.tableView];
    [_tableView mas_makeConstraints:^(MASConstraintMaker *make) {
        make.leading.and.trailing.equalTo(@0);
        make.top.equalTo(self.mas_topLayoutGuide);
        make.bottom.equalTo(self.mas_bottomLayoutGuideTop);
    }];
  • tableView代理方法: 注意因为采用Group样式tableView,其section的header和footer一定要给高度且不可以为0,否则,会显示默认高度。根据需求第一个section不应有header。
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView
{
    return self.dataArray.count;
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section
{
    return [[self.dataArray objectAtIndex:section] count];
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath
{
    static NSString *iden = @"CCPhotoListCell";
    CCPhotoListCell *cell = [tableView dequeueReusableCellWithIdentifier:iden];
    if (!cell) {
        cell = [[CCPhotoListCell alloc] initWithStyle:UITableViewCellStyleDefault reuseIdentifier:iden];
        cell.selectionStyle = UITableViewCellSelectionStyleNone;
    }
    
    cell.model = self.dataArray[indexPath.section][indexPath.row];
    
    return cell;
}

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath
{
    NSLog(@"-click --%ld--%ld", indexPath.section, indexPath.row);
}

- (CGFloat)tableView:(UITableView *)tableView heightForHeaderInSection:(NSInteger)section
{
    if (section == 0) {
        return 0.01;
    }
    return 25;
}

- (UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section
{
    if (section == 0) {
        
        return nil;
    }
    
    CCPhotoListModel *model = (CCPhotoListModel *)[self.dataArray[section] firstObject];
    
    NSDate *date = [NSDate dateWithTimeIntervalSince1970:[model.time longValue]/1000];
    NSDateFormatter *form = [[NSDateFormatter alloc] init];
    [form setDateFormat:@"yyyy"];
    NSString *dateStr = [form stringFromDate:date];
    
    UIView *headView = [[UIView alloc] init];
    headView.backgroundColor = [UIColor colorWithHex:0xf5f7fa];
    
    UILabel *label = [[UILabel alloc] initWithFrame:CGRectMake(15, 0, 100, 25)];
    label.textColor = [UIColor colorWithHex:0x434A54];
    label.font = [UIFont boldSystemFontOfSize:13];
    label.text = dateStr;
    [headView addSubview:label];
    
    return headView;
}

- (CGFloat)tableView:(UITableView *)tableView heightForFooterInSection:(NSInteger)section
{
    if (section == self.dataArray.count - 1) {
        return 0.01;
    }
    return 10;
}

- (UIView *)tableView:(UITableView *)tableView viewForFooterInSection:(NSInteger)section
{
    if (section == self.dataArray.count - 1) {
        return nil;
    }
    UIView *footView = [[UIView alloc] init];
    footView.backgroundColor = [UIColor colorWithHex:0xf5f7fa];
    return footView;
}
  • 自定义cell的代码就不贴出来了 这个根据自身需求自定义cell就可以了。

  • 根据显示内容显示并切换当前内容所在的年份: 很显然,需要在滑动的过程中想办法计算当前显示内容所在年份然后更改顶部年份label的值。

- (void)scrollViewDidScroll:(UIScrollView *)scrollView
{
    NSLog(@"contentOffset: %.2f", scrollView.contentOffset.y);
    NSArray *array = [self.tableView indexPathsForVisibleRows];
    NSIndexPath *indexPath = [array firstObject];
    CGRect rectInTableView = [self.tableView rectForRowAtIndexPath:[NSIndexPath indexPathForRow:0 inSection:indexPath.section]];
    CGRect rect = [self.tableView convertRect:rectInTableView toView:self.view];
    
    NSLog(@"%.2f", rect.origin.y);
    if (rect.origin.y <= (64 + 25)) {
        
        CCPhotoListModel *model = (CCPhotoListModel *)[self.dataArray[indexPath.section] firstObject];
        
        NSDate *date = [NSDate dateWithTimeIntervalSince1970:[model.time longValue]/1000];
        NSDateFormatter *form = [[NSDateFormatter alloc] init];
        [form setDateFormat:@"yyyy"];
        NSString *dateStr = [form stringFromDate:date];
        self.timeLabel.text = [NSString stringWithFormat:@"%@年", dateStr];
    }
    
    if (scrollView.contentOffset.y > 0) {
        //显示
        [self.view addSubview:self.displayTimeView];
        [self.displayTimeView addSubview:self.timeLabel];
        self.displayTimeView.hidden = NO;
        self.timeLabel.hidden = NO;
        
    }else {
        //隐藏
        self.displayTimeView.hidden = YES;
        self.timeLabel.hidden = YES;
    }
    
}
  • 判断更改顶部显示时间label的值时,并不一定是if (rect.origin.y <= (64 + 25)) {},应根据自身项目来修改临界值。比如,若你先将一个scrollView放到self.view上,设置contentSize的宽为两倍屏幕宽度,再将两个tableView放在scrollView上,如果想达到此效果,临界值应该是25而不是25 + 64了。

  • 需要用到两个知识点

    • 获取显示在手机上的tableView的第一行的数据

      NSArray *array = [self.tableView visibleCells];
      UITableViewCell *cell = [array firstObject];
      
      或者
      
      NSArray *array = [self.tableView indexPathsForVisibleRows];
      NSIndexPath *indexPath = [array firstObject];
      
    - 获取tableviewCell在当前屏幕中的坐标值
    
     ```
     CGRect rectInTableView = [tableView rectForRowAtIndexPath:indexPath];   
    CGRect rect = [tableView convertRect:rectInTableView toView:[tableView superview]]; 
     ```
    

本Demo中[tableView superview]是控制器的视图,即: self.view。

效果:

演示.gif

Demo地址

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,904评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,581评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,527评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,463评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,546评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,572评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,582评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,330评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,776评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,087评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,257评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,923评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,571评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,192评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,436评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,145评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容

  • 1、通过CocoaPods安装项目名称项目信息 AFNetworking网络请求组件 FMDB本地数据库组件 SD...
    阳明先生_X自主阅读 15,979评论 3 119
  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,359评论 8 265
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,094评论 1 32
  • 生活有时毫无目的,急切地想逃到另一个地方。换了一个环境,在不同的地方重新回到开始的问题,好像突然就清晰了。 其实心...
    马每每阅读 243评论 0 0
  • 那天路过母校“丹东二中”旧址时,正单曲循环着郭旭的《不找了》。依着栅栏的角落望着校园,除了半座教学楼还依稀熟悉,其...
    一只小金鱼阅读 704评论 0 2