Source/BoardView.m
author Jens Alfke <jens@mooseyard.com>
Tue Mar 11 17:09:50 2008 -0700 (2008-03-11)
changeset 4 d781b00f3ed4
parent 0 e9f7ba4718e1
child 5 3ba1f29595c7
permissions -rw-r--r--
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.
     4 
     5     Redistribution and use in source and binary forms, with or without modification, are permitted
     6     provided that the following conditions are met:
     7 
     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.
    13 
    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.
    22 */
    23 #import "BoardView.h"
    24 #import "Bit.h"
    25 #import "BitHolder.h"
    26 #import "Game.h"
    27 #import "QuartzUtils.h"
    28 #import "GGBUtils.h"
    29 
    30 
    31 @interface BoardView ()
    32 - (void) _findDropTarget: (NSPoint)pos;
    33 @end
    34 
    35 
    36 @implementation BoardView
    37 
    38 
    39 @synthesize game=_game, gameboard=_gameboard;
    40 
    41 
    42 - (void) dealloc
    43 {
    44     [_game release];
    45     [super dealloc];
    46 }
    47 
    48 
    49 - (void) startGameNamed: (NSString*)gameClassName
    50 {
    51     if( _gameboard ) {
    52         [_gameboard removeFromSuperlayer];
    53         _gameboard = nil;
    54     }
    55     _gameboard = [[CALayer alloc] init];
    56     _gameboard.frame = [self gameBoardFrame];
    57     _gameboard.autoresizingMask = kCALayerWidthSizable | kCALayerHeightSizable;
    58     [self.layer addSublayer: _gameboard];
    59     [_gameboard release];
    60     
    61     Class gameClass = NSClassFromString(gameClassName);
    62     setObj(&_game, [[gameClass alloc] initWithBoard: _gameboard]);
    63 }
    64 
    65 
    66 - (CGRect) gameBoardFrame
    67 {
    68     return self.layer.bounds;
    69 }
    70 
    71 
    72 - (void)resetCursorRects
    73 {
    74     [super resetCursorRects];
    75     [self addCursorRect: self.bounds cursor: [NSCursor openHandCursor]];
    76 }
    77 
    78 
    79 - (IBAction) enterFullScreen: (id)sender
    80 {
    81     if( self.isInFullScreenMode ) {
    82         [self exitFullScreenModeWithOptions: nil];
    83     } else {
    84         [self enterFullScreenMode: self.window.screen 
    85                       withOptions: nil];
    86     }
    87 }
    88 
    89 
    90 #pragma mark -
    91 #pragma mark KEY EVENTS:
    92 
    93 
    94 - (void) keyDown: (NSEvent*)ev
    95 {
    96     if( self.isInFullScreenMode ) {
    97         if( [ev.charactersIgnoringModifiers hasPrefix: @"\033"] )       // Esc key
    98             [self enterFullScreen: self];
    99     }
   100 }
   101 
   102 
   103 #pragma mark -
   104 #pragma mark HIT-TESTING:
   105 
   106 
   107 // Hit-testing callbacks (to identify which layers caller is interested in):
   108 typedef BOOL (*LayerMatchCallback)(CALayer*);
   109 
   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:)];}
   113 
   114 
   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
   121 {
   122     CGPoint where = NSPointToCGPoint([self convertPoint: locationInWindow fromView: nil]);
   123     where = [_gameboard convertPoint: where fromLayer: self.layer];
   124     CALayer *layer = [_gameboard hitTest: where];
   125     while( layer ) {
   126         if( match(layer) ) {
   127             CGPoint bitPos = [self.layer convertPoint: layer.position 
   128                               fromLayer: layer.superlayer];
   129             if( outOffset )
   130                 *outOffset = CGPointMake( bitPos.x-where.x, bitPos.y-where.y);
   131             return layer;
   132         } else
   133             layer = layer.superlayer;
   134     }
   135     return nil;
   136 }
   137 
   138 
   139 #pragma mark -
   140 #pragma mark MOUSE CLICKS & DRAGS:
   141 
   142 
   143 - (void) mouseDown: (NSEvent*)ev
   144 {
   145     BOOL placing = NO;
   146     _dragStartPos = ev.locationInWindow;
   147     _dragBit = (Bit*) [self hitTestPoint: _dragStartPos
   148                         forLayerMatching: layerIsBit 
   149                                   offset: &_dragOffset];
   150     
   151     if( ! _dragBit ) {
   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
   155                                                            offset: NULL];
   156         if( holder ) {
   157             _dragBit = [_game bitToPlaceInHolder: holder];
   158             if( _dragBit ) {
   159                 _dragOffset.x = _dragOffset.y = 0;
   160                 if( _dragBit.superlayer==nil )
   161                     _dragBit.position = NSPointToCGPoint([self convertPoint: _dragStartPos fromView: nil]);
   162                 placing = YES;
   163             }
   164         }
   165     }
   166     
   167     if( ! _dragBit ) {
   168         Beep();
   169         return;
   170     }
   171     
   172     // Clicked on a Bit:
   173     _dragMoved = NO;
   174     _dropTarget = nil;
   175     _oldHolder = _dragBit.holder;
   176     // Ask holder's and game's permission before dragging:
   177     if( _oldHolder ) {
   178         _dragBit = [_oldHolder canDragBit: _dragBit];
   179         if( _dragBit && ! [_game canBit: _dragBit moveFrom: _oldHolder] ) {
   180             [_oldHolder cancelDragBit: _dragBit];
   181             _dragBit = nil;
   182         }
   183         if( ! _dragBit ) {
   184             _oldHolder = nil;
   185             NSBeep();
   186             return;
   187         }
   188     }
   189     
   190     // Start dragging:
   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];
   197     
   198     if( placing ) {
   199         if( _oldSuperlayer )
   200             _dragBit.position = NSPointToCGPoint([self convertPoint: _dragStartPos fromView: nil]);
   201         _dragMoved = YES;
   202         [self _findDropTarget: _dragStartPos];
   203     }
   204 }
   205 
   206 
   207 - (void) mouseDragged: (NSEvent*)ev
   208 {
   209     if( _dragBit ) {
   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 )
   213             _dragMoved = YES;
   214         
   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;
   219         
   220         CGPoint newPos = [_dragBit.superlayer convertPoint: NSPointToCGPoint(where) 
   221                                                  fromLayer: self.layer];
   222 
   223         [CATransaction flush];
   224         [CATransaction begin];
   225         [CATransaction setValue:(id)kCFBooleanTrue
   226                          forKey:kCATransactionDisableActions];
   227         _dragBit.position = newPos;
   228         [CATransaction commit];
   229 
   230         // Find what it's over:
   231         [self _findDropTarget: pos];
   232     }
   233 }
   234 
   235 
   236 - (void) _findDropTarget: (NSPoint)locationInWindow
   237 {
   238     locationInWindow.x += _dragOffset.x;
   239     locationInWindow.y += _dragOffset.y;
   240     id<BitHolder> target = (id<BitHolder>) [self hitTestPoint: locationInWindow
   241                                              forLayerMatching: layerIsBitHolder
   242                                                        offset: NULL];
   243     if( target == _oldHolder )
   244         target = nil;
   245     if( target != _dropTarget ) {
   246         [_dropTarget willNotDropBit: _dragBit];
   247         _dropTarget.highlighted = NO;
   248         _dropTarget = nil;
   249     }
   250     if( target ) {
   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;
   257         }
   258     }
   259 }
   260 
   261 
   262 - (void) mouseUp: (NSEvent*)ev
   263 {
   264     if( _dragBit ) {
   265         if( _dragMoved ) {
   266             // Update the drag tracking to the final mouse position:
   267             [self mouseDragged: ev];
   268             _dropTarget.highlighted = NO;
   269             _dragBit.pickedUp = NO;
   270 
   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];
   278             } else {
   279                 // Nope, cancel:
   280                 [_dropTarget willNotDropBit: _dragBit];
   281                 if( _oldSuperlayer ) {
   282                     ChangeSuperlayer(_dragBit, _oldSuperlayer, _oldLayerIndex);
   283                     _dragBit.position = _oldPos;
   284                     [_oldHolder cancelDragBit: _dragBit];
   285                 } else {
   286                     [_dragBit removeFromSuperlayer];
   287                 }
   288             }
   289         } else {
   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] )
   296                 NSBeep();
   297         }
   298         _dropTarget = nil;
   299         _dragBit = nil;
   300         [NSCursor pop];
   301     }
   302 }
   303 
   304 
   305 #pragma mark -
   306 #pragma mark INCOMING DRAGS:
   307 
   308 
   309 // subroutine to call the target
   310 static int tell( id target, SEL selector, id arg, int defaultValue )
   311 {
   312     if( target && [target respondsToSelector: selector] )
   313         return (ssize_t) [target performSelector: selector withObject: arg];
   314     else
   315         return defaultValue;
   316 }
   317 
   318 
   319 - (NSDragOperation)draggingEntered:(id <NSDraggingInfo>)sender
   320 {
   321     _viewDropTarget = [self hitTestPoint: [sender draggingLocation]
   322                         forLayerMatching: layerIsDropTarget
   323                                   offset: NULL];
   324     _viewDropOp = _viewDropTarget ?[_viewDropTarget draggingEntered: sender] :NSDragOperationNone;
   325     return _viewDropOp;
   326 }
   327 
   328 - (NSDragOperation)draggingUpdated:(id <NSDraggingInfo>)sender
   329 {
   330     CALayer *target = [self hitTestPoint: [sender draggingLocation]
   331                         forLayerMatching: layerIsDropTarget 
   332                                   offset: NULL];
   333     if( target == _viewDropTarget ) {
   334         if( _viewDropTarget )
   335             _viewDropOp = tell(_viewDropTarget,@selector(draggingUpdated:),sender,_viewDropOp);
   336     } else {
   337         tell(_viewDropTarget,@selector(draggingExited:),sender,0);
   338         _viewDropTarget = target;
   339         if( _viewDropTarget )
   340             _viewDropOp = [_viewDropTarget draggingEntered: sender];
   341         else
   342             _viewDropOp = NSDragOperationNone;
   343     }
   344     return _viewDropOp;
   345 }
   346 
   347 - (BOOL)wantsPeriodicDraggingUpdates
   348 {
   349     return (_viewDropTarget!=nil);
   350 }
   351 
   352 - (void)draggingExited:(id <NSDraggingInfo>)sender
   353 {
   354     tell(_viewDropTarget,@selector(draggingExited:),sender,0);
   355     _viewDropTarget = nil;
   356 }
   357 
   358 - (BOOL)prepareForDragOperation:(id <NSDraggingInfo>)sender
   359 {
   360     return tell(_viewDropTarget,@selector(prepareForDragOperation:),sender,YES);
   361 }
   362 
   363 - (BOOL)performDragOperation:(id <NSDraggingInfo>)sender
   364 {
   365     return [_viewDropTarget performDragOperation: sender];
   366 }
   367 
   368 - (void)concludeDragOperation:(id <NSDraggingInfo>)sender
   369 {
   370     tell(_viewDropTarget,@selector(concludeDragOperation:),sender,0);
   371 }
   372 
   373 - (void)draggingEnded:(id <NSDraggingInfo>)sender
   374 {
   375     tell(_viewDropTarget,@selector(draggingEnded:),sender,0);
   376 }
   377 
   378 @end