iOS开发实现类似B站竖屏视频的拖动效果

最近尝试模仿实现B站的竖屏视频的拖动效果,实现的最终效果图如下:


最终效果图.gif

(视频有最大尺寸和最小尺寸限制,通过滑动UITableView来动态更改视频的高度)

github上Demo地址
相应的实现文件名称为:PullAndScrollViewController

项目开始前需要注意的点

在做这个项目时遇到了一些坑,在这里分享一下

使用Masonry.h进行view的初始化布局,之后在viewDidLayoutSubViews或者按钮的实现方法中改变view的frame,主要是改变高度

会发现,不论怎么写,界面上view的大小都不发生变化
但是使用RacObserve监听view的frame属性,就会发现,其实view的frame已经发生了变化

但是在界面上表现不出来
甚至在更改frame的大小后加上强制刷新的代码,界面上的表现依旧没什么反应

//强制刷新代码
[self.view setNeedsLayout];
[self.view layoutifNeeded];

后来发现,如果初始使用masonry布局进行约束,那么之后更改的话,同样需要使用masonry布局约束进行更改,这样可以很好的达到效果

如果前面布局使用frame直接布局,那么后面不论是更改frame还是通过masonry更改约束都能实现相应的效果

具体的原因我还没有确定,通过查询资料发现:
参考链接:https://www.sohu.com/a/195141167_163917
该文章中有提到:

首先你要知道autolayout和frame的关系,autolayout最终也是转成frame,masonry是建立在autolayout之上的。你没获取到正确的值,那是因为约束还没布局完成。相当于就是我们给一定的约束,系统内部自己去根据约束条件转成对应的frame,而这需要一个过程。想要拿到正确的frame最好的就是让autolayout完成之后,什么时候完成呢?那就是在layoutsubviews for view or didlayoutsubviews for controller 里获取,当然在控制器的viewdidappear里也拿得到,但是正确做法和最佳做法还是在控制器里的viewdidlayout里获取最好~因为autolayout会根据约束,不停的去改变frame,这方法里最后拿到的frame就是最终姿势.

意思就是masonry布局的并不能马上获取到frame的高度大小,autolayout转化为frame需要一定的时间,或许是因为使用masonry布局的,后续使用frame直接更改会出现一些问题

之后,去查看了masonry在github上的库,在其中的issue中看到了相同的提问


image.png

可惜,并没有进行解答
等后面找到相应的解答之后再更新在这里

项目中TestViewController就是为了验证这个问题所写的测试文件,其中使用#import <ReactiveObjC/ReactiveObjC.h>来对myView的frame属性进行监听
有兴趣的可以看看

具体的实现步骤

具体的实现文件为pullAndScrollViewConroller
在.h中定义相关的属性

@property (nonatomic, strong) UIView *myView;
@property (nonatomic, assign) CGFloat maxViewHeight;//最大高度
@property (nonatomic, assign) CGFloat minViewHeight;//最小高度

@property (nonatomic, strong) UITableView *tableView;
@property (nonatomic, assign) CGPoint scrollBeginDraggingOffset;

之后在.m中实现初始的基本的界面以及懒加载

- (void)viewDidLoad {
    [super viewDidLoad];
    
    self.view.backgroundColor = [UIColor whiteColor];
    self.navigationController.navigationBar.translucent = NO;
    self.title = @"pull scrollView Demo  使用frame来改变";
    
    //初始化高度
    self.minViewHeight = 200;
    self.maxViewHeight = 400;
    
    [self.view addSubview:self.myView];
    [self.view addSubview:self.tableView];
    //这个方法主要为了查看过程中一些属性的变化,在使用时可以将其注释掉
    [self addObserve];
}

