首页 > 解决方案 > 命中测试以确定哪个 CGPoint 最接近触摸 - 1 个 UIView 中的 16 个 CGPoint 点

问题描述

我一直在修改 LMinh 的这个名为 LMGaugeView 的自定义 UIView,以使其看起来像一个 16 瓶圆形“瓶转盘”。

想象一下 16 个点 (CGPoints) 均匀分布在圆 (UIView) 的边缘。我希望能够执行以下方案:

小瓶旋转木马

图片显示了 10 个小瓶,但你明白了。一旦我触摸圆形视图,我希望能够仅根据它们的 CGPoint 值来确定我点击了哪个“小瓶”。

我创建了一个需要该方法的应用程序(称为 Twinstones,只是为了把它扔出去)hitTest:withEvent:,但我正在处理可以触摸的 2 个子视图(在它们的父视图的框架内。)

为此,圆圈是唯一的视图(这意味着hitTest:withEvent:每次我接触它时都会返回圆圈视图。)

这是那个hitTest:...实现:

- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {
    if (!self.isUserInteractionEnabled || self.isHidden || self.alpha <= 0.01) {
        return nil;
    }

    CGRect touchRect = CGRectInset(self.bounds, -14, -14);
    if (CGRectContainsPoint(touchRect, point)) {
       for (UIView *subview in [self.subviews reverseObjectEnumerator]) {
            CGPoint convertedPoint = [subview convertPoint:point fromView:self];
            UIView *hitTestView = [subview hitTest:convertedPoint withEvent:event];
            if (hitTestView) {
                return hitTestView;
            }
        }
        return self;
    }
    return nil;
}

我需要使用另一种与 hitTest 相关的方法来使其正常工作吗?如果您需要查看更多代码,请告诉我。

标签: iosobjective-cuitouchhittestcgpoint

解决方案


勾股定理在这里很有用。您可以获取用户触摸屏幕的点,然后使用 map() 计算到每个小瓶的距离并找到最小值:

let p1 = //where your user touched the view
let vialDistances = vials.map { // your vials array
    let p2 = // vial position
    let diffX = p1.x - p2.x
    let diffY = p1.y - p2.y
    return diffX * diffX + diffY * diffY
}
let index = find(vialDistances, vialDistances.min())
let closestVial = vials[index]

推荐阅读