Физика SpriteKit Collisions/APLSpaceScene.m

/*
     File: APLSpaceScene.m
 Abstract: 
 This is the scene that implements the physics demo. It is responsible for handling keyboard inputs and driving the simulation in response to user input and physics interactions.
 
  Version: 1.2
 
 Disclaimer: IMPORTANT:  This Apple software is supplied to you by Apple
 Inc. ("Apple") in consideration of your agreement to the following
 terms, and your use, installation, modification or redistribution of
 this Apple software constitutes acceptance of these terms.  If you do
 not agree with these terms, please do not use, install, modify or
 redistribute this Apple software.
 
 In consideration of your agreement to abide by the following terms, and
 subject to these terms, Apple grants you a personal, non-exclusive
 license, under Apple's copyrights in this original Apple software (the
 "Apple Software"), to use, reproduce, modify and redistribute the Apple
 Software, with or without modifications, in source and/or binary forms;
 provided that if you redistribute the Apple Software in its entirety and
 without modifications, you must retain this notice and the following
 text and disclaimers in all such redistributions of the Apple Software.
 Neither the name, trademarks, service marks or logos of Apple Inc. may
 be used to endorse or promote products derived from the Apple Software
 without specific prior written permission from Apple.  Except as
 expressly stated in this notice, no other rights or licenses, express or
 implied, are granted by Apple herein, including but not limited to any
 patent rights that may be infringed by your derivative works or by other
 works in which the Apple Software may be incorporated.
 
 The Apple Software is provided by Apple on an "AS IS" basis.  APPLE
 MAKES NO WARRANTIES, EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION
 THE IMPLIED WARRANTIES OF NON-INFRINGEMENT, MERCHANTABILITY AND FITNESS
 FOR A PARTICULAR PURPOSE, REGARDING THE APPLE SOFTWARE OR ITS USE AND
 OPERATION ALONE OR IN COMBINATION WITH YOUR PRODUCTS.
 
 IN NO EVENT SHALL APPLE BE LIABLE FOR ANY SPECIAL, INDIRECT, INCIDENTAL
 OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 INTERRUPTION) ARISING IN ANY WAY OUT OF THE USE, REPRODUCTION,
 MODIFICATION AND/OR DISTRIBUTION OF THE APPLE SOFTWARE, HOWEVER CAUSED
 AND WHETHER UNDER THEORY OF CONTRACT, TORT (INCLUDING NEGLIGENCE),
 STRICT LIABILITY OR OTHERWISE, EVEN IF APPLE HAS BEEN ADVISED OF THE
 POSSIBILITY OF SUCH DAMAGE.
 
 Copyright (C) 2014 Apple Inc. All Rights Reserved.
 
 */
 
#import "APLSpaceScene.h"
#import "APLShipSprite.h"
 
 
@interface APLSpaceScene ()
@property BOOL contentCreated;
@property APLShipSprite *controlledShip;
@end
 
// Useful randomizer functions.
static inline CGFloat myRandf() {
    return rand() / (CGFloat) RAND_MAX;
}
 
static inline CGFloat myRand(CGFloat low, CGFloat high) {
    return myRandf() * (high - low) + low;
}
 
/* Simulation constants used to tweak  game play. */
 
// sizes for the various kinds of objects
static const CGFloat shotSize = 4;
static const CGFloat asteroidSize = 18;
static const CGFloat planetSize = 128;
 
// explosion constants
static const CFTimeInterval missileExplosionDuration = 0.1;
 
// collison constants
static const CGFloat collisonDamageThreshold = 3.0;
 
// missile constants
static const NSInteger missileDamage = 1;
 
@implementation APLSpaceScene
 
#pragma mark Initialization
 
-(id)initWithSize:(CGSize)size {    
    if (self = [super initWithSize:size]) {
    
    }
    return self;
}
 
- (void)didMoveToView:(SKView *)view
{
    if (!self.contentCreated)
    {
        [self createSceneContents];
        self.contentCreated = YES;
    }
}
 
