// Copyright (c) 2012 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. #include "base/mac/mac_util.h" #include "base/mac/sdk_forward_declarations.h" #import "ui/base/cocoa/nsview_additions.h" #include "ui/gfx/scoped_ns_graphics_context_save_gstate_mac.h" #include "base/logging.h" @implementation NSView (ChromeAdditions) - (CGFloat)cr_lineWidth { // All shipping retina macs run at least 10.7. if (![self respondsToSelector:@selector(convertSizeFromBacking:)]) return 1; return [self convertSizeFromBacking:NSMakeSize(1, 1)].width; } - (BOOL)cr_isMouseInView { NSPoint mouseLoc = [[self window] mouseLocationOutsideOfEventStream]; mouseLoc = [[self superview] convertPoint:mouseLoc fromView:nil]; return [self hitTest:mouseLoc] == self; } - (BOOL)cr_isBelowView:(NSView*)otherView { NSArray* subviews = [[self superview] subviews]; NSUInteger selfIndex = [subviews indexOfObject:self]; DCHECK_NE(NSNotFound, selfIndex); NSUInteger otherIndex = [subviews indexOfObject:otherView]; DCHECK_NE(NSNotFound, otherIndex); return selfIndex < otherIndex; } - (BOOL)cr_isAboveView:(NSView*)otherView { return ![self cr_isBelowView:otherView]; } - (void)cr_ensureSubview:(NSView*)subview isPositioned:(NSWindowOrderingMode)place relativeTo:(NSView *)otherView { DCHECK(place == NSWindowAbove || place == NSWindowBelow); BOOL isAbove = place == NSWindowAbove; if ([[subview superview] isEqual:self] && [subview cr_isAboveView:otherView] == isAbove) { return; } [subview removeFromSuperview]; [self addSubview:subview positioned:place relativeTo:otherView]; } - (NSColor*)cr_keyboardFocusIndicatorColor { return [[NSColor keyboardFocusIndicatorColor] colorWithAlphaComponent:0.5 / [self cr_lineWidth]]; } - (void)cr_recursivelySetNeedsDisplay:(BOOL)flag { [self setNeedsDisplay:YES]; for (NSView* child in [self subviews]) [child cr_recursivelySetNeedsDisplay:flag]; } static NSView* g_ancestorBeingDrawnFrom = nil; static NSView* g_childBeingDrawnTo = nil; - (void)cr_drawUsingAncestor:(NSView*)ancestorView inRect:(NSRect)rect { gfx::ScopedNSGraphicsContextSaveGState scopedGSState; NSRect frame = [self convertRect:[self bounds] toView:ancestorView]; NSAffineTransform* transform = [NSAffineTransform transform]; if ([self isFlipped] == [ancestorView isFlipped]) { [transform translateXBy:-NSMinX(frame) yBy:-NSMinY(frame)]; } else { [transform translateXBy:-NSMinX(frame) yBy:NSMaxY(frame)]; [transform scaleXBy:1.0 yBy:-1.0]; } [transform concat]; // This can be made robust to recursive calls, but is as of yet unneeded. DCHECK(!g_ancestorBeingDrawnFrom && !g_childBeingDrawnTo); g_ancestorBeingDrawnFrom = ancestorView; g_childBeingDrawnTo = self; [ancestorView drawRect:[ancestorView bounds]]; g_childBeingDrawnTo = nil; g_ancestorBeingDrawnFrom = nil; } - (NSView*)cr_viewBeingDrawnTo { if (!g_ancestorBeingDrawnFrom) return self; DCHECK(g_ancestorBeingDrawnFrom == self); return g_childBeingDrawnTo; } @end