V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
tunnyios
V2EX  ›  iOS

scrollView 中切换百度地图 View

  •  
  •   tunnyios · 2015-07-25 16:35:29 +08:00 · 2212 次点击
    这是一个创建于 3456 天前的主题,其中的信息可能已经有所发展或是发生改变。
    http://tunnyios.github.io/personal/BMKMap_ScrollView/

    ##scrollView中切换百度地图
    要实现如下功能:通过滑动切换View和BMKMapView,手势滑动超过一半后,显示对应的View。

    效果图如下:
    ![](http://7xke07.com1.z0.glb.clouddn.com/image/BMKMapView_scrollView.gif)


    ###实现方法

    此处使用storyboard来实现scrollView(使用autolayout方便做屏幕适配,storyboard如何实现请参考url:*http://www.cocoachina.com/ios/20150104/10810.html*)。

    使用UIScrollView来进行切换时,遇到一个问题:切换到BMKMapView视图上时,左右滑动的手势被UIScrollView拦截了,导致,BMKMapView上不能够左右滑动。因此需要自定scrollView。

    通过自定义scrollView,重写两个方法- (BOOL)touchesShouldBegin:(NSSet *)touches withEvent:(UIEvent *)event inContentView:(UIView *)view 和 - (BOOL)touchesShouldCancelInContentView:(UIView *)view,来实现。

    ###scrollView的原理

    当多个视图进行叠加的时候,touch事件是作用到最上面的视图上,但是如果父视图是UIScrollView,如果默认,可能touch子视图会造成UIScrollView的滚动。

    当手指触摸到UIScrollView内容的一瞬间,会产生下面的动作:

    * 拦截触摸事件
    * tracking属性变为YES
    * 一个内置的计时器开始生效,用来监控在极短的事件间隔内是否发生了手指移动
    *
    > * case1:当检测到时间间隔内手指发生了移动,UIScrollView自己触发滚动,取消发送tracking 给子视图。tracking属性变为NO。手指触摸下即使有(可以响应触摸事件的)内部控件也不会再响应触摸事件。
    > * case2:当检测到时间间隔内手指没有移动,当时间结束时,,UIScrollView会发送tracking events到子视图上。tracking属性保持YES。手指触摸下如果有(可以响应触摸事件的)内部控件,则将触摸事件传递给控件进行处理。

    UIScrollView的子控件想要接收touch事件,就是用户点击UIScrollView上的子视图时,要先处理子视图上的touch,而不发生滚动。这时候就需要自定义UIScrollView重载touchesShouldBegin:withEvent:inContentView: ,从而决定自己是否接受子视图中的touch事件。

    ###自定义scrollView的几个注意点

    1. touchesShouldBegin:withEvent:inContentView 决定自己是否接收 touch 事件,YES:(自己不接收,发送事件)即不滚动;NO:(自己接收,不发送)即滚动;(PS:默认是YES)
    2. touchesShouldCancelInContentView 开始发送 tracking messages 消息给 subview 的时候调用这个方法,决定是否发送 tracking messages 消息到subview。假如返回 NO,发送。YES 则不发送(PS:1和2配合使用)
    3. scorllView的属性:delaysContentTouches 是个布尔值,当值是 YES 的时候,用户触碰开始,scroll view要延迟一会,看看是否用户有意图滚动。假如滚动了,那么捕捉 touch-down 事件,否则就不捕捉。假如值是NO,当用户触碰, scroll view 会立即触发


    ###核心代码

    {% highlight objective-c %}

    //view是用户点击的视图
    - (BOOL)touchesShouldBegin:(NSSet *)touches withEvent:(UIEvent *)event inContentView:(UIView *)view
    {
    // 获取一个UITouch
    UITouch *touch = [touches anyObject];

    // 获取当前的位置
    CGPoint current = [touch locationInView:self];
    CGFloat x = [UIScreen mainScreen].bounds.size.width;
    if (current.x >= x + 10) {
    //在地图上
    NSLog(@"在地图上, 不滚动, view class is %@", view.class);
    return YES;
    } else {
    return [super touchesShouldBegin:touches withEvent:event inContentView:view];
    }
    }

    - (BOOL)touchesShouldCancelInContentView:(UIView *)view
    {
    NSLog(@"cancle class is %@", view.class);

    if ([view isKindOfClass:NSClassFromString(@"TapDetectingView")]) {
    return NO;
    } else {
    return [super touchesShouldCancelInContentView:view];
    }
    }


    {% endhighlight %}

    ###注意的问题

    1. 如果storyboard中使用的是系统自带的MKMapView,则不需要自定义scrollView
    2. 判断用户当前点击的视图时,点击BMKMapView对应的View类型是TapDetectingView,因此需要转换一下

    ###demo代码分享

    链接: http://pan.baidu.com/s/1o6vckjw 密码: vyx4
    目前尚无回复
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   实用小工具   ·   1262 人在线   最高记录 6679   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 27ms · UTC 23:38 · PVG 07:38 · LAX 15:38 · JFK 18:38
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.