- (void)createSceneContents
{
    self.backgroundColor = [SKColor blackColor];
    self.scaleMode = SKSceneScaleModeAspectFit;
    
    // Give the scene an edge and configure other physics info on the scene.
    self.physicsBody = [SKPhysicsBody bodyWithEdgeLoopFromRect:self.frame];
    self.physicsBody.categoryBitMask = edgeCategory;
    self.physicsBody.collisionBitMask = 0;
    self.physicsBody.contactTestBitMask = 0;
    self.physicsWorld.gravity = CGVectorMake(0,0);
    self.physicsWorld.contactDelegate = self;
    
    /* 
     In this sample, the positions of everything is hard coded. In an actual game, you might implement this in an archive that is loaded from a file.
     */
    self.controlledShip = [APLShipSprite shipSprite];
    self.controlledShip.position = CGPointMake (100,500);
    [self addChild:self.controlledShip];
    
    // this ship isn't connected to any controls so it doesn't move, except when it collides with something.
    SKNode *targetShip = [APLShipSprite shipSprite];
    targetShip.position = CGPointMake(500,500);
    [self addChild:targetShip];
    
    SKNode *rock = [self newAsteroidNode];
    rock.position = CGPointMake(100,200);
    [self addChild:rock];
    
    SKNode *planet = [self newPlanetNode];
    planet.position = CGPointMake(500,100);
    [self addChild:planet];
}
 
 
- (SKNode*) newMissileNode
{
    /*
     Creates and returns a new missile game object.
     This method loads a preconfigured emitter from an archive, and then configures it with a physics body.
     */
    SKEmitterNode *missile =  [NSKeyedUnarchiver unarchiveObjectWithFile:[[NSBundle mainBundle] pathForResource:@"missile" ofType:@"sks"]];
    
    // The missile particles should be spawned in the scene, not on the missile object.
    missile.targetNode = self;
    
    missile.physicsBody = [SKPhysicsBody bodyWithCircleOfRadius:shotSize];
    missile.physicsBody.categoryBitMask = missileCategory;
    missile.physicsBody.contactTestBitMask = shipCategory | asteroidCategory | planetCategory | edgeCategory;
    missile.physicsBody.collisionBitMask = 0;
    
    return missile;
}
 
- (SKNode*) newAsteroidNode
{
    /* Creates and returns a new asteroid game object.
     
     For this sample, we just use a shape node for the asteroid.
     */
    SKShapeNode *asteroid = [[SKShapeNode alloc] init];
    CGMutablePathRef myPath = CGPathCreateMutable();
    CGPathAddArc(myPath, NULL, 0,0, asteroidSize, 0, M_PI*2, YES);
    asteroid.path = myPath;
    CGPathRelease(myPath);
    asteroid.strokeColor = [SKColor clearColor];
    asteroid.fillColor = [SKColor brownColor];
    
    asteroid.physicsBody = [SKPhysicsBody bodyWithCircleOfRadius:asteroidSize];
    asteroid.physicsBody.categoryBitMask = asteroidCategory;
    asteroid.physicsBody.collisionBitMask = shipCategory | asteroidCategory | edgeCategory;
    asteroid.physicsBody.contactTestBitMask = planetCategory;
    
    return asteroid;
}
 
- (SKNode*) newPlanetNode
{
    /* Creates and returns a new planet game object.
     
     For this sample, we just use a shape node for the planet.
     */
    
    SKShapeNode *planet = [[SKShapeNode alloc] init];
    CGMutablePathRef myPath = CGPathCreateMutable();
    CGPathAddArc(myPath, NULL, 0,0, planetSize, 0, M_PI*2, YES);
    planet.path = myPath;
    CGPathRelease(myPath);
    planet.strokeColor = [SKColor clearColor];
    planet.fillColor = [SKColor greenColor];
    
    planet.physicsBody = [SKPhysicsBody bodyWithCircleOfRadius:planetSize];
    planet.physicsBody.categoryBitMask = planetCategory;
    planet.physicsBody.collisionBitMask = planetCategory | edgeCategory;
    planet.physicsBody.contactTestBitMask = 0;
    
    return planet;
}
 
