简单说下关于开发中容易遇到的父试图添加手势与子试图点击事件冲突,UIScrollView 嵌套 UIScrollView 、 UIScrollView 嵌套 UITableView的情况手势冲突问题;

点击冲突

如果给现有的基于 UIView 的 xkTestView 上加一个点击手势 gestTap,然后在 xkTestView 中间区域添加一个 tableview,我们想响应 gestTap,同时也想响应 tableview 的 cell 点击代理事件,这时可以添加 gestTap 点击手势代理:

<UIGestureRecognizerDelegate>

然后在点击事件代理方法中实现

- (BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldReceiveTouch:(UITouch *)touch {
if ([NSStringFromClass([touch.view class]) isEqualToString:@"xkTestView"]) {
return YES;
}
return NO;
}

scrollView 嵌套 tableView 类冲突

这里直接用 scrollView 嵌套 tableView 来处理下滑动时的手势冲突问题,其实苹果并不建议我们这样做,但是在实际项目中,有些需求会经常用嵌套来实现,在什么情况下滑动 tableView 不滑动 scrollView,什么情况下滑动 scrollView 不滑动 tableView,其实如果做其他的嵌套都是一样的,先看下最终效果图:

1)首先新建一个基于 UIScrollView 的 XKBaseScrollView ,并实现 <UIGestureRecognizerDelegate> 代理,XKBaseScrollView 用做主父试图来添加子试图内容

XKBaseScrollView.h

#import <UIKit/UIKit.h>

@interface XKBaseScrollView : UIScrollView <UIGestureRecognizerDelegate>

@end

XKBaseScrollView.m

#import "XKBaseScrollView.h"

@implementation XKBaseScrollView

//是否支持多时候触发,这里返回YES
-(BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer{
return YES;
} @end

2)然后新建一个基于 UITableView 的 XKTargetTableView ,并实现 <UIGestureRecognizerDelegate,UITableViewDelegate,UITableViewDataSource> 代理

XKTargetTableView.h

#import <UIKit/UIKit.h>

@interface XKTargetTableView : UITableView
///可否滑动
@property (nonatomic,assign) BOOL canSlide;
///滑动block通知
@property (nonatomic,copy) void (^slideDragBlock)(void);
@end

XKTargetTableView.m

#import "XKTargetTableView.h"
@interface XKTargetTableView ()<UIGestureRecognizerDelegate,UITableViewDelegate,UITableViewDataSource>
@property (nonatomic,assign) CGFloat currOffsetY;
@end
@implementation XKTargetTableView - (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 ;
}
- (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath{
return ;
}
- (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(, scrollView.contentOffset.y == ? : _currOffsetY);
}
_currOffsetY = scrollView.contentOffset.y;
if (scrollView.contentOffset.y < ) {
self.canSlide = NO;
scrollView.contentOffset = CGPointZero;
//到顶通知父视图改变状态
if (self.slideDragBlock) {
self.slideDragBlock();
}
}
scrollView.showsVerticalScrollIndicator = self.canSlide ? YES : NO;
}
@end

3)最后在使用的 ViewController 中实现

