1.scrollView 嵌套 tableView 类冲突
这里直接用 scrollView(后面称父视图) 嵌套 tableView(后面称子视图) 来处理下滑动时的手势冲突问题,其实苹果并不建议我们这样做,但是在实际项目中,有些需求会经常用嵌套来实现,在什么情况下滑动 tableView 不滑动 scrollView,什么情况下滑动 scrollView 不滑动 tableView,其实如果做其他的嵌套都是一样的,先看下最终效果图:
1)首先新建一个基于 UIScrollView 的 ADABaseScrollView ,并实现 代理,ADABaseScrollView 用做主父试图来添加子试图内容
ADABaseScrollView.h
import
@interface ADABaseScrollView : UIScrollView
@end
ADABaseScrollView.m
import "ADABaseScrollView.h"
@implementation ADABaseScrollView
//是否支持多时候触发,这里返回YES
-(BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer{
return YES;
}
@end
2)然后新建一个基于 UITableView 的 ADATargetTableView ,并实现 代理
ADATargetTableView.h
import
@interface ADATargetTableView : UITableView
///可否滑动
@property (nonatomic,assign) BOOL canSlide;
///滑动block通知
@property (nonatomic,copy) void (^slideDragBlock)(void);
@end
ADATargetTableView.m
import "ADATargetTableView.h"
@interface ADATargetTableView ()
@property (nonatomic,assign) CGFloat currOffsetY;
@end
@implementation ADATargetTableView
-
(instancetype)initWithFrame:(CGRect)frame style:(UITableViewStyle)style{
self = [super initWithFrame:frame style:style];
if (self) {
self.backgroundColor = [UIColor whiteColor]; self.delegate = self; self.dataSource = self; self.tableFooterView = [UIView new]; [self registerClass:[UITableViewCell class] forCellReuseIdentifier:@"UITableViewCell"];
}
return self;
}
//是否支持多时候触发,这里返回YES
-
(BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer{
return YES;
}
pragma mark ========== tableView 代理 ==========
-
(NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section{
return 20;
}
-
(CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath{
return 50;
}
-
(UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{
UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"UITableViewCell"];
cell.textLabel.text = [NSString stringWithFormat:@"%ld",indexPath.row];
return cell;
}
pragma mark ========== scrollview 代理 ==========
-
(void)scrollViewWillBeginDragging:(UIScrollView *)scrollView{
_currOffsetY = scrollView.contentOffset.y; // 记录滑动的偏移
}
//重要
-(void)scrollViewDidScroll:(UIScrollView *)scrollView{
if (!self.canSlide) { // 如果不能滑动,保持滑动的偏移,不再滑动
scrollView.contentOffset = CGPointMake(0, scrollView.contentOffset.y == 0 ? 0 : _currOffsetY);
}
_currOffsetY = scrollView.contentOffset.y; // 记录滑动的偏移
if (scrollView.contentOffset.y < 0 ) { // 触发自己不能滑动的效果,(当自己从上向下,滑到顶部的时候,应该不能滑了。)
self.canSlide = NO; // 记录自己不能滑动了。偏移设置默认值,并通知外部
scrollView.contentOffset = CGPointZero;
//到顶通知父视图改变状态
if (self.slideDragBlock) {
self.slideDragBlock();
}
}
scrollView.showsVerticalScrollIndicator = self.canSlide ? YES : NO; // 按照能不能滑动,来显示滑动条
}
@end
import "ViewController.h"
import "SDAutoLayout.h"
import "ADABaseScrollView.h"
import "ADATargetTableView.h"
@interface ViewController ()
///容器
@property (nonatomic,strong) ADABaseScrollView *scrollView;
@property (nonatomic,strong) ADATargetTableView *tableView;
///是否可以滑动 scrollView
@property (nonatomic,assign) BOOL canSlide;
/// 记录偏移量
@property (nonatomic,assign) CGFloat lastPositionY;
///滑动临界范围值,到这个位置就不能再上滑了。
@property (nonatomic,assign) CGFloat dragCriticalY;
@end
@implementation ViewController
-
(void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
_dragCriticalY = 200;
[self.view addSubview:self.scrollView];
self.scrollView.sd_layout.
topSpaceToView(self.view, 0).
leftSpaceToView(self.view, 0).
rightSpaceToView(self.view, 0).
bottomSpaceToView(self.view, 0);
[self.scrollView setupAutoContentSizeWithBottomView:self.tableView bottomMargin:0];
__weak __typeof__(self) weekSelf = self;
self.tableView.slideDragBlock = ^{
weekSelf.canSlide = YES;
weekSelf.tableView.canSlide = NO;
};
}
//重要
-(void)scrollViewDidScroll:(UIScrollView *)scrollView{
CGFloat currentPostion = scrollView.contentOffset.y;
if (currentPostion >= self.dragCriticalY) { // 向上到临界值
scrollView.contentOffset = CGPointMake(0, self.dragCriticalY);
if (self.canSlide) {
self.canSlide = NO;
self.tableView.canSlide = YES;
} else {
if (_lastPositionY - currentPostion > 0){ // 上一次的滚动值大于现在的滚动值,意思是向下滚动。
if (self.tableView.contentOffset.y > 0) { // 如果子视图向上偏移了。子视图可以继续向下
self.tableView.canSlide = YES;
self.canSlide = NO;
} else {
self.tableView.canSlide = NO; // 把子视图的位置滑完为止
self.canSlide = YES;
}
}
}
} else {
if (!self.canSlide && scrollView.contentOffset.y == self.dragCriticalY ) { // 当到达临界值时,不是初始值。应该给父视图保持偏移
scrollView.contentOffset = CGPointMake(0, self.dragCriticalY);
} else {
if (self.tableView.canSlide &&
self.tableView.contentOffset.y != 0) { // 当子视图能滚动,并且子视图偏移了。应该给父视图保持偏移
scrollView.contentOffset = CGPointMake(0, self.dragCriticalY);
} else{
}
}
}
_lastPositionY = currentPostion;
}
-
(ADABaseScrollView *)scrollView{
if (!_scrollView) {
_scrollView = [[ADABaseScrollView alloc]init]; _scrollView.showsVerticalScrollIndicator = NO; _scrollView.delegate = self; _scrollView.backgroundColor = [UIColor redColor]; UIView *view = [[UIView alloc]init]; view.backgroundColor = [UIColor blueColor]; [_scrollView addSubview:view]; view.sd_layout. topSpaceToView(_scrollView, 0). leftSpaceToView(_scrollView, 0). rightSpaceToView(_scrollView, 0). heightIs(300); [_scrollView addSubview:self.tableView]; self.tableView.sd_layout. topSpaceToView(view, 0). leftSpaceToView(_scrollView, 0). rightSpaceToView(_scrollView, 0). heightIs(self.view.bounds.size.height - (300 - self.dragCriticalY));
}
return _scrollView;
}
-
(ADATargetTableView *)tableView{
if(!_tableView){
_tableView = [[ADATargetTableView alloc]initWithFrame:CGRectZero style:UITableViewStylePlain];
}
return _tableView;
}
@end
注:文章出自http://blog.sina.com.cn/s/blog_14ecbf39401030nh7.html