blob: f864db10ff1f42bd8ad0a1cd3fce28fd7c5a579b [file] [log] [blame]
// Copyright (c) 2011 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#import "chrome/browser/ui/cocoa/image_button_cell.h"
#include "base/logging.h"
#include "base/mac/mac_util.h"
#import "chrome/browser/themes/theme_service.h"
#import "chrome/browser/ui/cocoa/rect_path_utils.h"
#import "chrome/browser/ui/cocoa/themed_window.h"
#import "ui/base/cocoa/nsview_additions.h"
#include "ui/gfx/image/image.h"
#include "ui/gfx/scoped_ns_graphics_context_save_gstate_mac.h"
// When the window doesn't have focus then we want to draw the button with a
// slightly lighter color. We do this by just reducing the alpha.
const CGFloat kImageNoFocusAlpha = 0.65;
const CGFloat kHoverAnimationDuration = 0.2; // seconds
namespace {
NSRect CenterImageInFrame(NSImage* image, NSRect frame) {
NSRect rect;
rect.size = [image size];
rect.origin.x =
frame.origin.x + std::round((NSWidth(frame) - NSWidth(rect)) / 2.0);
rect.origin.y =
frame.origin.y + std::round((NSHeight(frame) - NSHeight(rect)) / 2.0);
return rect;
}
}
@interface ImageButtonCell (Private)
- (void)sharedInit;
- (image_button_cell::ButtonState)currentButtonState;
- (NSImage*)imageForID:(NSInteger)imageID
controlView:(NSView*)controlView;
- (void)animationDidProgress;
@end
@interface HoverAnimation : NSAnimation
- (id)initWithOwner:(ImageButtonCell*)owner;
- (void)setCurrentProgress:(NSAnimationProgress)progress;
@end
@implementation HoverAnimation {
ImageButtonCell* owner_;
}
- (id)initWithOwner:(ImageButtonCell*)owner {
if ((self = [super initWithDuration:kHoverAnimationDuration
animationCurve:NSAnimationEaseInOut])) {
[self setAnimationBlockingMode:NSAnimationNonblocking];
owner_ = owner;
}
return self;
}
- (void)setCurrentProgress:(NSAnimationProgress)progress {
[super setCurrentProgress:progress];
[owner_ animationDidProgress];
}
@end
@implementation ImageButtonCell {
NSAnimation* hoverAnimation_;
image_button_cell::ButtonState oldState_;
}
@synthesize isMouseInside = isMouseInside_;
// For nib instantiations
- (id)initWithCoder:(NSCoder*)decoder {
if ((self = [super initWithCoder:decoder])) {
[self sharedInit];
}
return self;
}
// For programmatic instantiations
- (id)initTextCell:(NSString*)string {
if ((self = [super initTextCell:string])) {
[self sharedInit];
}
return self;
}
- (void)sharedInit {
[self setHighlightsBy:NSNoCellMask];
// We need to set this so that we can override |-mouseEntered:| and
// |-mouseExited:| to change the button image on hover states.
[self setShowsBorderOnlyWhileMouseInside:YES];
hoverAnimation_ = [[HoverAnimation alloc] initWithOwner:self];
}
- (void)dealloc {
[hoverAnimation_ stopAnimation];
[hoverAnimation_ release];
[super dealloc];
}
- (NSImage*)imageForState:(image_button_cell::ButtonState)state
view:(NSView*)controlView{
if (image_[state].imageId)
return [self imageForID:image_[state].imageId controlView:controlView];
return image_[state].image;
}
- (void)drawImageWithFrame:(NSRect)cellFrame inView:(NSView*)controlView {
image_button_cell::ButtonState state = [self currentButtonState];
BOOL windowHasFocus = [[controlView window] isMainWindow] ||
[[controlView window] isKeyWindow];
CGFloat alpha = [self imageAlphaForWindowState:[controlView window]];
NSImage* image = [self imageForState:state view:controlView];
NSImage* oldImage = nil;
CGFloat oldAlpha = 0.0;
if (!windowHasFocus) {
NSImage* defaultImage = [self
imageForState:image_button_cell::kDefaultStateBackground
view:controlView];
NSImage* hoverImage = [self
imageForState:image_button_cell::kHoverStateBackground
view:controlView];
if ([self currentButtonState] == image_button_cell::kDefaultState &&
defaultImage) {
image = defaultImage;
alpha = 1.0;
} else if ([self currentButtonState] == image_button_cell::kHoverState &&
hoverImage) {
image = hoverImage;
alpha = 1.0;
}
}
if ([hoverAnimation_ isAnimating]) {
oldImage = [self imageForState:oldState_ view:controlView];
oldAlpha = 1.0 - [hoverAnimation_ currentValue] * alpha;
alpha *= [hoverAnimation_ currentValue];
}
NSRect imageRect = CenterImageInFrame(image, cellFrame);
NSCompositingOperation op = [[controlView window] hasDarkTheme]
? NSCompositePlusLighter
: NSCompositePlusDarker;
if (oldImage) {
NSRect oldImageRect = CenterImageInFrame(oldImage, cellFrame);
[oldImage drawInRect:oldImageRect
fromRect:NSZeroRect
operation:op
fraction:oldAlpha
respectFlipped:YES
hints:nil];
}
[image drawInRect:imageRect
fromRect:NSZeroRect
operation:op
fraction:alpha
respectFlipped:YES
hints:nil];
}
- (void)setIsHoverDisabled:(BOOL)disable {
if (disable)
[self setIsMouseInside:NO];
isHoverDisabled_ = disable;
}
- (void)drawWithFrame:(NSRect)cellFrame inView:(NSView*)controlView {
[self drawImageWithFrame:cellFrame inView:controlView];
}
- (void)setImageID:(NSInteger)imageID
forButtonState:(image_button_cell::ButtonState)state {
DCHECK_GE(state, 0);
DCHECK_LT(state, image_button_cell::kButtonStateCount);
image_[state].image.reset();
image_[state].imageId = imageID;
[[self controlView] setNeedsDisplay:YES];
}
// Sets the image for the given button state using an image.
- (void)setImage:(NSImage*)image
forButtonState:(image_button_cell::ButtonState)state {
DCHECK_GE(state, 0);
DCHECK_LT(state, image_button_cell::kButtonStateCount);
image_[state].image.reset([image retain]);
image_[state].imageId = 0;
[[self controlView] setNeedsDisplay:YES];
}
- (CGFloat)imageAlphaForWindowState:(NSWindow*)window {
BOOL windowHasFocus = [window isMainWindow] || [window isKeyWindow];
return windowHasFocus ? 1.0 : kImageNoFocusAlpha;
}
- (const ui::ThemeProvider*)themeProviderForWindow:(NSWindow*)window {
return [window themeProvider];
}
- (image_button_cell::ButtonState)currentButtonState {
bool (^has)(image_button_cell::ButtonState) =
^(image_button_cell::ButtonState state) {
return image_[state].image || image_[state].imageId;
};
if (![self isEnabled] && has(image_button_cell::kDisabledState))
return image_button_cell::kDisabledState;
if ([self isHighlighted] && has(image_button_cell::kPressedState))
return image_button_cell::kPressedState;
if ([self isMouseInside] && has(image_button_cell::kHoverState))
return image_button_cell::kHoverState;
return image_button_cell::kDefaultState;
}
- (NSImage*)imageForID:(NSInteger)imageID
controlView:(NSView*)controlView {
if (!imageID)
return nil;
const ui::ThemeProvider* themeProvider =
[self themeProviderForWindow:[controlView window]];
if (!themeProvider)
return nil;
return themeProvider->GetNSImageNamed(imageID);
}
- (void)animationDidProgress {
NSView<ImageButton>* control =
static_cast<NSView<ImageButton>*>([self controlView]);
[control setNeedsDisplay:YES];
}
- (void)setIsMouseInside:(BOOL)isMouseInside {
if (isHoverDisabled_)
return;
if (isMouseInside_ != isMouseInside) {
oldState_ = [self currentButtonState];
isMouseInside_ = isMouseInside;
[hoverAnimation_ startAnimation];
NSView<ImageButton>* control =
static_cast<NSView<ImageButton>*>([self controlView]);
if ([control respondsToSelector:@selector(mouseInsideStateDidChange:)]) {
[control mouseInsideStateDidChange:isMouseInside];
}
[control setNeedsDisplay:YES];
}
}
- (void)setShowsBorderOnlyWhileMouseInside:(BOOL)show {
VLOG_IF(1, !show) << "setShowsBorderOnlyWhileMouseInside:NO ignored";
}
- (BOOL)showsBorderOnlyWhileMouseInside {
// Always returns YES so that buttons always get mouse tracking even when
// disabled. The reload button (and possibly others) depend on this.
return YES;
}
- (void)mouseEntered:(NSEvent*)theEvent {
[self setIsMouseInside:YES];
}
- (void)mouseExited:(NSEvent*)theEvent {
[self setIsMouseInside:NO];
}
@end