#import "ViewController.h"
#import <SDAutoLayout.h> #import "XKBaseScrollView.h"
#import "XKTargetTableView.h" @interface ViewController ()<UIScrollViewDelegate>
///容器
@property (nonatomic,strong) XKBaseScrollView *scrollView;
@property (nonatomic,strong) XKTargetTableView *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 = ;
[self.view addSubview:self.scrollView];
self.scrollView.sd_layout.
topSpaceToView(self.view, ).
leftSpaceToView(self.view, ).
rightSpaceToView(self.view, ).
bottomSpaceToView(self.view, ); [self.scrollView setupAutoContentSizeWithBottomView:self.tableView bottomMargin:];
__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(, self.dragCriticalY);
if (self.canSlide) {
self.canSlide = NO;
self.tableView.canSlide = YES;
}
else{
if (_lastPositionY - currentPostion > ){
if (self.tableView.contentOffset.y > ) {
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(, self.dragCriticalY);
}
else{
if (self.tableView.canSlide &&
self.tableView.contentOffset.y != ) {
scrollView.contentOffset = CGPointMake(, self.dragCriticalY);
}
else{ }
}
} _lastPositionY = currentPostion;
} - (XKBaseScrollView *)scrollView{
if (!_scrollView) {
_scrollView = [[XKBaseScrollView 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, ).
leftSpaceToView(_scrollView, ).
rightSpaceToView(_scrollView, ).
heightIs(); [_scrollView addSubview:self.tableView];
self.tableView.sd_layout.
topSpaceToView(view, ).
leftSpaceToView(_scrollView, ).
rightSpaceToView(_scrollView, ).
heightIs(self.view.bounds.size.height - ( - self.dragCriticalY));
}
return _scrollView;
}
- (XKTargetTableView *)tableView{
if(!_tableView){
_tableView = [[XKTargetTableView alloc]initWithFrame:CGRectZero style:UITableViewStylePlain];
}
return _tableView;
}
@end

注:此 demo 需引用 SDAutoLayout

iOS-关于一些手势冲突问题(scrollView 嵌套 tableView)的更多相关文章

  1. scrollview嵌套tableview

    之前写过一次,忘记了, 用的通知 FS

  2. iOS LeftMenu抽屉效果与ScrollView共存时的手势冲突

    公司有个项目,需要做左侧滑动,首页是ScrollView嵌套TableView.首页是一个ScrollView,所以当contentOffset是0.0的时候,无法直接滑动出抽屉效果,用户体验感非常差 ...

  3. 当ViewPager嵌套在ScrollView/ListView里时,手势冲突如何处理?

    有时我们需要将ViewPager嵌套在其他已经含有手势动作的ViewGroup里,如ScrollView,ListView时,会造成手势冲突,如表现为ViewPager向左划时,不小心向上移动了一点距 ...

  4. 解决嵌套在ScrollView中的TableView滑动手势冲突问题

    最近在迭代开发公司项目的时候遇到了一个问题,在可以左右切换标签视图的ScrollView中嵌套了两个TableView用于展示视图,感觉一切so easy的情况下,问题出现了,因为左右两个视图既可以实 ...

  5. (转)ViewPager,ScrollView 嵌套ViewPager滑动冲突解决

    ViewPager,ScrollView 嵌套ViewPager滑动冲突解决 本篇主要讲解一下几个问题 粗略地介绍一下View的事件分发机制 解决事件滑动冲突的思路及方法 ScrollView 里面嵌 ...

  6. 解决ScrollView嵌套ViewPager出现的滑动冲突问题

    /**       *         解决ScrollView嵌套ViewPager出现的滑动冲突问题       */       public class ScrollView1 extends ...

  7. iOS手势冲突问题

    今天在做一个效果的时候,由于子视图和父视图都有响应的事件,子视图的事件理所当然被父视图拦截掉了,接下来就做分析解决 1.  tableviewcell可以触发点击,同时tableview的父视图有点击 ...

  8. 解决iOS手势冲突问题

    今天在做一个效果的时候,由于子视图和父视图都有响应的事件,子视图的事件理所当然被父视图拦截掉了,接下来就做分析解决 1.  tableviewcell可以触发点击,同时tableview的父视图有点击 ...

  9. Android——MeasureSpec学习 - 解决ScrollView嵌套ListView和GridView冲突的方法

      原文地址:http://blog.csdn.net/yuhailong626/article/details/20639217   在自定义View和ViewGroup的时候,我们经常会遇到int ...

随机推荐

  1. 洛谷P1328==codevs3716 生活大爆炸版石头剪刀布[NOIP 2014 day1 T1]

    P1328 生活大爆炸版石头剪刀布 1.8K通过 2.6K提交 题目提供者2014白永忻 标签模拟NOIp提高组2014 难度普及- 提交该题 讨论 题解 记录 最新讨论 Who can help m ...

  2. BZOJ_3133_[Baltic2013]ballmachine_堆+倍增

    BZOJ_3133_[Baltic2013]ballmachine_堆+倍增 Description 有一个装球机器,构造可以看作是一棵树.有下面两种操作: 从根放入一个球,只要下方有空位,球会沿着树 ...

  3. bzoj2157 旅游——LCT

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=2157 仍然是LCT模板题~ 不过有一些需要注意的地方,点和边的区分,0号点的 mx 和 mn ...

  4. uva11149

    Consider an n-by-n matrix A. We define Ak = A ∗ A ∗ . . . ∗ A (k times). Here, ∗ denotes the usual m ...

  5. bzoj4407

    http://www.lydsy.com/JudgeOnline/problem.php?id=4407 以前写过一次线性筛 发现不是很理解 写了个欧拉筛的 t了 其实每次推式子,都会先推出一组的解法 ...

  6. 打开mat文件

    点击file目录,选择import data 然后选择所需.mat文件,就可以打开了

  7. E20170610-hm

    presence  n. 出席; 仪表; 风度; 鬼魂,神灵; defence   n. 防御; 辩护; 防御工事; 后卫; phyle  n. 种族,宗族; race  n. 赛跑; 民族; 人种; ...

  8. Nginx(三) 常用配置整理

    #定义Nginx运行的用户和用户组 user www www; #nginx进程数,建议设置为等于CPU总核心数. worker_processes 8; #全局错误日志定义类型,[ debug | ...

  9. [C陷阱和缺陷] 第5章 库函数

      有关库函数的使用,我们能给出的最好建议是尽量使用系统头文件,当然也可以自己造轮子,随个人喜好.本章将探讨某些常用的库函数,以及编程者在使用它们的过程中可能出错之处.   5.1 返回整数的getc ...

  10. 391 Perfect Rectangle 完美矩形

    有 N 个与坐标轴对齐的矩形, 其中 N > 0, 判断它们是否能精确地覆盖一个矩形区域.每个矩形用左下角的点和右上角的点的坐标来表示.例如, 一个单位正方形可以表示为 [1,1,2,2]. ( ...