Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 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 #import "chrome/browser/ui/cocoa/tabs/tab_strip_controller.h" | 5 #import "chrome/browser/ui/cocoa/tabs/tab_strip_controller.h" |
| 6 | 6 |
| 7 #import <QuartzCore/QuartzCore.h> | 7 #import <QuartzCore/QuartzCore.h> |
| 8 | 8 |
| 9 #include <cmath> | 9 #include <cmath> |
| 10 #include <limits> | 10 #include <limits> |
| (...skipping 173 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 184 [result addRepresentation:bmpImageRep]; | 184 [result addRepresentation:bmpImageRep]; |
| 185 } | 185 } |
| 186 [NSGraphicsContext restoreGraphicsState]; | 186 [NSGraphicsContext restoreGraphicsState]; |
| 187 | 187 |
| 188 return result.release(); | 188 return result.release(); |
| 189 } | 189 } |
| 190 | 190 |
| 191 // Takes a normal bitmap and a mask image and returns an image the size of the | 191 // Takes a normal bitmap and a mask image and returns an image the size of the |
| 192 // mask that has pixels from |image| but alpha information from |mask|. | 192 // mask that has pixels from |image| but alpha information from |mask|. |
| 193 NSImage* ApplyMask(NSImage* image, NSImage* mask) { | 193 NSImage* ApplyMask(NSImage* image, NSImage* mask) { |
| 194 DCHECK_EQ([image size].width, [mask size].width); | |
| 195 DCHECK_EQ([image size].height, [mask size].height); | |
| 196 | |
| 194 return [CreateImageWithSize([mask size], ^(NSSize size) { | 197 return [CreateImageWithSize([mask size], ^(NSSize size) { |
| 195 // Skip a few pixels from the top of the tab background gradient, because | 198 [image drawAtPoint:NSZeroPoint |
| 196 // the new tab button is not drawn at the very top of the browser window. | 199 fromRect:NSMakeRect(0, 0, size.width, size.height) |
| 197 const int kYOffset = 10; | 200 operation:NSCompositeCopy |
| 198 CGFloat width = size.width; | 201 fraction:1.0]; |
| 199 CGFloat height = size.height; | |
| 200 | |
| 201 // In some themes, the tab background image is narrower than the | |
| 202 // new tab button, so tile the background image. | |
| 203 CGFloat x = 0; | |
| 204 // The floor() is to make sure images with odd widths don't draw to the | |
| 205 // same pixel twice on retina displays. (Using NSDrawThreePartImage() | |
| 206 // caused a startup perf regression, so that cannot be used.) | |
| 207 CGFloat tileWidth = floor(std::min(width, [image size].width)); | |
| 208 while (x < width) { | |
| 209 [image drawAtPoint:NSMakePoint(x, 0) | |
| 210 fromRect:NSMakeRect(0, | |
| 211 [image size].height - height - kYOffset, | |
| 212 tileWidth, | |
| 213 height) | |
| 214 operation:NSCompositeCopy | |
| 215 fraction:1.0]; | |
| 216 x += tileWidth; | |
| 217 } | |
| 218 | |
| 219 [mask drawAtPoint:NSZeroPoint | 202 [mask drawAtPoint:NSZeroPoint |
| 220 fromRect:NSMakeRect(0, 0, width, height) | 203 fromRect:NSMakeRect(0, 0, size.width, size.height) |
| 221 operation:NSCompositeDestinationIn | 204 operation:NSCompositeDestinationIn |
| 222 fraction:1.0]; | 205 fraction:1.0]; |
| 223 }) autorelease]; | 206 }) autorelease]; |
| 224 } | 207 } |
| 225 | 208 |
| 226 // Creates a modified favicon used for the recording case. The mask is used for | 209 // Creates a modified favicon used for the recording case. The mask is used for |
| 227 // making part of the favicon transparent. (The part where the recording dot | 210 // making part of the favicon transparent. (The part where the recording dot |
| 228 // later is drawn.) | 211 // later is drawn.) |
| 229 NSImage* CreateMaskedFaviconForRecording(NSImage* image, | 212 NSImage* CreateMaskedFaviconForRecording(NSImage* image, |
| 230 NSImage* mask, | 213 NSImage* mask, |
| (...skipping 37 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 268 fromRect:NSMakeRect(0, 0, width, height) | 251 fromRect:NSMakeRect(0, 0, width, height) |
| 269 operation:NSCompositeCopy | 252 operation:NSCompositeCopy |
| 270 fraction:1.0]; | 253 fraction:1.0]; |
| 271 [overlay drawAtPoint:NSZeroPoint | 254 [overlay drawAtPoint:NSZeroPoint |
| 272 fromRect:NSMakeRect(0, 0, width, height) | 255 fromRect:NSMakeRect(0, 0, width, height) |
| 273 operation:NSCompositeSourceOver | 256 operation:NSCompositeSourceOver |
| 274 fraction:alpha]; | 257 fraction:alpha]; |
| 275 }) autorelease]; | 258 }) autorelease]; |
| 276 } | 259 } |
| 277 | 260 |
| 261 // Tiles |image| horizontally into a new NSImage of |size|. |y_inset| refers | |
| 262 // to the inset from the bottom of the returned image at which |image| should | |
| 263 // be painted. | |
| 264 NSImage* TileHorizontally(NSImage* image, CGFloat y_inset, NSSize size) { | |
| 265 return [CreateImageWithSize(size, ^(NSSize size) { | |
| 266 CGFloat width = size.width; | |
| 267 CGFloat height = size.height; | |
| 268 | |
| 269 CGFloat x = 0; | |
| 270 // The floor() is to make sure images with odd widths and height don't | |
| 271 // draw to the same pixel twice on retina displays. (Using | |
| 272 // NSDrawThreePartImage() caused a startup perf regression, so that | |
| 273 // cannot be used.) | |
| 274 CGFloat tileWidth = floor(std::min(width, [image size].width)); | |
| 275 while (x < width) { | |
| 276 [image drawAtPoint:NSMakePoint(x, 0) | |
| 277 fromRect:NSMakeRect(0, | |
| 278 [image size].height - y_inset, | |
| 279 tileWidth, | |
| 280 height) | |
| 281 operation:NSCompositeCopy | |
| 282 fraction:1.0]; | |
| 283 x += tileWidth; | |
| 284 } | |
| 285 }) autorelease]; | |
| 286 } | |
| 287 | |
| 278 } // namespace | 288 } // namespace |
| 279 | 289 |
| 280 @interface TabStripController (Private) | 290 @interface TabStripController (Private) |
| 281 - (void)addSubviewToPermanentList:(NSView*)aView; | 291 - (void)addSubviewToPermanentList:(NSView*)aView; |
| 282 - (void)regenerateSubviewList; | 292 - (void)regenerateSubviewList; |
| 283 - (NSInteger)indexForContentsView:(NSView*)view; | 293 - (NSInteger)indexForContentsView:(NSView*)view; |
| 284 - (NSImageView*)iconImageViewForContents:(content::WebContents*)contents; | 294 - (NSImageView*)iconImageViewForContents:(content::WebContents*)contents; |
| 285 - (void)updateFaviconForContents:(content::WebContents*)contents | 295 - (void)updateFaviconForContents:(content::WebContents*)contents |
| 286 atIndex:(NSInteger)modelIndex; | 296 atIndex:(NSInteger)modelIndex; |
| 287 - (void)layoutTabsWithAnimation:(BOOL)animate | 297 - (void)layoutTabsWithAnimation:(BOOL)animate |
| (...skipping 1973 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 2261 ThemeService *theme = | 2271 ThemeService *theme = |
| 2262 static_cast<ThemeService*>([[tabStripView_ window] themeProvider]); | 2272 static_cast<ThemeService*>([[tabStripView_ window] themeProvider]); |
| 2263 if (!theme) | 2273 if (!theme) |
| 2264 return; | 2274 return; |
| 2265 | 2275 |
| 2266 ResourceBundle& rb = ResourceBundle::GetSharedInstance(); | 2276 ResourceBundle& rb = ResourceBundle::GetSharedInstance(); |
| 2267 NSImage* mask = rb.GetNativeImageNamed(IDR_NEWTAB_BUTTON_MASK).ToNSImage(); | 2277 NSImage* mask = rb.GetNativeImageNamed(IDR_NEWTAB_BUTTON_MASK).ToNSImage(); |
| 2268 NSImage* normal = rb.GetNativeImageNamed(IDR_NEWTAB_BUTTON).ToNSImage(); | 2278 NSImage* normal = rb.GetNativeImageNamed(IDR_NEWTAB_BUTTON).ToNSImage(); |
| 2269 NSImage* hover = rb.GetNativeImageNamed(IDR_NEWTAB_BUTTON_H).ToNSImage(); | 2279 NSImage* hover = rb.GetNativeImageNamed(IDR_NEWTAB_BUTTON_H).ToNSImage(); |
| 2270 NSImage* pressed = rb.GetNativeImageNamed(IDR_NEWTAB_BUTTON_P).ToNSImage(); | 2280 NSImage* pressed = rb.GetNativeImageNamed(IDR_NEWTAB_BUTTON_P).ToNSImage(); |
| 2281 NSImage* ntp_bg_active = theme->GetNSImageNamed( | |
|
Nico
2013/07/17 18:03:04
useCamelCapsInObjCCode
| |
| 2282 IDR_THEME_TAB_BACKGROUND, true); | |
| 2283 NSImage* ntp_bg_active_overlay = theme->GetNSImageNamed( | |
| 2284 IDR_THEME_TAB_BACKGROUND_OVERLAY, false); | |
| 2271 | 2285 |
| 2272 NSImage* foreground = ApplyMask( | 2286 // Compute the y insets so that |ntp_bg_active| lines up vertically with the |
| 2273 theme->GetNSImageNamed(IDR_THEME_TAB_BACKGROUND, true), mask); | 2287 // frame and that |ntp_bg_active_overlay| lines up vertically with the the |
| 2288 // tab. The tab strip has not been layed out yet so | |
| 2289 // themePatternPhaseForAlignment cannot be used. | |
| 2290 int distance_between_tab_strip_and_button = [[self class] defaultTabHeight] - | |
| 2291 NSMaxY([newTabButton_ bounds]); | |
| 2292 int distance_between_frame_and_button = NSHeight([tabStripView_ bounds]) - | |
|
Nico
2013/07/17 18:03:04
Can you move most of this stuff into a different m
pkotwicz
2013/08/15 07:09:59
I now compute this value in NewTabButtonThemeBackg
| |
| 2293 NSMaxY([newTabButton_ bounds]); | |
| 2294 // ImageButtonCell draws the image centered vertically. | |
| 2295 int y_offset_in_button = roundf( | |
| 2296 (NSHeight([newTabButton_ bounds]) - [mask size].height) / 2); | |
| 2297 // The constants below match those used in | |
| 2298 // BrowserWindowUtils::themePatternPhaseForAlignment | |
|
Nico
2013/07/17 18:03:04
can they be shared somehow instead?
pkotwicz
2013/08/15 07:09:59
These values are now shared thanks to https://code
| |
| 2299 int y_inset_align_with_tab_strip = distance_between_tab_strip_and_button + | |
| 2300 y_offset_in_button + [mask size].height + 2; | |
| 2301 int y_inset_align_with_frame = distance_between_frame_and_button + | |
| 2302 y_offset_in_button + [mask size].height + 3; | |
| 2303 | |
| 2304 // Create the foreground image by combining the tinted frame image | |
| 2305 // (IDR_THEME_TAB_BACKGROUND) and the custom theme provided overlay | |
| 2306 // (IDR_THEME_TAB_BACKGROUND_OVERLAY). Tile the images because the images | |
| 2307 // are narrower than the new tab button for some themes. | |
| 2308 NSImage* unmasked_foreground = TileHorizontally(ntp_bg_active, | |
| 2309 y_inset_align_with_frame, [mask size]); | |
| 2310 if (ntp_bg_active_overlay) { | |
| 2311 NSImage* tiled_overlay = TileHorizontally(ntp_bg_active_overlay, | |
| 2312 y_inset_align_with_tab_strip, [mask size]); | |
| 2313 unmasked_foreground = Overlay(unmasked_foreground, tiled_overlay, 1.0); | |
| 2314 } | |
| 2315 NSImage* foreground = ApplyMask(unmasked_foreground, mask); | |
| 2274 | 2316 |
| 2275 [[newTabButton_ cell] setImage:Overlay(foreground, normal, 1.0) | 2317 [[newTabButton_ cell] setImage:Overlay(foreground, normal, 1.0) |
| 2276 forButtonState:image_button_cell::kDefaultState]; | 2318 forButtonState:image_button_cell::kDefaultState]; |
| 2277 [[newTabButton_ cell] setImage:Overlay(foreground, hover, 1.0) | 2319 [[newTabButton_ cell] setImage:Overlay(foreground, hover, 1.0) |
| 2278 forButtonState:image_button_cell::kHoverState]; | 2320 forButtonState:image_button_cell::kHoverState]; |
| 2279 [[newTabButton_ cell] setImage:Overlay(foreground, pressed, 1.0) | 2321 [[newTabButton_ cell] setImage:Overlay(foreground, pressed, 1.0) |
| 2280 forButtonState:image_button_cell::kPressedState]; | 2322 forButtonState:image_button_cell::kPressedState]; |
| 2281 | 2323 |
| 2282 // IDR_THEME_TAB_BACKGROUND_INACTIVE is only used with the default theme. | 2324 // IDR_THEME_TAB_BACKGROUND_INACTIVE is only used with the default theme. |
| 2283 if (theme->UsingDefaultTheme()) { | 2325 if (theme->UsingDefaultTheme()) { |
| 2284 const CGFloat alpha = tabs::kImageNoFocusAlpha; | 2326 const CGFloat alpha = tabs::kImageNoFocusAlpha; |
| 2285 NSImage* background = ApplyMask( | 2327 NSImage* unmasked_background = TileHorizontally( |
| 2286 theme->GetNSImageNamed(IDR_THEME_TAB_BACKGROUND_INACTIVE, true), mask); | 2328 rb.GetNativeImageNamed(IDR_THEME_TAB_BACKGROUND_INACTIVE).ToNSImage(), |
| 2329 y_inset_align_with_tab_strip, | |
| 2330 [mask size]); | |
| 2331 NSImage* background = ApplyMask(unmasked_background, mask); | |
| 2287 [[newTabButton_ cell] setImage:Overlay(background, normal, alpha) | 2332 [[newTabButton_ cell] setImage:Overlay(background, normal, alpha) |
| 2288 forButtonState:image_button_cell::kDefaultStateBackground]; | 2333 forButtonState:image_button_cell::kDefaultStateBackground]; |
| 2289 [[newTabButton_ cell] setImage:Overlay(background, hover, alpha) | 2334 [[newTabButton_ cell] setImage:Overlay(background, hover, alpha) |
| 2290 forButtonState:image_button_cell::kHoverStateBackground]; | 2335 forButtonState:image_button_cell::kHoverStateBackground]; |
| 2291 } else { | 2336 } else { |
| 2292 [[newTabButton_ cell] setImage:nil | 2337 [[newTabButton_ cell] setImage:nil |
| 2293 forButtonState:image_button_cell::kDefaultStateBackground]; | 2338 forButtonState:image_button_cell::kDefaultStateBackground]; |
| 2294 [[newTabButton_ cell] setImage:nil | 2339 [[newTabButton_ cell] setImage:nil |
| 2295 forButtonState:image_button_cell::kHoverStateBackground]; | 2340 forButtonState:image_button_cell::kHoverStateBackground]; |
| 2296 } | 2341 } |
| 2297 } | 2342 } |
| 2298 | 2343 |
| 2299 @end | 2344 @end |
| 2300 | 2345 |
| 2301 NSView* GetSheetParentViewForWebContents(WebContents* web_contents) { | 2346 NSView* GetSheetParentViewForWebContents(WebContents* web_contents) { |
| 2302 // View hierarchy of the contents view: | 2347 // View hierarchy of the contents view: |
| 2303 // NSView -- switchView, same for all tabs | 2348 // NSView -- switchView, same for all tabs |
| 2304 // +- NSView -- TabContentsController's view | 2349 // +- NSView -- TabContentsController's view |
| 2305 // +- TabContentsViewCocoa | 2350 // +- TabContentsViewCocoa |
| 2306 // | 2351 // |
| 2307 // Changing it? Do not forget to modify | 2352 // Changing it? Do not forget to modify |
| 2308 // -[TabStripController swapInTabAtIndex:] too. | 2353 // -[TabStripController swapInTabAtIndex:] too. |
| 2309 return [web_contents->GetView()->GetNativeView() superview]; | 2354 return [web_contents->GetView()->GetNativeView() superview]; |
| 2310 } | 2355 } |
| OLD | NEW |