Source/Piece.m
author Jens Alfke <jens@mooseyard.com>
Fri Mar 07 11:43:02 2008 -0800 (2008-03-07)
changeset 0 e9f7ba4718e1
child 1 3eb7be1dd7b6
permissions -rw-r--r--
Initial check-in into Mercurial. Branched from 1.0 release of Apple's sample code. No longer requires garbage collection. Fixed some memory leaks of CG objects. Fixed a bug when advancing to the 8th row in the Checkers game.
jens@0
     1
/*  This code is based on Apple's "GeekGameBoard" sample code, version 1.0.
jens@0
     2
    http://developer.apple.com/samplecode/GeekGameBoard/
jens@0
     3
    Copyright © 2007 Apple Inc. Copyright © 2008 Jens Alfke. All Rights Reserved.
jens@0
     4
jens@0
     5
    Redistribution and use in source and binary forms, with or without modification, are permitted
jens@0
     6
    provided that the following conditions are met:
jens@0
     7
jens@0
     8
    * Redistributions of source code must retain the above copyright notice, this list of conditions
jens@0
     9
      and the following disclaimer.
jens@0
    10
    * Redistributions in binary form must reproduce the above copyright notice, this list of
jens@0
    11
      conditions and the following disclaimer in the documentation and/or other materials provided
jens@0
    12
      with the distribution.
jens@0
    13
jens@0
    14
    THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR
jens@0
    15
    IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND 
jens@0
    16
    FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRI-
jens@0
    17
    BUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
jens@0
    18
    (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR 
jens@0
    19
    PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 
jens@0
    20
    CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF 
jens@0
    21
    THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
jens@0
    22
*/
jens@0
    23
#import "Piece.h"
jens@0
    24
#import "QuartzUtils.h"
jens@0
    25
jens@0
    26
jens@0
    27
@implementation Piece
jens@0
    28
jens@0
    29
jens@0
    30
- (id) initWithImageNamed: (NSString*)imageName
jens@0
    31
                    scale: (CGFloat)scale
jens@0
    32
{
jens@0
    33
    self = [super init];
jens@0
    34
    if (self != nil) {
jens@0
    35
        self.imageName = imageName;
jens@0
    36
        [self setImage: GetCGImageNamed(imageName) scale: scale];
jens@0
    37
        self.zPosition = kPieceZ;
jens@0
    38
    }
jens@0
    39
    return self;
jens@0
    40
}
jens@0
    41
jens@0
    42
jens@0
    43
- (id) initWithCoder: (NSCoder*)decoder
jens@0
    44
{
jens@0
    45
    self = [super initWithCoder: decoder];
jens@0
    46
    if( self ) {
jens@0
    47
        self.imageName = [decoder decodeObjectForKey: @"imageName"];
jens@0
    48
        // (actual image (self.contents) was already restord by superclass)
jens@0
    49
    }
jens@0
    50
    return self;
jens@0
    51
}
jens@0
    52
jens@0
    53
- (void) encodeWithCoder: (NSCoder*)coder
jens@0
    54
{
jens@0
    55
    [super encodeWithCoder: coder];
jens@0
    56
    [coder encodeObject: _imageName forKey: @"imageName"];
jens@0
    57
}
jens@0
    58
jens@0
    59
jens@0
    60
- (void) dealloc
jens@0
    61
{
jens@0
    62
    [_imageName release];
jens@0
    63
    [super dealloc];
jens@0
    64
}
jens@0
    65
jens@0
    66
jens@0
    67
- (NSString*) description
jens@0
    68
{
jens@0
    69
    return [NSString stringWithFormat: @"%@[%@]", 
jens@0
    70
            [self class],
jens@0
    71
            _imageName.lastPathComponent.stringByDeletingPathExtension];
jens@0
    72
}
jens@0
    73
jens@0
    74
jens@0
    75
@synthesize imageName=_imageName;
jens@0
    76
jens@0
    77
jens@0
    78
- (void) setImage: (CGImageRef)image scale: (CGFloat)scale
jens@0
    79
{
jens@0
    80
    self.contents = (id) image;
jens@0
    81
    self.contentsGravity = @"resize";
jens@0
    82
    self.minificationFilter = kCAFilterLinear;
jens@0
    83
    int width = CGImageGetWidth(image), height = CGImageGetHeight(image);
jens@0
    84
    if( scale > 0 ) {
jens@0
    85
        if( scale >= 4.0 )
jens@0
    86
            scale /= MAX(width,height);             // interpret scale as target dimensions
jens@0
    87
        width = ceil( width * scale);
jens@0
    88
        height= ceil( height* scale);
jens@0
    89
    }
jens@0
    90
    self.bounds = CGRectMake(0,0,width,height);
jens@0
    91
    self.imageName = nil;
jens@0
    92
}
jens@0
    93
jens@0
    94
- (void) setImage: (CGImageRef)image
jens@0
    95
{
jens@0
    96
    CGSize size = self.bounds.size;
jens@0
    97
    [self setImage: image scale: MAX(size.width,size.height)];
jens@0
    98
}
jens@0
    99
jens@0
   100
- (void) setImageNamed: (NSString*)name
jens@0
   101
{
jens@0
   102
    [self setImage: GetCGImageNamed(name)];
jens@0
   103
    self.imageName = name;
jens@0
   104
}
jens@0
   105
jens@0
   106
jens@0
   107
- (BOOL)containsPoint:(CGPoint)p
jens@0
   108
{
jens@0
   109
    // Overrides CGLayer's implementation,
jens@0
   110
    // returning YES only for pixels at which this layer's alpha is at least 0.5.
jens@0
   111
    // This takes into account the opacity, bg color, and background image's alpha channel.
jens@0
   112
    if( ! [super containsPoint: p] )
jens@0
   113
        return NO;
jens@0
   114
    float opacity = self.opacity;
jens@0
   115
    if( opacity < 0.5 )
jens@0
   116
        return NO;
jens@0
   117
    float thresholdAlpha = 0.5 / self.opacity;
jens@0
   118
    
jens@0
   119
    CGColorRef bg = self.backgroundColor;
jens@0
   120
    float alpha = bg ?CGColorGetAlpha(bg) :0.0;
jens@0
   121
    if( alpha < thresholdAlpha ) {
jens@0
   122
        CGImageRef image = (CGImageRef)self.contents;
jens@0
   123
        if( image ) {
jens@0
   124
            // Note: This makes the convenient assumption that the image entirely fills the bounds.
jens@0
   125
            alpha = MAX(alpha, GetPixelAlpha(image, self.bounds.size, p));
jens@0
   126
        }
jens@0
   127
    }
jens@0
   128
    return alpha >= thresholdAlpha;
jens@0
   129
}
jens@0
   130
jens@0
   131
jens@0
   132
@end