Detecting Touch on SKShapeNode that is a Line

2019-09-09 08:05发布

问题:

I have an SKShapeNode that I have created and given a CGPath to. This is in my GameScene.swift in didMoveToView:

let myNode = SKShapeNode()
let lineToDraw = CGPathCreateMutable()
CGPathMoveToPoint(lineToDraw, nil, 0, 0)
CGPathAddLineToPoint(lineToDraw, nil, 87, 120)
myNode.path = lineToDraw
myNode.strokeColor = SKColor.redColor()
myNode.lineWidth = 20
myNode.name = "My Node!"
world.addChild(myNode) // World is a higher-level node in my scene

And this is how I'm detecting touches, again in the Scene:

override func touchesEnded(touches: NSSet!, withEvent event: UIEvent!) {
    if let touch = touches.anyObject() as? UITouch {
        if let shapeNode = nodeAtPoint(touch.locationInNode(self)) as? SKShapeNode {
            println("Touched \(shapeNode.name)")
        } else {
            println("Nothing here")
        }
    }
}

The line node is showing up with no issues. However, it is not registering touchesEnded at all.

Really, I have three questions nebulous to this:

  1. Is there a better way to create an SKShapeNode that is a line between two points without calling a convenience initializer? I plan on subclassing SKShapeNode to add additional logic to the node, and the subclass doesn't have access to convenience initializers of the superclass.

  2. More importantly, how do I get the scene to recognize my line node there and trigger touchesEnded?

  3. Is there a way, using that mechanism, I can make it a bit more "fuzzy", so it handles touches close to the line, instead of on the line? My eventual plan is to make it thinner, but I'd still like to have a large touch zone. I figure if nothing else, I can create two nodes: one clear/thick and the other red/thin, and handle touch events on the clear one, but I would like to know if there's a better way.

回答1:

In answer to your main question, there is a problem in your touchesEnded method. The template shown by apple recommends the following layout:

override func touchesEnded(touches: NSSet, withEvent event: UIEvent) {
        for touch: AnyObject in touches {
            let location = touch.locationInNode(self)
    }
}

You can then use this method to see if the location value is inside the lines frame (in order to do this, i created the "myNode" variable outside of the didMoveToView so that i could access it from the rest of the functions):

override func touchesEnded(touches: NSSet, withEvent event: UIEvent) {
        for touch: AnyObject in touches {
            let location = touch.locationInNode(world)

            if CGRectContainsPoint(myNode.frame, location) {
                println("Touched \(myNode.name)")

            } else {

                println("Nothing here")
        }

    }
}

As for a "fuzzier" line, you can simply check a larger CGRect. In the code above, i check myNode.frame but you could create a variable with a CGRect which is slightly larger than the line so that you can detect touches which don't directly hit it.

As for more concise code, i cannot think of any at the moment but that isn't to say that there isn't a way. However, I don't quite understand what you mean about subclasses not having access to convenience methods as they can have access to whatever the superclass does so long as you import correctly.

I hope this helps.