OLD | NEW |
1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | 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 | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "chrome/browser/cocoa/gradient_button_cell.h" | 5 #include "chrome/browser/cocoa/gradient_button_cell.h" |
6 | 6 |
7 #include "base/logging.h" | 7 #include "base/logging.h" |
8 #import "base/scoped_nsobject.h" | 8 #import "base/scoped_nsobject.h" |
9 #import "chrome/browser/browser_theme_provider.h" | 9 #import "chrome/browser/browser_theme_provider.h" |
10 #import "chrome/browser/cocoa/image_utils.h" | 10 #import "chrome/browser/cocoa/image_utils.h" |
11 #import "chrome/browser/cocoa/themed_window.h" | 11 #import "chrome/browser/cocoa/themed_window.h" |
12 #include "grit/theme_resources.h" | 12 #include "grit/theme_resources.h" |
13 #import "third_party/GTM/AppKit/GTMNSColor+Luminance.h" | 13 #import "third_party/GTM/AppKit/GTMNSColor+Luminance.h" |
14 | 14 |
15 @interface GradientButtonCell (Private) | 15 @interface GradientButtonCell (Private) |
16 - (void)sharedInit; | 16 - (void)sharedInit; |
17 | 17 |
18 // Get drawing parameters for a given cell frame in a given view. The inner | 18 // Get drawing parameters for a given cell frame in a given view. The inner |
19 // frame is the one required by |-drawInteriorWithFrame:inView:|. The inner and | 19 // frame is the one required by |-drawInteriorWithFrame:inView:|. The inner and |
20 // outer paths are the ones required by |-drawBorderAndFillForTheme:...|. The | 20 // outer paths are the ones required by |-drawBorderAndFillForTheme:...|. The |
21 // outer path also gives the area in which to clip. Any of the |return...| | 21 // outer path also gives the area in which to clip. Any of the |return...| |
22 // arguments may be NULL (in which case the given parameter won't be returned). | 22 // arguments may be NULL (in which case the given parameter won't be returned). |
23 // If |returnInnerPath| or |returnOuterPath|, |*returnInnerPath| or | 23 // If |returnInnerPath| or |returnOuterPath|, |*returnInnerPath| or |
24 // |*returnOuterPath| should be nil, respectively. | 24 // |*returnOuterPath| should be nil, respectively. |
25 - (void)getDrawParamsForFrame:(NSRect)cellFrame | 25 - (void)getDrawParamsForFrame:(NSRect)cellFrame |
26 inView:(NSView*)controlView | 26 inView:(NSView*)controlView |
27 innerFrame:(NSRect*)returnInnerFrame | 27 innerFrame:(NSRect*)returnInnerFrame |
28 innerPath:(NSBezierPath**)returnInnerPath | 28 innerPath:(NSBezierPath**)returnInnerPath |
29 clipPath:(NSBezierPath**)returnClipPath; | 29 clipPath:(NSBezierPath**)returnClipPath; |
| 30 |
| 31 |
30 @end | 32 @end |
31 | 33 |
| 34 |
32 static const NSTimeInterval kAnimationShowDuration = 0.2; | 35 static const NSTimeInterval kAnimationShowDuration = 0.2; |
33 static const NSTimeInterval kAnimationHideDuration = 0.4; | 36 static const NSTimeInterval kAnimationHideDuration = 0.4; |
| 37 static const NSTimeInterval kAnimationContinuousCycleDuration = 0.4; |
34 | 38 |
35 @implementation GradientButtonCell | 39 @implementation GradientButtonCell |
| 40 |
36 @synthesize hoverAlpha = hoverAlpha_; | 41 @synthesize hoverAlpha = hoverAlpha_; |
37 | 42 |
38 - (void)adjustHoverValue { | |
39 NSTimeInterval thisUpdate = [NSDate timeIntervalSinceReferenceDate]; | |
40 | |
41 NSTimeInterval elapsed = thisUpdate - lastHoverUpdate_; | |
42 | |
43 CGFloat opacity = [self hoverAlpha]; | |
44 if (isMouseInside_) { | |
45 opacity += elapsed / kAnimationShowDuration; | |
46 } else { | |
47 opacity -= elapsed / kAnimationHideDuration; | |
48 } | |
49 | |
50 if (!isMouseInside_ && opacity < 0) { | |
51 opacity = 0; | |
52 } else if (isMouseInside_ && opacity > 1) { | |
53 opacity = 1; | |
54 } else { | |
55 [self performSelector:_cmd withObject:nil afterDelay:0.02]; | |
56 } | |
57 lastHoverUpdate_ = thisUpdate; | |
58 [self setHoverAlpha:opacity]; | |
59 | |
60 [[self controlView] setNeedsDisplay:YES]; | |
61 } | |
62 | |
63 - (void)setMouseInside:(BOOL)flag animate:(BOOL)animated { | |
64 isMouseInside_ = flag; | |
65 if (animated) { | |
66 lastHoverUpdate_ = [NSDate timeIntervalSinceReferenceDate]; | |
67 [self adjustHoverValue]; | |
68 } else { | |
69 [NSObject cancelPreviousPerformRequestsWithTarget:self]; | |
70 [self setHoverAlpha:flag ? 1.0 : 0.0]; | |
71 } | |
72 [[self controlView] setNeedsDisplay:YES]; | |
73 } | |
74 | |
75 // For nib instantiations | 43 // For nib instantiations |
76 - (id)initWithCoder:(NSCoder*)decoder { | 44 - (id)initWithCoder:(NSCoder*)decoder { |
77 if ((self = [super initWithCoder:decoder])) { | 45 if ((self = [super initWithCoder:decoder])) { |
78 [self sharedInit]; | 46 [self sharedInit]; |
79 } | 47 } |
80 return self; | 48 return self; |
81 } | 49 } |
82 | 50 |
83 // For programmatic instantiations | 51 // For programmatic instantiations |
84 - (id)initTextCell:(NSString*)string { | 52 - (id)initTextCell:(NSString*)string { |
85 if ((self = [super initTextCell:string])) { | 53 if ((self = [super initTextCell:string])) { |
86 [self sharedInit]; | 54 [self sharedInit]; |
87 } | 55 } |
88 return self; | 56 return self; |
89 } | 57 } |
90 | 58 |
91 - (void)dealloc { | 59 - (void)dealloc { |
92 if (trackingArea_) { | 60 if (trackingArea_) { |
93 [[self controlView] removeTrackingArea:trackingArea_]; | 61 [[self controlView] removeTrackingArea:trackingArea_]; |
94 trackingArea_.reset(); | 62 trackingArea_.reset(); |
95 } | 63 } |
96 [super dealloc]; | 64 [super dealloc]; |
97 } | 65 } |
98 | 66 |
| 67 // Return YES if we are pulsing (towards another state or continuously). |
| 68 - (BOOL)pulsing { |
| 69 if ((pulseState_ == gradient_button_cell::kPulsingOn) || |
| 70 (pulseState_ == gradient_button_cell::kPulsingOff) || |
| 71 (pulseState_ == gradient_button_cell::kPulsingContinuous)) |
| 72 return YES; |
| 73 return NO; |
| 74 } |
| 75 |
| 76 // Perform one pulse step when animating a pulse. |
| 77 - (void)performOnePulseStep { |
| 78 NSTimeInterval thisUpdate = [NSDate timeIntervalSinceReferenceDate]; |
| 79 NSTimeInterval elapsed = thisUpdate - lastHoverUpdate_; |
| 80 CGFloat opacity = [self hoverAlpha]; |
| 81 |
| 82 // Update opacity based on state. |
| 83 // Adjust state if we have finished. |
| 84 switch (pulseState_) { |
| 85 case gradient_button_cell::kPulsingOn: |
| 86 opacity += elapsed / kAnimationShowDuration; |
| 87 if (opacity > 1.0) { |
| 88 [self setPulseState:gradient_button_cell::kPulsedOn]; |
| 89 return; |
| 90 } |
| 91 break; |
| 92 case gradient_button_cell::kPulsingOff: |
| 93 opacity -= elapsed / kAnimationHideDuration; |
| 94 if (opacity < 0.0) { |
| 95 [self setPulseState:gradient_button_cell::kPulsedOff]; |
| 96 return; |
| 97 } |
| 98 break; |
| 99 case gradient_button_cell::kPulsingContinuous: |
| 100 opacity += elapsed / kAnimationContinuousCycleDuration * pulseMultiplier_; |
| 101 if (opacity > 1.0) { |
| 102 opacity = 1.0; |
| 103 pulseMultiplier_ *= -1.0; |
| 104 } else if (opacity < 0.0) { |
| 105 opacity = 0.0; |
| 106 pulseMultiplier_ *= -1.0; |
| 107 } |
| 108 outerStrokeAlphaMult_ = opacity; |
| 109 break; |
| 110 default: |
| 111 NOTREACHED() << "unknown pulse state"; |
| 112 } |
| 113 |
| 114 // Update our control. |
| 115 lastHoverUpdate_ = thisUpdate; |
| 116 [self setHoverAlpha:opacity]; |
| 117 [[self controlView] setNeedsDisplay:YES]; |
| 118 |
| 119 // If our state needs it, keep going. |
| 120 if ([self pulsing]) { |
| 121 [self performSelector:_cmd withObject:nil afterDelay:0.02]; |
| 122 } |
| 123 } |
| 124 |
| 125 - (gradient_button_cell::PulseState)pulseState { |
| 126 return pulseState_; |
| 127 } |
| 128 |
| 129 // Set the pulsing state. This can either set the pulse to on or off |
| 130 // immediately (e.g. kPulsedOn, kPulsedOff) or initiate an animated |
| 131 // state change. |
| 132 - (void)setPulseState:(gradient_button_cell::PulseState)pstate { |
| 133 pulseState_ = pstate; |
| 134 pulseMultiplier_ = 0.0; |
| 135 [NSObject cancelPreviousPerformRequestsWithTarget:self]; |
| 136 lastHoverUpdate_ = [NSDate timeIntervalSinceReferenceDate]; |
| 137 |
| 138 switch (pstate) { |
| 139 case gradient_button_cell::kPulsedOn: |
| 140 case gradient_button_cell::kPulsedOff: |
| 141 outerStrokeAlphaMult_ = 1.0; |
| 142 [self setHoverAlpha:((pulseState_ == gradient_button_cell::kPulsedOn) ? |
| 143 1.0 : 0.0)]; |
| 144 [[self controlView] setNeedsDisplay:YES]; |
| 145 break; |
| 146 case gradient_button_cell::kPulsingOn: |
| 147 case gradient_button_cell::kPulsingOff: |
| 148 outerStrokeAlphaMult_ = 1.0; |
| 149 // Set initial value then engage timer. |
| 150 [self setHoverAlpha:((pulseState_ == gradient_button_cell::kPulsingOn) ? |
| 151 0.0 : 1.0)]; |
| 152 [self performOnePulseStep]; |
| 153 break; |
| 154 case gradient_button_cell::kPulsingContinuous: |
| 155 // Semantics of continuous pulsing are that we pulse independent |
| 156 // of mouse position. |
| 157 pulseMultiplier_ = 1.0; |
| 158 [self performOnePulseStep]; |
| 159 break; |
| 160 default: |
| 161 CHECK(0); |
| 162 break; |
| 163 } |
| 164 } |
| 165 |
| 166 - (void)safelyStopPulsing { |
| 167 [NSObject cancelPreviousPerformRequestsWithTarget:self]; |
| 168 } |
| 169 |
| 170 - (void)setIsContinuousPulsing:(BOOL)continuous { |
| 171 if (!continuous && pulseState_ != gradient_button_cell::kPulsingContinuous) |
| 172 return; |
| 173 if (continuous) { |
| 174 [self setPulseState:gradient_button_cell::kPulsingContinuous]; |
| 175 } else { |
| 176 [self setPulseState:(isMouseInside_ ? gradient_button_cell::kPulsedOn : |
| 177 gradient_button_cell::kPulsedOff)]; |
| 178 } |
| 179 } |
| 180 |
| 181 - (BOOL)isContinuousPulsing { |
| 182 return (pulseState_ == gradient_button_cell::kPulsingContinuous) ? |
| 183 YES : NO; |
| 184 } |
| 185 |
| 186 // If we are not continuously pulsing, perform a pulse animation to |
| 187 // reflect our new state. |
| 188 - (void)setMouseInside:(BOOL)flag animate:(BOOL)animated { |
| 189 isMouseInside_ = flag; |
| 190 if (pulseState_ != gradient_button_cell::kPulsingContinuous) { |
| 191 if (animated) { |
| 192 [self setPulseState:(isMouseInside_ ? gradient_button_cell::kPulsingOn : |
| 193 gradient_button_cell::kPulsingOff)]; |
| 194 } else { |
| 195 [self setPulseState:(isMouseInside_ ? gradient_button_cell::kPulsedOn : |
| 196 gradient_button_cell::kPulsedOff)]; |
| 197 } |
| 198 } |
| 199 } |
| 200 |
99 - (NSGradient*)gradientForHoverAlpha:(CGFloat)hoverAlpha | 201 - (NSGradient*)gradientForHoverAlpha:(CGFloat)hoverAlpha |
100 isThemed:(BOOL)themed { | 202 isThemed:(BOOL)themed { |
101 CGFloat startAlpha = 0.6 + 0.3 * hoverAlpha; | 203 CGFloat startAlpha = 0.6 + 0.3 * hoverAlpha; |
102 CGFloat endAlpha = 0.333 * hoverAlpha; | 204 CGFloat endAlpha = 0.333 * hoverAlpha; |
103 | 205 |
104 if (themed) { | 206 if (themed) { |
105 startAlpha = 0.2 + 0.35 * hoverAlpha; | 207 startAlpha = 0.2 + 0.35 * hoverAlpha; |
106 endAlpha = 0.333 * hoverAlpha; | 208 endAlpha = 0.333 * hoverAlpha; |
107 } | 209 } |
108 | 210 |
109 NSColor* startColor = | 211 NSColor* startColor = |
110 [NSColor colorWithCalibratedWhite:1.0 | 212 [NSColor colorWithCalibratedWhite:1.0 |
111 alpha:startAlpha]; | 213 alpha:startAlpha]; |
112 NSColor* endColor = | 214 NSColor* endColor = |
113 [NSColor colorWithCalibratedWhite:1.0 - 0.15 * hoverAlpha | 215 [NSColor colorWithCalibratedWhite:1.0 - 0.15 * hoverAlpha |
114 alpha:endAlpha]; | 216 alpha:endAlpha]; |
115 NSGradient* gradient = [[NSGradient alloc] initWithColorsAndLocations: | 217 NSGradient* gradient = [[NSGradient alloc] initWithColorsAndLocations: |
116 startColor, hoverAlpha * 0.33, | 218 startColor, hoverAlpha * 0.33, |
117 endColor, 1.0, nil]; | 219 endColor, 1.0, nil]; |
118 | 220 |
119 return [gradient autorelease]; | 221 return [gradient autorelease]; |
120 } | 222 } |
121 | 223 |
122 - (void)sharedInit { | 224 - (void)sharedInit { |
123 shouldTheme_ = YES; | 225 shouldTheme_ = YES; |
| 226 pulseState_ = gradient_button_cell::kPulsedOff; |
| 227 pulseMultiplier_ = 1.0; |
| 228 outerStrokeAlphaMult_ = 1.0; |
124 gradient_.reset([[self gradientForHoverAlpha:0.0 isThemed:NO] retain]); | 229 gradient_.reset([[self gradientForHoverAlpha:0.0 isThemed:NO] retain]); |
125 } | 230 } |
126 | 231 |
127 - (void)setShouldTheme:(BOOL)shouldTheme { | 232 - (void)setShouldTheme:(BOOL)shouldTheme { |
128 shouldTheme_ = shouldTheme; | 233 shouldTheme_ = shouldTheme; |
129 } | 234 } |
130 | 235 |
131 - (NSImage*)overlayImage { | 236 - (NSImage*)overlayImage { |
132 return overlayImage_.get(); | 237 return overlayImage_.get(); |
133 } | 238 } |
(...skipping 142 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
276 [NSGraphicsContext restoreGraphicsState]; | 381 [NSGraphicsContext restoreGraphicsState]; |
277 } | 382 } |
278 | 383 |
279 // Don't draw anything else for disabled flat buttons. | 384 // Don't draw anything else for disabled flat buttons. |
280 if (isFlatButton && ![self isEnabled]) | 385 if (isFlatButton && ![self isEnabled]) |
281 return; | 386 return; |
282 | 387 |
283 // Draw the outer stroke. | 388 // Draw the outer stroke. |
284 NSColor* strokeColor = nil; | 389 NSColor* strokeColor = nil; |
285 if (showClickedGradient) { | 390 if (showClickedGradient) { |
286 strokeColor = [NSColor colorWithCalibratedWhite:0.0 alpha:0.3]; | 391 strokeColor = [NSColor |
| 392 colorWithCalibratedWhite:0.0 |
| 393 alpha:0.3 * outerStrokeAlphaMult_]; |
287 } else { | 394 } else { |
288 strokeColor = themeProvider ? themeProvider->GetNSColor( | 395 strokeColor = themeProvider ? themeProvider->GetNSColor( |
289 active ? BrowserThemeProvider::COLOR_TOOLBAR_BUTTON_STROKE : | 396 active ? BrowserThemeProvider::COLOR_TOOLBAR_BUTTON_STROKE : |
290 BrowserThemeProvider::COLOR_TOOLBAR_BUTTON_STROKE_INACTIVE, | 397 BrowserThemeProvider::COLOR_TOOLBAR_BUTTON_STROKE_INACTIVE, |
291 true) : [NSColor colorWithCalibratedWhite:0.0 alpha:0.6]; | 398 true) : [NSColor colorWithCalibratedWhite:0.0 |
| 399 alpha:0.6 * outerStrokeAlphaMult_]; |
292 } | 400 } |
293 [strokeColor setStroke]; | 401 [strokeColor setStroke]; |
294 | 402 |
295 [innerPath setLineWidth:1]; | 403 [innerPath setLineWidth:1]; |
296 [innerPath stroke]; | 404 [innerPath stroke]; |
297 } | 405 } |
298 | 406 |
299 // TODO(viettrungluu): clean this up. | 407 // TODO(viettrungluu): clean this up. |
300 // (Private) | 408 // (Private) |
301 - (void)getDrawParamsForFrame:(NSRect)cellFrame | 409 - (void)getDrawParamsForFrame:(NSRect)cellFrame |
(...skipping 57 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
359 | 467 |
360 BOOL pressed = [self isHighlighted]; | 468 BOOL pressed = [self isHighlighted]; |
361 NSWindow* window = [controlView window]; | 469 NSWindow* window = [controlView window]; |
362 ThemeProvider* themeProvider = [window themeProvider]; | 470 ThemeProvider* themeProvider = [window themeProvider]; |
363 BOOL active = [window isKeyWindow] || [window isMainWindow]; | 471 BOOL active = [window isKeyWindow] || [window isMainWindow]; |
364 | 472 |
365 // Stroke the borders and appropriate fill gradient. If we're borderless, | 473 // Stroke the borders and appropriate fill gradient. If we're borderless, |
366 // the only time we want to draw the inner gradient is if we're highlighted. | 474 // the only time we want to draw the inner gradient is if we're highlighted. |
367 if (([self isBordered] && ![self showsBorderOnlyWhileMouseInside]) || | 475 if (([self isBordered] && ![self showsBorderOnlyWhileMouseInside]) || |
368 pressed || | 476 pressed || |
369 [self isMouseInside]) { | 477 [self isMouseInside] || |
| 478 [self pulsing]) { |
| 479 |
| 480 // When pulsing we want the bookmark to stand out a little more. |
| 481 BOOL showClickedGradient = pressed || |
| 482 (pulseState_ == gradient_button_cell::kPulsingContinuous); |
370 | 483 |
371 [self drawBorderAndFillForTheme:themeProvider | 484 [self drawBorderAndFillForTheme:themeProvider |
372 controlView:controlView | 485 controlView:controlView |
373 innerPath:innerPath | 486 innerPath:innerPath |
374 showClickedGradient:pressed | 487 showClickedGradient:showClickedGradient |
375 showHighlightGradient:[self isHighlighted] | 488 showHighlightGradient:[self isHighlighted] |
376 hoverAlpha:[self hoverAlpha] | 489 hoverAlpha:[self hoverAlpha] |
377 active:active | 490 active:active |
378 cellFrame:cellFrame | 491 cellFrame:cellFrame |
379 defaultGradient:nil]; | 492 defaultGradient:nil]; |
380 } | 493 } |
381 | 494 |
382 // If this is the left side of a segmented button, draw a slight shadow. | 495 // If this is the left side of a segmented button, draw a slight shadow. |
383 ButtonType type = [[(NSControl*)controlView cell] tag]; | 496 ButtonType type = [[(NSControl*)controlView cell] tag]; |
384 if (type == kLeftButtonWithShadowType) { | 497 if (type == kLeftButtonWithShadowType) { |
(...skipping 73 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
458 NSBezierPath* boundingPath = nil; | 571 NSBezierPath* boundingPath = nil; |
459 [self getDrawParamsForFrame:cellFrame | 572 [self getDrawParamsForFrame:cellFrame |
460 inView:controlView | 573 inView:controlView |
461 innerFrame:NULL | 574 innerFrame:NULL |
462 innerPath:NULL | 575 innerPath:NULL |
463 clipPath:&boundingPath]; | 576 clipPath:&boundingPath]; |
464 return boundingPath; | 577 return boundingPath; |
465 } | 578 } |
466 | 579 |
467 @end | 580 @end |
OLD | NEW |