假设我有以下 View 层次结构;
这些 View 实际上是 ShinobiCharts,它们是 UIView 的子类。 View1 充当用户可以触摸的主图表(捏合、平移、长按等)。该 View 反过来控制其他 View (View2、View3 等)关于特定的、与触摸相关的属性(用户平移 View1,因此 View 2 和 3...必须相应地进行操作并平移)。
但是,一旦用户滚动 UIScrollView,View1 可能会从屏幕上消失,只留下 View 2、3 等可见,它们没有相应的手势识别器,因此用户无法再与图表交互,糟糕用户体验。
我当然也可以将识别器添加到这些附加图表中,但是在捏合期间,这意味着两个手指都必须位于单个 View 中,用户不能再触摸他想要的位置以进行捏合和缩放,糟糕的用户体验。
为了简短起见,我需要某种触摸拦截器来覆盖 UIScrollView 的整个内容区域,以便当用户捏合或平移时,相应的触摸将被转发到主图表(View1),这反过来可以更新其他 subview 。
UIScrollView 应该始终可以垂直滚动。
第一个实验:
我尝试向覆盖 UIScrollView 的 ViewController 添加一个透明的 UIView。
但是,即使直接引用 View1,触摸也不会被转发。
-(void) touchesBeganNSSet *)touches withEventUIEvent *)event {
[self.chartReference touchesBegan:touches withEvent:event];
}
我不太确定这是否是实现这一目标的正确方法。
第二次实验:
在 UIScrollView 中的图表 subview 上禁用 userInteraction 并将我自己的捏合和平移手势(在 UIViewController 中)添加到 UIScrollView。
问题;
1. UIScrollView 不再滚动。
2. 如何将这些手势转发给View1?
恐怕我现在无法提供更多示例代码,因为还没有太多相关的代码可以展示。
编辑:
实验二的小笔记;
手势识别器已添加如下;
UIPinchGestureRecognizer *pinchGestureRecognizer = [[UIPinchGestureRecognizer alloc] initWithTarget:self actionselector(pinchGesture];
pinchGestureRecognizer.cancelsTouchesInView = NO;
[scrollView addGestureRecognizer:pinchGestureRecognizer];
已启用 UIViewController 上的同时手势识别;
- (BOOL)gestureRecognizerUIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizerUIGestureRecognizer *)otherGestureRecognizer {
return YES;
}
编辑:
实验 2 的问题 2 已解决。我没有为平移/捏合手势设置正确的代表。哦!
剩下的就是如何转发捏/平移手势。
Best Answer-推荐答案 strong>
您可以在其他 View 之上放置一个透明 View 。但是,您需要覆盖另外一种方法。
您想在透明 View 上覆盖 hitTest:withEvent。在遍历响应者链以查看哪个 View 处理特定区域中的触摸时使用此方法。如果 View 处理了该触摸,它会返回它自己。如果它想将它传递给它下面的下一个 View ,它会返回 nil。如果它知道另一个 View 处理该触摸,它可以返回该 View 。
因此,在您的情况下,如果该点位于顶部透明 View 的目标区域中,则返回 view1。然后应该调用 View1 的手势识别器。
示例:
InterceptorView 是一个位于 scrollView 顶部的透明 View 。 TargetView 是 scrollView 内部的一个 View ,并附加了一个 TapGestureRecognizer。
class InterceptorView: UIView {
required init(coder aDecoder: NSCoder) {
super.init(coder: aDecoder)
}
@IBOutlet weak var targetView1: UIView?
override func hitTest(point: CGPoint, withEvent event: UIEvent?) -> UIView? {
print("[Interceptor] Testing point: \(point) ")
if self.pointInside(point, withEvent: event) {
println("Hit")
return targetView1
}
else {
println()
return nil;
}
}
}
--
class TargetView: UIView {
required init(coder aDecoder: NSCoder) {
super.init(coder: aDecoder)
}
@IBAction func handleGesture(gestureRecognizer: UIGestureRecognizer) {
let location = gestureRecognizer.locationInView(self)
print("[TargetView] Got gesture. Location \(location) ")
if (pointInside(location, withEvent: nil)) {
println("Inside");
}
else {
println("Outside");
}
}
}
项目如下:
https://github.com/annabd351/GestureForwarding
(里面还有一些其他的东西,但它有效!)
关于ios - UIScrollView 与 contentSize 的触摸拦截器来控制 subview ,我们在Stack Overflow上找到一个类似的问题:
https://stackoverflow.com/questions/27060966/
|