Text, playing cards, and Klondike solitaire all work on iPhone now. (Regression: Klondike UI layout has changed, and is awkward on Mac now. Need to special case that.)
1 /* This code is based on Apple's "GeekGameBoard" sample code, version 1.0.
2 http://developer.apple.com/samplecode/GeekGameBoard/
3 Copyright © 2007 Apple Inc. Copyright © 2008 Jens Alfke. All Rights Reserved.
5 Redistribution and use in source and binary forms, with or without modification, are permitted
6 provided that the following conditions are met:
8 * Redistributions of source code must retain the above copyright notice, this list of conditions
9 and the following disclaimer.
10 * Redistributions in binary form must reproduce the above copyright notice, this list of
11 conditions and the following disclaimer in the documentation and/or other materials provided
12 with the distribution.
14 THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR
15 IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
16 FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRI-
17 BUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
18 (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
19 PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
20 CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
21 THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 #import "QuartzUtils.h"
31 @interface BoardView ()
32 - (void) _findDropTarget: (NSPoint)pos;
36 @implementation BoardView
39 @synthesize game=_game, gameboard=_gameboard;
49 - (void) startGameNamed: (NSString*)gameClassName
52 [_gameboard removeFromSuperlayer];
55 _gameboard = [[CALayer alloc] init];
56 _gameboard.frame = [self gameBoardFrame];
57 _gameboard.autoresizingMask = kCALayerWidthSizable | kCALayerHeightSizable;
58 [self.layer addSublayer: _gameboard];
61 Class gameClass = NSClassFromString(gameClassName);
62 setObj(&_game, [[gameClass alloc] initWithBoard: _gameboard]);
66 - (CGRect) gameBoardFrame
68 return self.layer.bounds;
72 - (void)resetCursorRects
74 [super resetCursorRects];
75 [self addCursorRect: self.bounds cursor: [NSCursor openHandCursor]];
79 - (IBAction) enterFullScreen: (id)sender
81 if( self.isInFullScreenMode ) {
82 [self exitFullScreenModeWithOptions: nil];
84 [self enterFullScreenMode: self.window.screen
91 #pragma mark KEY EVENTS:
94 - (void) keyDown: (NSEvent*)ev
96 if( self.isInFullScreenMode ) {
97 if( [ev.charactersIgnoringModifiers hasPrefix: @"\033"] ) // Esc key
98 [self enterFullScreen: self];
104 #pragma mark HIT-TESTING:
107 // Hit-testing callbacks (to identify which layers caller is interested in):
108 typedef BOOL (*LayerMatchCallback)(CALayer*);
110 static BOOL layerIsBit( CALayer* layer ) {return [layer isKindOfClass: [Bit class]];}
111 static BOOL layerIsBitHolder( CALayer* layer ) {return [layer conformsToProtocol: @protocol(BitHolder)];}
112 static BOOL layerIsDropTarget( CALayer* layer ) {return [layer respondsToSelector: @selector(draggingEntered:)];}
115 /** Locates the layer at a given point in window coords.
116 If the leaf layer doesn't pass the layer-match callback, the nearest ancestor that does is returned.
117 If outOffset is provided, the point's position relative to the layer is stored into it. */
118 - (CALayer*) hitTestPoint: (NSPoint)locationInWindow
119 forLayerMatching: (LayerMatchCallback)match
120 offset: (CGPoint*)outOffset
122 CGPoint where = NSPointToCGPoint([self convertPoint: locationInWindow fromView: nil]);
123 where = [_gameboard convertPoint: where fromLayer: self.layer];
124 CALayer *layer = [_gameboard hitTest: where];
127 CGPoint bitPos = [self.layer convertPoint: layer.position
128 fromLayer: layer.superlayer];
130 *outOffset = CGPointMake( bitPos.x-where.x, bitPos.y-where.y);
133 layer = layer.superlayer;
140 #pragma mark MOUSE CLICKS & DRAGS:
143 - (void) mouseDown: (NSEvent*)ev
146 _dragStartPos = ev.locationInWindow;
147 _dragBit = (Bit*) [self hitTestPoint: _dragStartPos
148 forLayerMatching: layerIsBit
149 offset: &_dragOffset];
152 // If no bit was clicked, see if it's a BitHolder the game will let the user add a Bit to:
153 id<BitHolder> holder = (id<BitHolder>) [self hitTestPoint: _dragStartPos
154 forLayerMatching: layerIsBitHolder
157 _dragBit = [_game bitToPlaceInHolder: holder];
159 _dragOffset.x = _dragOffset.y = 0;
160 if( _dragBit.superlayer==nil )
161 _dragBit.position = NSPointToCGPoint([self convertPoint: _dragStartPos fromView: nil]);
175 _oldHolder = _dragBit.holder;
176 // Ask holder's and game's permission before dragging:
178 _dragBit = [_oldHolder canDragBit: _dragBit];
179 if( _dragBit && ! [_game canBit: _dragBit moveFrom: _oldHolder] ) {
180 [_oldHolder cancelDragBit: _dragBit];
191 _oldSuperlayer = _dragBit.superlayer;
192 _oldLayerIndex = [_oldSuperlayer.sublayers indexOfObjectIdenticalTo: _dragBit];
193 _oldPos = _dragBit.position;
194 ChangeSuperlayer(_dragBit, self.layer, self.layer.sublayers.count);
195 _dragBit.pickedUp = YES;
196 [[NSCursor closedHandCursor] push];
200 _dragBit.position = NSPointToCGPoint([self convertPoint: _dragStartPos fromView: nil]);
202 [self _findDropTarget: _dragStartPos];
207 - (void) mouseDragged: (NSEvent*)ev
210 // Get the mouse position, and see if we've moved 3 pixels since the mouseDown:
211 NSPoint pos = ev.locationInWindow;
212 if( fabs(pos.x-_dragStartPos.x)>=3 || fabs(pos.y-_dragStartPos.y)>=3 )
215 // Move the _dragBit (without animation -- it's unnecessary and slows down responsiveness):
216 NSPoint where = [self convertPoint: pos fromView: nil];
217 where.x += _dragOffset.x;
218 where.y += _dragOffset.y;
220 CGPoint newPos = [_dragBit.superlayer convertPoint: NSPointToCGPoint(where)
221 fromLayer: self.layer];
223 [CATransaction flush];
224 [CATransaction begin];
225 [CATransaction setValue:(id)kCFBooleanTrue
226 forKey:kCATransactionDisableActions];
227 _dragBit.position = newPos;
228 [CATransaction commit];
230 // Find what it's over:
231 [self _findDropTarget: pos];
236 - (void) _findDropTarget: (NSPoint)locationInWindow
238 locationInWindow.x += _dragOffset.x;
239 locationInWindow.y += _dragOffset.y;
240 id<BitHolder> target = (id<BitHolder>) [self hitTestPoint: locationInWindow
241 forLayerMatching: layerIsBitHolder
243 if( target == _oldHolder )
245 if( target != _dropTarget ) {
246 [_dropTarget willNotDropBit: _dragBit];
247 _dropTarget.highlighted = NO;
251 CGPoint targetPos = [(CALayer*)target convertPoint: _dragBit.position
252 fromLayer: _dragBit.superlayer];
253 if( [target canDropBit: _dragBit atPoint: targetPos]
254 && [_game canBit: _dragBit moveFrom: _oldHolder to: target] ) {
255 _dropTarget = target;
256 _dropTarget.highlighted = YES;
262 - (void) mouseUp: (NSEvent*)ev
266 // Update the drag tracking to the final mouse position:
267 [self mouseDragged: ev];
268 _dropTarget.highlighted = NO;
269 _dragBit.pickedUp = NO;
271 // Is the move legal?
272 if( _dropTarget && [_dropTarget dropBit: _dragBit
273 atPoint: [(CALayer*)_dropTarget convertPoint: _dragBit.position
274 fromLayer: _dragBit.superlayer]] ) {
275 // Yes, notify the interested parties:
276 [_oldHolder draggedBit: _dragBit to: _dropTarget];
277 [_game bit: _dragBit movedFrom: _oldHolder to: _dropTarget];
280 [_dropTarget willNotDropBit: _dragBit];
281 if( _oldSuperlayer ) {
282 ChangeSuperlayer(_dragBit, _oldSuperlayer, _oldLayerIndex);
283 _dragBit.position = _oldPos;
284 [_oldHolder cancelDragBit: _dragBit];
286 [_dragBit removeFromSuperlayer];
290 // Just a click, without a drag:
291 _dropTarget.highlighted = NO;
292 _dragBit.pickedUp = NO;
293 ChangeSuperlayer(_dragBit, _oldSuperlayer, _oldLayerIndex);
294 [_oldHolder cancelDragBit: _dragBit];
295 if( ! [_game clickedBit: _dragBit] )
306 #pragma mark INCOMING DRAGS:
309 // subroutine to call the target
310 static int tell( id target, SEL selector, id arg, int defaultValue )
312 if( target && [target respondsToSelector: selector] )
313 return (ssize_t) [target performSelector: selector withObject: arg];
319 - (NSDragOperation)draggingEntered:(id <NSDraggingInfo>)sender
321 _viewDropTarget = [self hitTestPoint: [sender draggingLocation]
322 forLayerMatching: layerIsDropTarget
324 _viewDropOp = _viewDropTarget ?[_viewDropTarget draggingEntered: sender] :NSDragOperationNone;
328 - (NSDragOperation)draggingUpdated:(id <NSDraggingInfo>)sender
330 CALayer *target = [self hitTestPoint: [sender draggingLocation]
331 forLayerMatching: layerIsDropTarget
333 if( target == _viewDropTarget ) {
334 if( _viewDropTarget )
335 _viewDropOp = tell(_viewDropTarget,@selector(draggingUpdated:),sender,_viewDropOp);
337 tell(_viewDropTarget,@selector(draggingExited:),sender,0);
338 _viewDropTarget = target;
339 if( _viewDropTarget )
340 _viewDropOp = [_viewDropTarget draggingEntered: sender];
342 _viewDropOp = NSDragOperationNone;
347 - (BOOL)wantsPeriodicDraggingUpdates
349 return (_viewDropTarget!=nil);
352 - (void)draggingExited:(id <NSDraggingInfo>)sender
354 tell(_viewDropTarget,@selector(draggingExited:),sender,0);
355 _viewDropTarget = nil;
358 - (BOOL)prepareForDragOperation:(id <NSDraggingInfo>)sender
360 return tell(_viewDropTarget,@selector(prepareForDragOperation:),sender,YES);
363 - (BOOL)performDragOperation:(id <NSDraggingInfo>)sender
365 return [_viewDropTarget performDragOperation: sender];
368 - (void)concludeDragOperation:(id <NSDraggingInfo>)sender
370 tell(_viewDropTarget,@selector(concludeDragOperation:),sender,0);
373 - (void)draggingEnded:(id <NSDraggingInfo>)sender
375 tell(_viewDropTarget,@selector(draggingEnded:),sender,0);