Home | History | Annotate | Download | only in wrench_menu
      1 // Copyright (c) 2010 The Chromium Authors. All rights reserved.
      2 // Use of this source code is governed by a BSD-style license that can be
      3 // found in the LICENSE file.
      4 
      5 #import "chrome/browser/ui/cocoa/wrench_menu/menu_tracked_button.h"
      6 
      7 #include <cmath>
      8 
      9 @interface MenuTrackedButton (Private)
     10 - (void)doHighlight:(BOOL)highlight;
     11 - (void)checkMouseInRect;
     12 - (NSRect)insetBounds;
     13 - (BOOL)shouldHighlightOnHover;
     14 @end
     15 
     16 @implementation MenuTrackedButton
     17 
     18 @synthesize tracking = tracking_;
     19 
     20 - (void)updateTrackingAreas {
     21   [super updateTrackingAreas];
     22   [self removeTrackingRect:trackingTag_];
     23   trackingTag_ = [self addTrackingRect:NSInsetRect([self bounds], 1, 1)
     24                                  owner:self
     25                               userData:NULL
     26                           assumeInside:NO];
     27 }
     28 
     29 - (void)viewDidMoveToWindow {
     30   [self updateTrackingAreas];
     31   [self doHighlight:NO];
     32 }
     33 
     34 - (void)mouseEntered:(NSEvent*)theEvent {
     35   if (!tracking_) {
     36     didEnter_ = YES;
     37   }
     38   [self doHighlight:YES];
     39   [super mouseEntered:theEvent];
     40 }
     41 
     42 - (void)mouseExited:(NSEvent*)theEvent {
     43   didEnter_ = NO;
     44   tracking_ = NO;
     45   [self doHighlight:NO];
     46   [super mouseExited:theEvent];
     47 }
     48 
     49 - (void)mouseDragged:(NSEvent*)theEvent {
     50   tracking_ = !didEnter_;
     51 
     52   NSPoint point = [self convertPoint:[theEvent locationInWindow] fromView:nil];
     53   BOOL highlight = NSPointInRect(point, [self insetBounds]);
     54   [self doHighlight:highlight];
     55 
     56   // If tracking in non-sticky mode, poll the mouse cursor to see if it is still
     57   // over the button and thus needs to be highlighted.  The delay is the
     58   // smallest that still produces the effect while minimizing jank. Smaller
     59   // values make the selector fire too close to immediately/now for the mouse to
     60   // have moved off the receiver, and larger values produce lag.
     61   if (tracking_ && [self shouldHighlightOnHover]) {
     62     [self performSelector:@selector(checkMouseInRect)
     63                withObject:nil
     64                afterDelay:0.05
     65                   inModes:[NSArray arrayWithObject:NSEventTrackingRunLoopMode]];
     66   }
     67   [super mouseDragged:theEvent];
     68 }
     69 
     70 - (void)mouseUp:(NSEvent*)theEvent {
     71   [self doHighlight:NO];
     72   if (!tracking_) {
     73     return [super mouseUp:theEvent];
     74   }
     75   [self performClick:self];
     76   tracking_ = NO;
     77 }
     78 
     79 - (void)doHighlight:(BOOL)highlight {
     80   if (![self shouldHighlightOnHover]) {
     81     return;
     82   }
     83   [[self cell] setHighlighted:highlight];
     84   [self setNeedsDisplay];
     85 }
     86 
     87 // Checks if the user's current mouse location is over this button.  If it is,
     88 // the user is merely hovering here.  If it is not, then disable the highlight.
     89 // If the menu is opened in non-sticky mode, the button does not receive enter/
     90 // exit mouse events and thus polling is necessary.
     91 - (void)checkMouseInRect {
     92   NSPoint point = [NSEvent mouseLocation];
     93   point = [[self window] convertScreenToBase:point];
     94   point = [self convertPoint:point fromView:nil];
     95   if (!NSPointInRect(point, [self insetBounds])) {
     96     [self doHighlight:NO];
     97   }
     98 }
     99 
    100 // Returns the bounds of the receiver slightly inset to avoid highlighting both
    101 // buttons in a pair that overlap.
    102 - (NSRect)insetBounds {
    103   return NSInsetRect([self bounds], 2, 1);
    104 }
    105 
    106 - (BOOL)shouldHighlightOnHover {
    107   // Apple does not define NSAppKitVersionNumber10_5 when using the 10.5 SDK.
    108   // The Internets have come up with this solution.
    109   #ifndef NSAppKitVersionNumber10_5
    110   #define NSAppKitVersionNumber10_5 949
    111   #endif
    112 
    113   // There's a cell drawing bug in 10.5 that was fixed on 10.6.  Hover states
    114   // look terrible due to this, so disable highlighting on 10.5.
    115   return std::floor(NSAppKitVersionNumber) > NSAppKitVersionNumber10_5;
    116 }
    117 
    118 @end
    119