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 "TicTacToeGame.h"
|
jens@0
|
24 |
#import "Grid.h"
|
jens@0
|
25 |
#import "Dispenser.h"
|
jens@0
|
26 |
#import "Piece.h"
|
jens@0
|
27 |
#import "QuartzUtils.h"
|
jens@0
|
28 |
|
jens@0
|
29 |
|
jens@0
|
30 |
@implementation TicTacToeGame
|
jens@0
|
31 |
|
jens@7
|
32 |
- (Piece*) pieceForPlayer: (int)playerNumber
|
jens@0
|
33 |
{
|
jens@7
|
34 |
Piece *p = [[Piece alloc] initWithImageNamed: (playerNumber ? @"O.tiff" :@"X.tiff")
|
jens@7
|
35 |
scale: 80];
|
jens@0
|
36 |
p.owner = [self.players objectAtIndex: playerNumber];
|
jens@7
|
37 |
p.name = (playerNumber ?@"O" :@"X");
|
jens@7
|
38 |
return [p autorelease];
|
jens@0
|
39 |
}
|
jens@0
|
40 |
|
jens@1
|
41 |
- (id) initWithBoard: (GGBLayer*)board
|
jens@0
|
42 |
{
|
jens@0
|
43 |
self = [super initWithBoard: board];
|
jens@0
|
44 |
if (self != nil) {
|
jens@0
|
45 |
[self setNumberOfPlayers: 2];
|
jens@0
|
46 |
|
jens@0
|
47 |
// Create a 3x3 grid:
|
jens@0
|
48 |
CGFloat center = floor(CGRectGetMidX(board.bounds));
|
jens@1
|
49 |
_grid = [[RectGrid alloc] initWithRows: 3 columns: 3 frame: CGRectMake(center-150,0, 300,300)];
|
jens@0
|
50 |
[_grid addAllCells];
|
jens@0
|
51 |
_grid.allowsMoves = _grid.allowsCaptures = NO;
|
jens@1
|
52 |
_grid.cellColor = CreateGray(1.0, 0.25);
|
jens@0
|
53 |
_grid.lineColor = kTranslucentLightGrayColor;
|
jens@0
|
54 |
[board addSublayer: _grid];
|
jens@0
|
55 |
|
jens@0
|
56 |
// Create piece dispensers for the two players:
|
jens@7
|
57 |
for( int playerNumber=0; playerNumber<=1; playerNumber++ ) {
|
jens@7
|
58 |
Piece *p = [self pieceForPlayer: playerNumber];
|
jens@7
|
59 |
CGFloat x = floor(CGRectGetMidX(_board.bounds));
|
jens@7
|
60 |
#if TARGET_OS_ASPEN
|
jens@7
|
61 |
x = x - 80 + 160*playerNumber;
|
jens@7
|
62 |
CGFloat y = 360;
|
jens@7
|
63 |
#else
|
jens@7
|
64 |
x += (playerNumber==0 ?-230 :230);
|
jens@7
|
65 |
CGFloat y = 175;
|
jens@7
|
66 |
#endif
|
jens@7
|
67 |
_dispenser[playerNumber] = [[Dispenser alloc] initWithPrototype: p quantity: 0
|
jens@7
|
68 |
frame: CGRectMake(x-45,y-45, 90,90)];
|
jens@7
|
69 |
[_board addSublayer: _dispenser[playerNumber]];
|
jens@7
|
70 |
}
|
jens@0
|
71 |
|
jens@0
|
72 |
// And they're off!
|
jens@0
|
73 |
[self nextPlayer];
|
jens@0
|
74 |
}
|
jens@0
|
75 |
return self;
|
jens@0
|
76 |
}
|
jens@0
|
77 |
|
jens@7
|
78 |
|
jens@7
|
79 |
- (NSString*) stateString
|
jens@7
|
80 |
{
|
jens@7
|
81 |
unichar str[10];
|
jens@7
|
82 |
for( int i=0; i<9; i++ ) {
|
jens@7
|
83 |
NSString *ident = [_grid cellAtRow: i/3 column: i%3].bit.name;
|
jens@7
|
84 |
if( ident==nil )
|
jens@7
|
85 |
str[i] = '-';
|
jens@7
|
86 |
else
|
jens@7
|
87 |
str[i] = [ident characterAtIndex: 0];
|
jens@7
|
88 |
}
|
jens@7
|
89 |
return [NSString stringWithCharacters: str length: 9];
|
jens@7
|
90 |
}
|
jens@7
|
91 |
|
jens@7
|
92 |
- (void) setStateString: (NSString*)stateString
|
jens@7
|
93 |
{
|
jens@7
|
94 |
for( int i=0; i<9; i++ ) {
|
jens@7
|
95 |
Piece *piece;
|
jens@7
|
96 |
switch( [stateString characterAtIndex: i] ) {
|
jens@7
|
97 |
case 'X': case 'x': piece = [self pieceForPlayer: 0]; break;
|
jens@7
|
98 |
case 'O': case 'o': piece = [self pieceForPlayer: 1]; break;
|
jens@7
|
99 |
default: piece = nil; break;
|
jens@7
|
100 |
}
|
jens@7
|
101 |
[_grid cellAtRow: i/3 column: i%3].bit = piece;
|
jens@7
|
102 |
}
|
jens@7
|
103 |
}
|
jens@7
|
104 |
|
jens@7
|
105 |
|
jens@3
|
106 |
- (Bit*) bitToPlaceInHolder: (id<BitHolder>)holder
|
jens@3
|
107 |
{
|
jens@3
|
108 |
if( holder.bit==nil && [holder isKindOfClass: [Square class]] )
|
jens@3
|
109 |
return _dispenser[self.currentPlayer.index].bit;
|
jens@3
|
110 |
else
|
jens@3
|
111 |
return nil;
|
jens@3
|
112 |
}
|
jens@3
|
113 |
|
jens@7
|
114 |
|
jens@7
|
115 |
- (void) bit: (Bit*)bit movedFrom: (id<BitHolder>)src to: (id<BitHolder>)dst
|
jens@7
|
116 |
{
|
jens@7
|
117 |
Square *square = (Square*)dst;
|
jens@7
|
118 |
int squareIndex = 3*square.row + square.column;
|
jens@7
|
119 |
[self.currentMove appendFormat: @"%@%i", bit.name, squareIndex];
|
jens@7
|
120 |
[super bit: bit movedFrom: src to: dst];
|
jens@7
|
121 |
}
|
jens@7
|
122 |
|
jens@0
|
123 |
- (void) nextPlayer
|
jens@0
|
124 |
{
|
jens@0
|
125 |
[super nextPlayer];
|
jens@0
|
126 |
// Give the next player another piece to put down:
|
jens@0
|
127 |
_dispenser[self.currentPlayer.index].quantity = 1;
|
jens@0
|
128 |
}
|
jens@0
|
129 |
|
jens@0
|
130 |
static Player* ownerAt( Grid *grid, int index )
|
jens@0
|
131 |
{
|
jens@0
|
132 |
return [grid cellAtRow: index/3 column: index%3].bit.owner;
|
jens@0
|
133 |
}
|
jens@0
|
134 |
|
jens@0
|
135 |
/** Should return the winning player, if the current position is a win. */
|
jens@0
|
136 |
- (Player*) checkForWinner
|
jens@0
|
137 |
{
|
jens@0
|
138 |
static const int kWinningTriples[8][3] = { {0,1,2}, {3,4,5}, {6,7,8}, // rows
|
jens@0
|
139 |
{0,3,6}, {1,4,7}, {2,5,8}, // cols
|
jens@0
|
140 |
{0,4,8}, {2,4,6} }; // diagonals
|
jens@0
|
141 |
for( int i=0; i<8; i++ ) {
|
jens@0
|
142 |
const int *triple = kWinningTriples[i];
|
jens@0
|
143 |
Player *p = ownerAt(_grid,triple[0]);
|
jens@0
|
144 |
if( p && p == ownerAt(_grid,triple[1]) && p == ownerAt(_grid,triple[2]) )
|
jens@0
|
145 |
return p;
|
jens@0
|
146 |
}
|
jens@0
|
147 |
return nil;
|
jens@0
|
148 |
}
|
jens@0
|
149 |
|
jens@0
|
150 |
@end
|