- (SKEmitterNode*) newExplosionNode: (CFTimeInterval) explosionDuration
{
    SKEmitterNode *emitter =  [NSKeyedUnarchiver unarchiveObjectWithFile:[[NSBundle mainBundle] pathForResource:@"explosion" ofType:@"sks"]];
    
    // Explosions always place their particles into the scene.
    emitter.targetNode = self;
    
    // Stop spawning particles after enough have been spawned.
    emitter.numParticlesToEmit = explosionDuration * emitter.particleBirthRate;
    
    // Calculate a time value that allows all the spawned particles to die. After this, the emitter node can be removed.
 
    CFTimeInterval totalTime = explosionDuration + emitter.particleLifetime+emitter.particleLifetimeRange/2;
    [emitter runAction:[SKAction sequence:@[[SKAction waitForDuration:totalTime],
                                            [SKAction removeFromParent]]]];
    return emitter;
}
 
#pragma mark Physics Handling and Game Logic
 
- (void)detonateMissile:(SKNode *)missile
{
    SKEmitterNode *explosion = [self newExplosionNode: missileExplosionDuration];
    explosion.position = missile.position;
    [self addChild:explosion];
    [missile removeFromParent];
}
 
- (void) attackTarget: (SKPhysicsBody*) target withMissile: (SKNode*) missile
{
    // Only ships take damage from missiles.
    if ((target.categoryBitMask & shipCategory) != 0)
    {
        APLShipSprite *targetShip = (APLShipSprite*) target.node;
        [targetShip applyDamage:missileDamage];
    }
    
    [self detonateMissile:missile];
}
 
- (void)didBeginContact:(SKPhysicsContact *)contact
{
    // Handle contacts between two physics bodies.
    
    // Contacts are often a double dispatch problem; the effect you want is based
    // on the type of both bodies in the contact. This sample  solves
    // this in a brute force way, by checking the types of each. A more complicated
    // example might use methods on objects to perform the type checking.
    
    SKPhysicsBody *firstBody;
    SKPhysicsBody *secondBody;
 
    // The contacts can appear in either order, and so normally you'd need to check
    // each against the other. In this example, the category types are well ordered, so
    // the code swaps the two bodies if they are out of order. This allows the code
    // to only test collisions once.
    
    if (contact.bodyA.categoryBitMask < contact.bodyB.categoryBitMask)
    {
        firstBody = contact.bodyA;
        secondBody = contact.bodyB;
    }
    else
    {
        firstBody = contact.bodyB;
        secondBody = contact.bodyA;
    }
    
    // Missiles attack whatever they hit, then explode.
    
    if ((firstBody.categoryBitMask & missileCategory) != 0)
    {
        [self attackTarget: secondBody withMissile:firstBody.node];
    }
    
    // Ships collide and take damage. The collision damage is based on the strength of the collision.
    if ((firstBody.categoryBitMask & shipCategory) != 0)
    {
        // The edge exists just to keep all gameplay on one screen, so ships should not take damage when they hit the
        // edge.
        
        if ((contact.collisionImpulse > collisonDamageThreshold) && ((secondBody.categoryBitMask & edgeCategory) == 0))
        {
            APLShipSprite *targetShip = (APLShipSprite*)firstBody.node;
            [targetShip applyDamage:contact.collisionImpulse / collisonDamageThreshold];
            
            // If two ships collide with each other, both take damage. Planets and asteroids take no damage from ships.
            if (secondBody.categoryBitMask & shipCategory)
            {
                targetShip = (APLShipSprite*)secondBody.node;
                [targetShip applyDamage:contact.collisionImpulse / collisonDamageThreshold];
            }
        }
    }
    
    // Asteroids that hit planets are destroyed.
    if (((firstBody.categoryBitMask & asteroidCategory) != 0) &&
        ((secondBody.categoryBitMask & planetCategory) != 0))
    {
        [firstBody.node removeFromParent];
    }
}
 
 
#pragma mark - Controls and Control Logic
 
 
 
- (void)update:(NSTimeInterval)currentTime
{
    // This runs once every frame. Other sorts of logic might run from here. For example,
    // if the target ship was controlled by the computer, you might run AI from this routine.
    
    [self updatePlayerShip:currentTime];
}
 
