Приведение кастомного класса SKSpriteNode после столкновения

У меня есть несколько пользовательских классов, полученных из SKSpriteNode, и когда они сталкиваются, я хочу вызвать метод из пользовательского класса "Treat", но он падает при ударе с ошибкой. "Неожиданно обнаружен Nil при развертывании необязательного значения"

Это класс Лечить...

import UIKit
import SpriteKit

class Treat: SKSpriteNode {

var isActive: Bool!

override init(texture: SKTexture!, color: SKColor, size: CGSize) {
    self.isActive = true

    super.init(texture: texture, color: color, size: size)

    self.texture = SKTexture(imageNamed: "treat1")
    self.zPosition = 3
    self.size = CGSize(width: 25, height: 25)
    self.physicsBody = SKPhysicsBody(rectangleOf: self.size)

    self.physicsBody?.friction = 0.1
    self.physicsBody?.restitution = 0.8
    self.physicsBody?.mass = 0.01
    self.physicsBody?.affectedByGravity = true
    self.physicsBody?.allowsRotation = true
    self.physicsBody?.isDynamic = true
    self.physicsBody?.categoryBitMask = PhysicsCategory.Treat
    self.physicsBody?.contactTestBitMask = PhysicsCategory.Player
    self.physicsBody?.collisionBitMask = PhysicsCategory.Ground
    self.physicsBody?.usesPreciseCollisionDetection = true
    self.name = "Treat";

    let rotateDuration = Double(arc4random()) / 0xFFFFFFFF
    let rotateAction = SKAction.rotate(byAngle: CGFloat(M_PI), duration: rotateDuration)
    let rotateActionRepeatingForever = SKAction.repeatForever(rotateAction)
    self.run(rotateActionRepeatingForever)

}

convenience init(color: SKColor, isActive: Bool = false) {
    let size = CGSize(width:0, height: 0);
    self.init(texture:nil, color: color, size: size)
    self.isActive = isActive
}

required init?(coder aDecoder: NSCoder) {
    super.init(coder: aDecoder)
}

func throwTreat() {
    NSLog("Throwing Treat")
    let arc4randoMax:Double = 0x100000000
    let upper = 2.0
    let lower = 4.4
    let throwableAngle = Float32((Double(arc4random()) / arc4randoMax) * (upper - lower) + lower)

    self.physicsBody?.applyImpulse(CGVector(dx: CGFloat(-throwableAngle),dy: CGFloat(throwableAngle)))
}

func removeTreat(showEffect: Bool) {
    if self.isActive != false {
        self.isActive = false
        if showEffect != false {

        }
        self.removeFromParent()
    }

}

func onImpactWithPlayer() {
    self.removeFromParent()
}


}

И это физическое обнаружение столкновений..

 func didBegin(_ contact: SKPhysicsContact) {

    var firstBody: SKPhysicsBody
    var secondBody: SKPhysicsBody

    if contact.bodyA.categoryBitMask < contact.bodyB.categoryBitMask {
        firstBody = contact.bodyA
        secondBody = contact.bodyB
    } else {
        firstBody = contact.bodyB
        secondBody = contact.bodyA
    }

    if ((firstBody.categoryBitMask & PhysicsCategory.Player != 0) &&
        (secondBody.categoryBitMask & PhysicsCategory.Ground != 0)) {
        self.playerNode.isJumping = false
    }


    if ((firstBody.categoryBitMask & PhysicsCategory.Player != 0) &&
        (secondBody.categoryBitMask & PhysicsCategory.Treat != 0)) {
        levelScore = levelScore + 1
        (secondBody.node as! Treat).onImpactWithPlayer() // This line crashes

    }

}

Я новичок в стремительном и объектно-ориентированном программировании, так что на данный момент это не имеет особого смысла. Я знаю, это просто исправить, и я глупый...

1 ответ

Решение

Ваш код кажется в основном правильным.

То, что я считаю, происходит здесь

Когда сталкиваются 2 физических тела, didBegin(_ contact: SKPhysicsContact) может вызываться несколько раз в одном кадре.

Так как здесь

(secondBody.node as! Treat).onImpactWithPlayer()

Вы удаляете Лечить из графа сцены, в следующий раз didBegin(_ contact:) вызывается в том же кадре, вы получаете сбой, потому что узел был удален.

Решение

Просто замени это

(secondBody.node as! Treat).onImpactWithPlayer()

с этим

 (secondBody.node as? Treat)?.onImpactWithPlayer()
Другие вопросы по тегам