glass_browser_frame_view.cc revision 58537e28ecd584eab876aee8be7156509866d23a
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
3// found in the LICENSE file.
4
5#include "chrome/browser/ui/views/frame/glass_browser_frame_view.h"
6
7#include "base/command_line.h"
8#include "base/prefs/pref_service.h"
9#include "base/strings/utf_string_conversions.h"
10#include "chrome/app/chrome_command_ids.h"
11#include "chrome/app/chrome_dll_resource.h"
12#include "chrome/browser/chrome_notification_types.h"
13#include "chrome/browser/profiles/profile.h"
14#include "chrome/browser/themes/theme_properties.h"
15#include "chrome/browser/ui/views/avatar_menu_button.h"
16#include "chrome/browser/ui/views/frame/browser_view.h"
17#include "chrome/browser/ui/views/tabs/tab.h"
18#include "chrome/browser/ui/views/tabs/tab_strip.h"
19#include "chrome/common/chrome_switches.h"
20#include "chrome/common/pref_names.h"
21#include "content/public/browser/notification_service.h"
22#include "grit/generated_resources.h"
23#include "grit/theme_resources.h"
24#include "grit/ui_resources.h"
25#include "ui/base/l10n/l10n_util.h"
26#include "ui/base/resource/resource_bundle_win.h"
27#include "ui/base/theme_provider.h"
28#include "ui/gfx/canvas.h"
29#include "ui/gfx/dpi_win.h"
30#include "ui/gfx/icon_util.h"
31#include "ui/gfx/image/image.h"
32#include "ui/views/controls/label.h"
33#include "ui/views/layout/layout_constants.h"
34#include "ui/views/win/hwnd_util.h"
35#include "ui/views/window/client_view.h"
36
37HICON GlassBrowserFrameView::throbber_icons_[
38    GlassBrowserFrameView::kThrobberIconCount];
39
40namespace {
41// There are 3 px of client edge drawn inside the outer frame borders.
42const int kNonClientBorderThickness = 3;
43// Besides the frame border, there's another 9 px of empty space atop the
44// window in restored mode, to use to drag the window around.
45const int kNonClientRestoredExtraThickness = 9;
46// In the window corners, the resize areas don't actually expand bigger, but the
47// 16 px at the end of the top and bottom edges triggers diagonal resizing.
48const int kResizeAreaCornerSize = 16;
49// The avatar ends 2 px above the bottom of the tabstrip (which, given the
50// way the tabstrip draws its bottom edge, will appear like a 1 px gap to the
51// user).
52const int kAvatarBottomSpacing = 2;
53// Space between the frame border and the left edge of the avatar.
54const int kAvatarLeftSpacing = 2;
55// Space between the right edge of the avatar and the tabstrip.
56const int kAvatarRightSpacing = -2;
57// The content left/right images have a shadow built into them.
58const int kContentEdgeShadowThickness = 2;
59// The top 3 px of the tabstrip is shadow; in maximized mode we push this off
60// the top of the screen so the tabs appear flush against the screen edge.
61const int kTabstripTopShadowThickness = 3;
62// In restored mode, the New Tab button isn't at the same height as the caption
63// buttons, but the space will look cluttered if it actually slides under them,
64// so we stop it when the gap between the two is down to 5 px.
65const int kNewTabCaptionRestoredSpacing = 5;
66// In maximized mode, where the New Tab button and the caption buttons are at
67// similar vertical coordinates, we need to reserve a larger, 16 px gap to avoid
68// looking too cluttered.
69const int kNewTabCaptionMaximizedSpacing = 16;
70// How far to indent the tabstrip from the left side of the screen when there
71// is no avatar icon.
72const int kTabStripIndent = -6;
73
74}  // namespace
75
76///////////////////////////////////////////////////////////////////////////////
77// GlassBrowserFrameView, public:
78
79GlassBrowserFrameView::GlassBrowserFrameView(BrowserFrame* frame,
80                                             BrowserView* browser_view)
81    : BrowserNonClientFrameView(frame, browser_view),
82      throbber_running_(false),
83      throbber_frame_(0) {
84  if (browser_view->ShouldShowWindowIcon())
85    InitThrobberIcons();
86
87  UpdateAvatarInfo();
88  if (!browser_view->IsOffTheRecord()) {
89    registrar_.Add(this, chrome::NOTIFICATION_PROFILE_CACHED_INFO_CHANGED,
90                   content::NotificationService::AllSources());
91  }
92}
93
94GlassBrowserFrameView::~GlassBrowserFrameView() {
95}
96
97///////////////////////////////////////////////////////////////////////////////
98// GlassBrowserFrameView, BrowserNonClientFrameView implementation:
99
100gfx::Rect GlassBrowserFrameView::GetBoundsForTabStrip(
101    views::View* tabstrip) const {
102  int minimize_button_offset =
103      std::min(frame()->GetMinimizeButtonOffset(), width());
104  int tabstrip_x = browser_view()->ShouldShowAvatar() ?
105      (avatar_bounds_.right() + kAvatarRightSpacing) :
106      NonClientBorderThickness() + kTabStripIndent;
107  // In RTL languages, we have moved an avatar icon left by the size of window
108  // controls to prevent it from being rendered over them. So, we use its x
109  // position to move this tab strip left when maximized. Also, we can render
110  // a tab strip until the left end of this window without considering the size
111  // of window controls in RTL languages.
112  if (base::i18n::IsRTL()) {
113    if (!browser_view()->ShouldShowAvatar() && frame()->IsMaximized())
114      tabstrip_x += avatar_bounds_.x();
115    minimize_button_offset = width();
116  }
117  int tabstrip_width = minimize_button_offset - tabstrip_x -
118      (frame()->IsMaximized() ?
119          kNewTabCaptionMaximizedSpacing : kNewTabCaptionRestoredSpacing);
120  return gfx::Rect(tabstrip_x, GetTabStripInsets(false).top,
121                   std::max(0, tabstrip_width),
122                   tabstrip->GetPreferredSize().height());
123}
124
125BrowserNonClientFrameView::TabStripInsets
126GlassBrowserFrameView::GetTabStripInsets(bool restored) const {
127  // TODO: include OTR and caption.
128  return TabStripInsets(NonClientTopBorderHeight(restored), 0, 0);
129}
130
131int GlassBrowserFrameView::GetThemeBackgroundXInset() const {
132  return 0;
133}
134
135void GlassBrowserFrameView::UpdateThrobber(bool running) {
136  if (throbber_running_) {
137    if (running) {
138      DisplayNextThrobberFrame();
139    } else {
140      StopThrobber();
141    }
142  } else if (running) {
143    StartThrobber();
144  }
145}
146
147gfx::Size GlassBrowserFrameView::GetMinimumSize() {
148  gfx::Size min_size(browser_view()->GetMinimumSize());
149
150  // Account for the client area insets.
151  gfx::Insets insets = GetClientAreaInsets();
152  min_size.Enlarge(insets.width(), insets.height());
153  // Client area insets do not include the shadow thickness.
154  min_size.Enlarge(2 * kContentEdgeShadowThickness, 0);
155
156  // Ensure that the minimum width is enough to hold a tab strip with minimum
157  // width at its usual insets.
158  if (browser_view()->IsTabStripVisible()) {
159    TabStrip* tabstrip = browser_view()->tabstrip();
160    int min_tabstrip_width = tabstrip->GetMinimumSize().width();
161    int min_tabstrip_area_width =
162        width() - GetBoundsForTabStrip(tabstrip).width() + min_tabstrip_width;
163    min_size.set_width(std::max(min_tabstrip_area_width, min_size.width()));
164  }
165
166  return min_size;
167}
168
169///////////////////////////////////////////////////////////////////////////////
170// GlassBrowserFrameView, views::NonClientFrameView implementation:
171
172gfx::Rect GlassBrowserFrameView::GetBoundsForClientView() const {
173  return client_view_bounds_;
174}
175
176gfx::Rect GlassBrowserFrameView::GetWindowBoundsForClientBounds(
177    const gfx::Rect& client_bounds) const {
178  HWND hwnd = views::HWNDForWidget(frame());
179  if (!browser_view()->IsTabStripVisible() && hwnd) {
180    // If we don't have a tabstrip, we're either a popup or an app window, in
181    // which case we have a standard size non-client area and can just use
182    // AdjustWindowRectEx to obtain it. We check for a non-NULL window handle in
183    // case this gets called before the window is actually created.
184    RECT rect = client_bounds.ToRECT();
185    AdjustWindowRectEx(&rect, GetWindowLong(hwnd, GWL_STYLE), FALSE,
186                       GetWindowLong(hwnd, GWL_EXSTYLE));
187    return gfx::Rect(rect);
188  }
189
190  gfx::Insets insets = GetClientAreaInsets();
191  return gfx::Rect(std::max(0, client_bounds.x() - insets.left()),
192                   std::max(0, client_bounds.y() - insets.top()),
193                   client_bounds.width() + insets.width(),
194                   client_bounds.height() + insets.height());
195}
196
197int GlassBrowserFrameView::NonClientHitTest(const gfx::Point& point) {
198  // If the browser isn't in normal mode, we haven't customized the frame, so
199  // Windows can figure this out.  If the point isn't within our bounds, then
200  // it's in the native portion of the frame, so again Windows can figure it
201  // out.
202  if (!browser_view()->IsBrowserTypeNormal() || !bounds().Contains(point))
203    return HTNOWHERE;
204
205  // See if the point is within the avatar menu button or within the avatar
206  // label.
207  if (avatar_button() && avatar_button()->GetMirroredBounds().Contains(point))
208    return HTCLIENT;
209
210  int frame_component = frame()->client_view()->NonClientHitTest(point);
211
212  // See if we're in the sysmenu region.  We still have to check the tabstrip
213  // first so that clicks in a tab don't get treated as sysmenu clicks.
214  int nonclient_border_thickness = NonClientBorderThickness();
215  if (gfx::Rect(nonclient_border_thickness, GetSystemMetrics(SM_CXSIZEFRAME),
216                GetSystemMetrics(SM_CXSMICON),
217                GetSystemMetrics(SM_CYSMICON)).Contains(point))
218    return (frame_component == HTCLIENT) ? HTCLIENT : HTSYSMENU;
219
220  if (frame_component != HTNOWHERE)
221    return frame_component;
222
223  int frame_border_thickness = FrameBorderThickness();
224  int window_component = GetHTComponentForFrame(point, frame_border_thickness,
225      nonclient_border_thickness, frame_border_thickness,
226      kResizeAreaCornerSize - frame_border_thickness,
227      frame()->widget_delegate()->CanResize());
228  // Fall back to the caption if no other component matches.
229  return (window_component == HTNOWHERE) ? HTCAPTION : window_component;
230}
231
232///////////////////////////////////////////////////////////////////////////////
233// GlassBrowserFrameView, views::View overrides:
234
235void GlassBrowserFrameView::OnPaint(gfx::Canvas* canvas) {
236  if (!browser_view()->IsTabStripVisible())
237    return;  // Nothing is visible, so don't bother to paint.
238
239  PaintToolbarBackground(canvas);
240  if (!frame()->IsMaximized())
241    PaintRestoredClientEdge(canvas);
242}
243
244void GlassBrowserFrameView::Layout() {
245  LayoutAvatar();
246  LayoutClientView();
247}
248
249bool GlassBrowserFrameView::HitTestRect(const gfx::Rect& rect) const {
250  return (avatar_button() &&
251          avatar_button()->GetMirroredBounds().Intersects(rect)) ||
252          !frame()->client_view()->bounds().Intersects(rect);
253}
254
255///////////////////////////////////////////////////////////////////////////////
256// GlassBrowserFrameView, private:
257
258int GlassBrowserFrameView::FrameBorderThickness() const {
259  return (frame()->IsMaximized() || frame()->IsFullscreen()) ?
260      0 : GetSystemMetrics(SM_CXSIZEFRAME);
261}
262
263int GlassBrowserFrameView::NonClientBorderThickness() const {
264  if (frame()->IsMaximized() || frame()->IsFullscreen())
265    return 0;
266
267  return kNonClientBorderThickness;
268}
269
270int GlassBrowserFrameView::NonClientTopBorderHeight(
271    bool restored) const {
272  if (!restored && frame()->IsFullscreen())
273    return 0;
274
275  // We'd like to use FrameBorderThickness() here, but the maximized Aero glass
276  // frame has a 0 frame border around most edges and a CYSIZEFRAME-thick border
277  // at the top (see AeroGlassFrame::OnGetMinMaxInfo()).
278  return gfx::win::GetSystemMetricsInDIP(SM_CYSIZEFRAME) +
279      ((!restored && !frame()->ShouldLeaveOffsetNearTopBorder()) ?
280      -kTabstripTopShadowThickness : kNonClientRestoredExtraThickness);
281}
282
283void GlassBrowserFrameView::PaintToolbarBackground(gfx::Canvas* canvas) {
284  ui::ThemeProvider* tp = GetThemeProvider();
285
286  gfx::Rect toolbar_bounds(browser_view()->GetToolbarBounds());
287  gfx::Point toolbar_origin(toolbar_bounds.origin());
288  View::ConvertPointToTarget(browser_view(), this, &toolbar_origin);
289  toolbar_bounds.set_origin(toolbar_origin);
290  int x = toolbar_bounds.x();
291  int w = toolbar_bounds.width();
292  int left_x = x - kContentEdgeShadowThickness;
293
294  gfx::ImageSkia* theme_toolbar = tp->GetImageSkiaNamed(IDR_THEME_TOOLBAR);
295  gfx::ImageSkia* toolbar_left = tp->GetImageSkiaNamed(
296      IDR_CONTENT_TOP_LEFT_CORNER);
297  gfx::ImageSkia* toolbar_center = tp->GetImageSkiaNamed(
298      IDR_CONTENT_TOP_CENTER);
299
300  // Tile the toolbar image starting at the frame edge on the left and where
301  // the tabstrip is on the top.
302  int y = toolbar_bounds.y();
303  int dest_y = y + (kFrameShadowThickness * 2);
304  canvas->TileImageInt(*theme_toolbar,
305                       x + GetThemeBackgroundXInset(),
306                       dest_y - GetTabStripInsets(false).top, x,
307                       dest_y, w, theme_toolbar->height());
308
309  // Draw rounded corners for the tab.
310  gfx::ImageSkia* toolbar_left_mask =
311      tp->GetImageSkiaNamed(IDR_CONTENT_TOP_LEFT_CORNER_MASK);
312  gfx::ImageSkia* toolbar_right_mask =
313      tp->GetImageSkiaNamed(IDR_CONTENT_TOP_RIGHT_CORNER_MASK);
314
315  // We mask out the corners by using the DestinationIn transfer mode,
316  // which keeps the RGB pixels from the destination and the alpha from
317  // the source.
318  SkPaint paint;
319  paint.setXfermodeMode(SkXfermode::kDstIn_Mode);
320
321  // Mask out the top left corner.
322  canvas->DrawImageInt(*toolbar_left_mask, left_x, y, paint);
323
324  // Mask out the top right corner.
325  int right_x =
326      x + w + kContentEdgeShadowThickness - toolbar_right_mask->width();
327  canvas->DrawImageInt(*toolbar_right_mask, right_x, y, paint);
328
329  // Draw left edge.
330  canvas->DrawImageInt(*toolbar_left, left_x, y);
331
332  // Draw center edge.
333  canvas->TileImageInt(*toolbar_center, left_x + toolbar_left->width(), y,
334      right_x - (left_x + toolbar_left->width()), toolbar_center->height());
335
336  // Right edge.
337  canvas->DrawImageInt(*tp->GetImageSkiaNamed(IDR_CONTENT_TOP_RIGHT_CORNER),
338                       right_x, y);
339
340  // Draw the content/toolbar separator.
341  canvas->FillRect(
342      gfx::Rect(x + kClientEdgeThickness,
343                toolbar_bounds.bottom() - kClientEdgeThickness,
344                w - (2 * kClientEdgeThickness),
345                kClientEdgeThickness),
346      ThemeProperties::GetDefaultColor(
347          ThemeProperties::COLOR_TOOLBAR_SEPARATOR));
348}
349
350void GlassBrowserFrameView::PaintRestoredClientEdge(gfx::Canvas* canvas) {
351  ui::ThemeProvider* tp = GetThemeProvider();
352  gfx::Rect client_area_bounds = CalculateClientAreaBounds(width(), height());
353
354  // The client edges start below the toolbar upper corner images regardless
355  // of how tall the toolbar itself is.
356  int client_area_top = frame()->client_view()->y() +
357      browser_view()->GetToolbarBounds().y() +
358      tp->GetImageSkiaNamed(IDR_CONTENT_TOP_LEFT_CORNER)->height();
359  int client_area_bottom =
360      std::max(client_area_top, height() - NonClientBorderThickness());
361  int client_area_height = client_area_bottom - client_area_top;
362
363  // Draw the client edge images.
364  gfx::ImageSkia* right = tp->GetImageSkiaNamed(IDR_CONTENT_RIGHT_SIDE);
365  canvas->TileImageInt(*right, client_area_bounds.right(), client_area_top,
366                       right->width(), client_area_height);
367  canvas->DrawImageInt(
368      *tp->GetImageSkiaNamed(IDR_CONTENT_BOTTOM_RIGHT_CORNER),
369      client_area_bounds.right(), client_area_bottom);
370  gfx::ImageSkia* bottom = tp->GetImageSkiaNamed(IDR_CONTENT_BOTTOM_CENTER);
371  canvas->TileImageInt(*bottom, client_area_bounds.x(),
372      client_area_bottom, client_area_bounds.width(),
373      bottom->height());
374  gfx::ImageSkia* bottom_left =
375      tp->GetImageSkiaNamed(IDR_CONTENT_BOTTOM_LEFT_CORNER);
376  canvas->DrawImageInt(*bottom_left,
377      client_area_bounds.x() - bottom_left->width(), client_area_bottom);
378  gfx::ImageSkia* left = tp->GetImageSkiaNamed(IDR_CONTENT_LEFT_SIDE);
379  canvas->TileImageInt(*left, client_area_bounds.x() - left->width(),
380      client_area_top, left->width(), client_area_height);
381
382  // Draw the toolbar color so that the client edges show the right color even
383  // where not covered by the toolbar image.  NOTE: We do this after drawing the
384  // images because the images are meant to alpha-blend atop the frame whereas
385  // these rects are meant to be fully opaque, without anything overlaid.
386  SkColor toolbar_color = tp->GetColor(ThemeProperties::COLOR_TOOLBAR);
387  canvas->FillRect(gfx::Rect(client_area_bounds.x() - kClientEdgeThickness,
388      client_area_top, kClientEdgeThickness,
389      client_area_bottom + kClientEdgeThickness - client_area_top),
390      toolbar_color);
391  canvas->FillRect(gfx::Rect(client_area_bounds.x(), client_area_bottom,
392                             client_area_bounds.width(), kClientEdgeThickness),
393                   toolbar_color);
394  canvas->FillRect(gfx::Rect(client_area_bounds.right(), client_area_top,
395       kClientEdgeThickness,
396       client_area_bottom + kClientEdgeThickness - client_area_top),
397       toolbar_color);
398}
399
400void GlassBrowserFrameView::LayoutAvatar() {
401  // Even though the avatar is used for both incognito and profiles we always
402  // use the incognito icon to layout the avatar button. The profile icon
403  // can be customized so we can't depend on its size to perform layout.
404  gfx::ImageSkia incognito_icon = browser_view()->GetOTRAvatarIcon();
405
406  int avatar_x = NonClientBorderThickness() + kAvatarLeftSpacing;
407  // Move this avatar icon by the size of window controls to prevent it from
408  // being rendered over them in RTL languages. This code also needs to adjust
409  // the width of a tab strip to avoid decreasing this size twice. (See the
410  // comment in GetBoundsForTabStrip().)
411  if (base::i18n::IsRTL())
412    avatar_x += width() - frame()->GetMinimizeButtonOffset();
413
414  int avatar_bottom = GetTabStripInsets(false).top +
415      browser_view()->GetTabStripHeight() - kAvatarBottomSpacing;
416  int avatar_restored_y = avatar_bottom - incognito_icon.height();
417  int avatar_y = frame()->IsMaximized() ?
418      (NonClientTopBorderHeight(false) + kTabstripTopShadowThickness) :
419      avatar_restored_y;
420  avatar_bounds_.SetRect(avatar_x, avatar_y, incognito_icon.width(),
421      browser_view()->ShouldShowAvatar() ? (avatar_bottom - avatar_y) : 0);
422  if (avatar_button())
423    avatar_button()->SetBoundsRect(avatar_bounds_);
424}
425
426void GlassBrowserFrameView::LayoutClientView() {
427  client_view_bounds_ = CalculateClientAreaBounds(width(), height());
428}
429
430gfx::Insets GlassBrowserFrameView::GetClientAreaInsets() const {
431  if (!browser_view()->IsTabStripVisible())
432    return gfx::Insets();
433
434  const int top_height = NonClientTopBorderHeight(false);
435  const int border_thickness = NonClientBorderThickness();
436  return gfx::Insets(top_height,
437                     border_thickness,
438                     border_thickness,
439                     border_thickness);
440}
441
442gfx::Rect GlassBrowserFrameView::CalculateClientAreaBounds(int width,
443                                                           int height) const {
444  gfx::Rect bounds(0, 0, width, height);
445  bounds.Inset(GetClientAreaInsets());
446  return bounds;
447}
448
449void GlassBrowserFrameView::StartThrobber() {
450  if (!throbber_running_) {
451    throbber_running_ = true;
452    throbber_frame_ = 0;
453    InitThrobberIcons();
454    SendMessage(views::HWNDForWidget(frame()), WM_SETICON,
455                static_cast<WPARAM>(ICON_SMALL),
456                reinterpret_cast<LPARAM>(throbber_icons_[throbber_frame_]));
457  }
458}
459
460void GlassBrowserFrameView::StopThrobber() {
461  if (throbber_running_) {
462    throbber_running_ = false;
463
464    HICON frame_icon = NULL;
465
466    // Check if hosted BrowserView has a window icon to use.
467    if (browser_view()->ShouldShowWindowIcon()) {
468      gfx::ImageSkia icon = browser_view()->GetWindowIcon();
469      if (!icon.isNull())
470        frame_icon = IconUtil::CreateHICONFromSkBitmap(*icon.bitmap());
471    }
472
473    // Fallback to class icon.
474    if (!frame_icon) {
475      frame_icon = reinterpret_cast<HICON>(GetClassLongPtr(
476          views::HWNDForWidget(frame()), GCLP_HICONSM));
477    }
478
479    // This will reset the small icon which we set in the throbber code.
480    // WM_SETICON with NULL icon restores the icon for title bar but not
481    // for taskbar. See http://crbug.com/29996
482    SendMessage(views::HWNDForWidget(frame()), WM_SETICON,
483                static_cast<WPARAM>(ICON_SMALL),
484                reinterpret_cast<LPARAM>(frame_icon));
485  }
486}
487
488void GlassBrowserFrameView::DisplayNextThrobberFrame() {
489  throbber_frame_ = (throbber_frame_ + 1) % kThrobberIconCount;
490  SendMessage(views::HWNDForWidget(frame()), WM_SETICON,
491              static_cast<WPARAM>(ICON_SMALL),
492              reinterpret_cast<LPARAM>(throbber_icons_[throbber_frame_]));
493}
494
495void GlassBrowserFrameView::Observe(
496    int type,
497    const content::NotificationSource& source,
498    const content::NotificationDetails& details) {
499  switch (type) {
500    case chrome::NOTIFICATION_PROFILE_CACHED_INFO_CHANGED:
501      UpdateAvatarInfo();
502      break;
503    default:
504      NOTREACHED() << "Got a notification we didn't register for!";
505      break;
506  }
507}
508
509// static
510void GlassBrowserFrameView::InitThrobberIcons() {
511  static bool initialized = false;
512  if (!initialized) {
513    for (int i = 0; i < kThrobberIconCount; ++i) {
514      throbber_icons_[i] =
515          ui::LoadThemeIconFromResourcesDataDLL(IDI_THROBBER_01 + i);
516      DCHECK(throbber_icons_[i]);
517    }
518    initialized = true;
519  }
520}
521