- (void)updatePlayerShip:(NSTimeInterval)currentTime
{
    /* 
     Use the stored key information to control the ship.
     */
    
    if (actions[kPlayerForward])
    {
        [self.controlledShip activateMainEngine];
    }
    else
    {
        [self.controlledShip deactivateMainEngine];
    }
    
    if (actions[kPlayerBack])
    {
        [self.controlledShip reverseThrust];
    }
    
    if (actions[kPlayerLeft])
    {
        [self.controlledShip rotateShipLeft];
    }
    
    if (actions[kPlayerRight])
    {
        [self.controlledShip rotateShipRight];
    }
    
    if (actions[kPlayerAction])
    {
        [self.controlledShip attemptMissileLaunch:currentTime];
    }
}
 
 
- (void)keyDown:(NSEvent *)theEvent
{
    /*
     Convert key down events into game actions
     */
     
    // first we check the arrow keys since they are on the numeric keypad
    if ([theEvent modifierFlags] & NSNumericPadKeyMask)
    { // arrow keys have this mask
        NSString *theArrow = [theEvent charactersIgnoringModifiers];
        unichar keyChar = 0;
        if ( [theArrow length] == 1 ) {
            keyChar = [theArrow characterAtIndex:0];
            switch (keyChar) {
                case NSLeftArrowFunctionKey:
                    actions[kPlayerLeft] = YES;
                    break;
                case NSRightArrowFunctionKey:
                    actions[kPlayerRight] = YES;
                    break;
                case NSUpArrowFunctionKey:
                    actions[kPlayerForward] = YES;
                    break;
                case NSDownArrowFunctionKey:
                    actions[kPlayerBack] = YES;
                    break;
            }
        }
    }
    
    // and now we check the keyboard
    NSString *characters = [theEvent characters];
    if ([characters length]) {
        for (int s = 0; s<[characters length]; s++) {
            unichar character = [characters characterAtIndex:s];
            switch (character) {
                case 'w':
                    actions[kPlayerForward] = YES;
                    break;
                case 'a':
                    actions[kPlayerLeft] = YES;
                    break;
                case 'd':
                    actions[kPlayerRight] = YES;
                    break;
                case 's':
                    actions[kPlayerBack] = YES;
                    break;
                case ' ':
                    actions[kPlayerAction] = YES;
                    break;
                case 'r':
                {
                    APLSpaceScene *reset = [[APLSpaceScene alloc] initWithSize: self.frame.size];
                    [self.view presentScene:reset transition:[SKTransition flipVerticalWithDuration:0.35]];
                }
                    break;
            }
        }
    }
}
 
- (void)keyUp:(NSEvent *)theEvent
{
    /*
     Convert key up events into game actions
     */
    
    if ([theEvent modifierFlags] & NSNumericPadKeyMask)
    { 
        NSString *theArrow = [theEvent charactersIgnoringModifiers];
        unichar keyChar = 0;
        if ( [theArrow length] == 1 ) {
            keyChar = [theArrow characterAtIndex:0];
            switch (keyChar) {
                case NSLeftArrowFunctionKey:
                    actions[kPlayerLeft] = NO;
                    break;
                case NSRightArrowFunctionKey:
                    actions[kPlayerRight] = NO;
                    break;
                case NSUpArrowFunctionKey:
                    actions[kPlayerForward] = NO;
                    break;
                case NSDownArrowFunctionKey:
                    actions[kPlayerBack] = NO;
                    break;
            }
        }
    }
    
    NSString *characters = [theEvent characters];
    if ([characters length]) {
        for (int s = 0; s<[characters length]; s++) {
            unichar character = [characters characterAtIndex:s];
            switch (character) {
                case 'w':
                    actions[kPlayerForward] = NO;
                    break;
                case 'a':
                    actions[kPlayerLeft] = NO;
                    break;
                case 'd':
                    actions[kPlayerRight] = NO;
                    break;
                case 's':
                    actions[kPlayerBack] = NO;
                    break;
                case ' ':
                    actions[kPlayerAction] = NO;
                    break;
            }
        }
    }
}
 
 
@end