- (void)addObserve {
    
    typeof(self) __weak weakSelf = self;
    [RACObserve(self.myView, frame) subscribeNext:^(id  _Nullable x) {
        typeof(weakSelf) __strong self = weakSelf;
        NSLog(@"--------------");
        NSLog(@"height高度发生了变化%f",self.myView.frame.size.height);
    }];
    [RACObserve(self, scrollBeginDraggingOffset) subscribeNext:^(id  _Nullable x) {
        typeof(weakSelf) __strong self = weakSelf;
        NSLog(@"1111111111111111");
        NSLog(@"scrollBeginDraggingOffSet发生了变化%f",self.scrollBeginDraggingOffset.y);
    }];
    [RACObserve(self.tableView, contentOffset) subscribeNext:^(id  _Nullable x) {
        typeof(weakSelf) __strong self = weakSelf;
        NSLog(@"222222222222222");
        NSLog(@"contentOffsetY发生了变化%f",self.tableView.contentOffset.y);
    }];
}

- (void)viewDidLayoutSubviews {
    [super viewDidLayoutSubviews];

    self.tableView.frame = CGRectMake(0, CGRectGetMaxY(self.myView.frame), self.view.bounds.size.height, self.view.bounds.size.height - CGRectGetMaxY(self.myView.frame));
}

相应的懒加载为

#pragma mark - lazy load
- (UIView *)myView {
    if (_myView) {
        return _myView;
    }
    _myView = [[UIView alloc] initWithFrame:CGRectMake(0, 0, self.view.bounds.size.width, 400)];
    _myView.backgroundColor = [UIColor yellowColor];
    return _myView;
}

- (UITableView *)tableView {
    if (_tableView) {
        return _tableView;
    }
    _tableView = [[UITableView alloc] initWithFrame:CGRectMake(0, 400, self.view.bounds.size.width, 200) style:UITableViewStylePlain];
    _tableView.backgroundColor = [UIColor clearColor];
    _tableView.showsVerticalScrollIndicator = YES;
    _tableView.delegate = self;
    _tableView.dataSource = self;
    [_tableView registerClass:[UITableViewCell class] forCellReuseIdentifier:@"UITableViewCell"];
    return _tableView;
}

实现UITableView的delegate/datasource协议

#pragma mark - UITableViewDelegate/DataSource
- (NSInteger)numberOfSectionsInTableView:(UITableView *)tableView {
    return 1;
}

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section {
    return 100;
}

- (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath {
    UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"UITableViewCell"];
    cell.textLabel.text = [NSString stringWithFormat:@"第%ld个cell",(long)indexPath.row];
    return cell;
}

- (void)tableView:(UITableView *)tableView didSelectRowAtIndexPath:(NSIndexPath *)indexPath {
    [tableView deselectRowAtIndexPath:indexPath animated:YES];
    NSLog(@"点击了第%ld个cell",(long)indexPath.row);
}

- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {
    return 44;
}

这样基金的界面就已经写好了,运行后的效果为:


image.png

此时滑动的话,上方黄色的UIView不会更换大小
为了达到我们最初的效果,我们的思路是在滑动的时候根据UITableView的contentOffset.y的大小与视频高度的比较判断来设置UITableView的偏移量

以此达到我们的效果
在viewDidLayoutSubViews中,我们设置了UITableView的顶部与myView的底部紧挨着

UITableView的滑动调用的就是UIScrollViewDelegate,前面有一篇文章专门写了UIScrollViewDelegate中各个协议方法的调用顺序。
ScrollView滑动协议方法探究

主要的就是在ScrollViewDidScroll协议方法中进行相应的逻辑处理

