iOS-Swift点击位置判断和坐标转换

image.png

当前View里面有两个View,绿色的bigView和红色的smallView,smallView在bigView里面。现在要在当前View的touchesBegan方法判断点击位置是否在最里层的smallView里。

有多种方法可以判断。其实它们本质都是一样的,最重要的是转换参考系。

  • 要想判断point是否在frame里面,必须保证它们是同一个参考系。
  • frame的坐标是view在它的父view参考系里的坐标。
  • touch.location(in: target)方法可以获取点击位置在target参考系里的坐标。target可以是屏幕上任意的view。
  • view.convert(point, to: target)方法可以将:point在view里的坐标转换成 -> point在另一个target里的坐标。这两个view可以是屏幕上任意的view。
class OurView: UIView {
    
    lazy var bigView = UIView(frame: CGRect(x: 100, y: 200, width: 200, height: 200))
    
    lazy var smallView = UIView(frame: CGRect(x: 100, y: 100, width: 100, height: 100))
    
    override init(frame: CGRect) {
        super.init(frame: frame)
        backgroundColor = .white
        
        bigView.backgroundColor = .green
        addSubview(bigView)
        
        smallView.backgroundColor = .red
        bigView.addSubview(smallView)
    }
    
    required init?(coder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
    
    override func touchesBegan(_ touches: Set<UITouch>, with event: UIEvent?) {
        for touch in touches {
            // 方法1:用bigView参考系
            let point = touch.location(in: bigView)
            let contains = smallView.frame.contains(point)
            print("smallView touches: \(contains)")

            // 方法2:用smallView参考系
            let point = touch.location(in: smallView)
            let contains = smallView.bounds.contains(point)
            print("smallView touches: \(contains)")

            // 方法3:获取点击位置在self参考系坐标,用convert方法转换成bigView参考系坐标
            var point = touch.location(in: self)
            point = self.convert(point, to: bigView)
            let contains = smallView.frame.contains(point)
            print("smallView touches: \(contains)")

            // 方法4:获取点击位置在self参考系坐标,手动转换成bigView参考系坐标
            var point = touch.location(in: self)
            point.x -= bigView.frame.minX
            point.y -= bigView.frame.minY
            let contains = smallView.frame.contains(point)
            print("smallView touches: \(contains)")

            break
        }
    }
}
posted @ 2022-07-18 17:46  rome753  阅读(775)  评论(0编辑  收藏  举报