I have a helper method which extends SKPhysicsContact
extension SKPhysicsContact {
/// - returns: `[SKPhysicsBody]` containing all the bodies that match `mask`
func bodiesMatchingCategory(mask: UInt32) -> [SKPhysicsBody] {
let bodies = [bodyA, bodyB]
return bodies.filter { ($0.categoryBitMask & mask) != 0 }
}
}
in didBeginContact()
I call this method on the passed in contact
.
func didBeginContact(contact: SKPhysicsContact) {
let ballMask: UInt32 = 0x1 << 2
let ball = contact.bodiesMatchingCategory(ballMask)
...
I get this error message sometimes (like 1 in 5) which crashes the app:
-[PKPhysicsContact bodiesMatchingCategory:]: unrecognized selector sent to instance 0x165f2350
I looked up PKPhysicsContact
and it's part of a private framework (link). SKPhysicsContact
looks like it's just an empty class definition which exposes only certain properties of PKPhysicsContact
.
I feel like this is an Objective-C hack on the SpriteKit team that breaks Swift's strong typing.
Help?
How to make sure I always get SKPhysicsContact
back?
I added a check to test for SKPhysicsContact
let test = contact as Any
print("Test is: \(test)")
guard test is SKPhysicsContact else {
return
}
Which correctly catches the type mis-match.
In fact, it NEVER returns a SKPhysicsContact
!!?
I've tried doing this in Objective-C (as suggested by responder) and I'm getting the same result.
I have a discussion on the Apple Dev Forums which may provide future answer-seekers some help.
Here's the Objective-C code for reference:
@interface SKPhysicsContact (MatchingBodies)
- (NSArray *)bodiesMatchingCategory:(UInt32)category;
@end
@implementation SKPhysicsContact (MatchingBodies)
- (NSArray *)bodiesMatchingCategory:(UInt32)category {
NSArray *bodies = @[self.bodyA, self.bodyB];
NSPredicate *predicate = [NSPredicate predicateWithBlock:^BOOL(SKPhysicsBody *body, NSDictionary *bindings) {
return (body.categoryBitMask & category) != 0;
}];
NSArray *matching = [bodies filteredArrayUsingPredicate:predicate];
return matching;
}
@end
Called Here:
-(void)didBeginContact:(SKPhysicsContact *)contact
{
static const uint32_t MarbleContact = 0x1 <<1; // 2
static const uint32_t GoalContact = 0x1 <<2; // 4
SKPhysicsBody *ball = [contact bodiesMatchingCategory:MarbleContact].firstObject;
NSLog(@"Ball: %@", ball);
...
Returns this crash:
-[PKPhysicsContact bodiesMatchingCategory:]: unrecognized selector sent to instance 0x17dad9e0
*** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[PKPhysicsContact bodiesMatchingCategory:]: unrecognized selector sent to instance 0x17dad9e0'
Added Bug Report for Apple, #23332190
I get the same error with this simple code:
extension SKPhysicsContact {
func bodiesMatchingCategory(mask: UInt32) -> [SKPhysicsBody] {
let bodies = [bodyA, bodyB]
return bodies.filter { ($0.categoryBitMask & mask) != 0 }
}
}
let contact = SKPhysicsContact()
let body = contact.bodiesMatchingCategory(0)
The problem is, the type of contact
is PKPhysicsContact
(as you've noticed), even when you explicitly tell it to be an SKPhysicsContact
, and the extension is on SKPhysicsContact
. You'd have to be able to make an extension to PKPhysicsContact
for this to work. From this logic, we can say that no instance methods will work in SKPhysicsContact
extensions at the moment. I'd say it's a bug with SpriteKit, and you should file a radar. Class methods still work since you call them on the class itself.
In the meantime, you should be able to move that method into your scene or another object and call it there successfully.
For the record, this is not a Swift-specific problem. If you make the same method in an Objective-C category on SKPhysicsContact
you'll get the same crash.
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With