- (void)scrollViewDidScroll:(UIScrollView *)scrollView {
  //获取UITableView的偏移量
   CGFloat offsetY = scrollView.contentOffset.y;
  //计算UITableView的最大偏移量
   CGFloat maxOffsetY = scrollView.contentSize.height - 
   scrollView.contentInset.top - scrollView.contentInset.bottom - 
   scrollView.frame.size.height;
  if (offsetY > 0) {
        NSLog(@"向上滑动offsetY为正值,值的大小为%f",offsetY);
    } else {
        NSLog(@"向下滑动offsetY为负值,值的大小为%f",offsetY);
    }
    CGFloat height = self.myView.bounds.size.height;
    CGFloat currentHeight = self.myView.bounds.size.height;
    //根据当前view的高度判断,是否处在maxViewHeight和minViewHeight之间,如果处在之间,需要修改view的高度,不需要改变UITableView的contentOffset
 //下面的逻辑就是处在最大高度和最小高度之间,偏移多少,就修改高度多少,这样UITableView就不需要改变contentOffsetY
  if (offsetY > 0) {
        //表示向上滑动
        if (currentHeight > self.minViewHeight) {
            height = height - offsetY;
        }
    } else {
        //表示向下滑动
        if (currentHeight < self.maxViewHeight) {
            height = height - offsetY;
        }
    }
   //判断height在减去offsetY之后的高度是否还处于maxViewHeight和minViewHeight之间
   if (height < self.minViewHeight) {
        height = self.minViewHeight;
    } else if (height > self.maxViewHeight) {
        height = self.maxViewHeight;
    }
//当height的高度不等于currentHeight时,说明view的height发生了变化,需要修改view的frame的大小,UITableView的不需要再添加代码修改,UITableView的frame修改我们一直放在了viewDidLoadLayoutSubViews中
if (height != currentHeight) {
        self.myView.frame = CGRectMake(0, CGRectGetMinY(self.myView.frame), CGRectGetWidth(self.view.frame), height);
        [self.view setNeedsLayout];
    }
}

这样的话,相应的逻辑基本上就实现了,但是运行之后,看到效果并不如我们所想的那样
这样运行的效果图为:


初步效果图.gif

从图中可以看出,view的高度变化总是快速变化,和我们预期的想法不一致

后面使用RACObserve监听UITableView的contentOffset属性

[RACObserve(self.tableView, contentOffset) subscribeNext:^(id  _Nullable x) {
        typeof(weakSelf) __strong self = weakSelf;
        NSLog(@"222222222222222");
        NSLog(@"contentOffsetY发生了变化%f",self.tableView.contentOffset.y);
    }];

经过调试发现了逻辑上的漏洞

首先需要明确一点,对于UITableView,如果改变它的frame的位置,比如向上移动100,它的contentOffsyY会保持原状,不会发生变化
但是如果通过滑动来改变位置的话,contentOffsetY会发生一些变化
这部分可以通过自己编写例子验证,在Test2ViewController中我进行的这个验证
因为只要滑动,contentOffsetY就会有变化

上面的逻辑漏洞也就不难发现,在

if (height != currentHeight) {
        self.myView.frame = CGRectMake(0, CGRectGetMinY(self.myView.frame), CGRectGetWidth(self.view.frame), height);
        [self.view setNeedsLayout];
    }

这里,我们修改myView的frame之后,viewDidLayoutSubViews中会跟着修改UITableView的frame,这个过程中按照我们的设想,contentOffsetY不应该发生变化,甚至在滑动的过程中,修改的都是view的height高度,不应该改变contentOffstY

所以,最直接的就是记录下最初滑动前UITableView的contentOffsetY,之后在改变myView的frame之后,立马使用setContentOffset设置UITableView的偏移量和滑动前相同即可

记录滑动前的偏移量,我们可以在- (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView这个方法中国呢记录

- (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView {
    NSLog(@"scrollViewWillBeginDragging...");
    CGPoint p = scrollView.contentOffset;
    CGFloat maxOffsetY = scrollView.contentSize.height - scrollView.contentInset.bottom - scrollView.contentInset.top - scrollView.frame.size.height;
    if (p.y >= maxOffsetY) {
        p.y = maxOffsetY;
    }
    self.scrollBeginDraggingOffset = p;
}

之后,scrollViewDidSCroll中的逻辑需要添加以下代码

CGFloat originOffsetY = MAX(0, self.scrollBeginDraggingOffset.y);
offsetY = MIN(offsetY, maxOffsetY) - originOffsetY;
其他的相同
if (height != currentHeight) {
        self.myView.frame = CGRectMake(0, CGRectGetMinY(self.myView.frame), CGRectGetWidth(self.view.frame), height);
        //加一句这个代码
        [scrollView setContentOffset:CGPointMake(0, originOffsetY)];
        [self.view setNeedsLayout];
    }

这样运行后,最终的效果图


动态改变视频大小.gif

和我们预期的结果一致

总结

github上Demo地址
相应的实现文件名称为:PullAndScrollViewController

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

推荐阅读更多精彩内容