blob: e58e090bb074a97e6389f9653c2ce92fc5a4705b [file] [log] [blame]
// 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 "chrome/browser/ui/views/frame/browser_view_layout.h"
#include "chrome/browser/ui/browser_finder.h"
#include "chrome/browser/ui/find_bar/find_bar.h"
#include "chrome/browser/ui/find_bar/find_bar_controller.h"
#include "chrome/browser/ui/view_ids.h"
#include "chrome/browser/ui/views/bookmarks/bookmark_bar_view.h"
#include "chrome/browser/ui/views/download/download_shelf_view.h"
#include "chrome/browser/ui/views/frame/browser_frame.h"
#include "chrome/browser/ui/views/frame/browser_view.h"
#include "chrome/browser/ui/views/frame/contents_container.h"
#include "chrome/browser/ui/views/infobars/infobar_container_view.h"
#include "chrome/browser/ui/views/location_bar/location_bar_container.h"
#include "chrome/browser/ui/views/tabs/tab_strip.h"
#include "chrome/browser/ui/views/toolbar_view.h"
#include "ui/base/hit_test.h"
#include "ui/gfx/point.h"
#include "ui/gfx/scrollbar_size.h"
#include "ui/gfx/size.h"
#include "ui/views/controls/single_split_view.h"
namespace {
// The visible height of the shadow above the tabs. Clicks in this area are
// treated as clicks to the frame, rather than clicks to the tab.
const int kTabShadowSize = 2;
// The vertical overlap between the TabStrip and the Toolbar.
const int kToolbarTabStripVerticalOverlap = 3;
// The number of pixels the bookmark bar should overlap the spacer by if the
// spacer is visible.
const int kSpacerBookmarkBarOverlap = 1;
// The number of pixels the metro switcher is offset from the right edge.
const int kWindowSwitcherOffsetX = 7;
// Combines View::ConvertPointToTarget and View::HitTest for a given |point|.
// Converts |point| from |src| to |dst| and hit tests it against |dst|. The
// converted |point| can then be retrieved and used for additional tests.
bool ConvertedHitTest(views::View* src, views::View* dst, gfx::Point* point) {
DCHECK(src);
DCHECK(dst);
DCHECK(point);
views::View::ConvertPointToTarget(src, dst, point);
return dst->HitTestPoint(*point);
}
} // namespace
////////////////////////////////////////////////////////////////////////////////
// BrowserViewLayout, public:
BrowserViewLayout::BrowserViewLayout()
: tabstrip_(NULL),
toolbar_(NULL),
contents_split_(NULL),
contents_container_(NULL),
infobar_container_(NULL),
download_shelf_(NULL),
active_bookmark_bar_(NULL),
browser_view_(NULL),
find_bar_y_(0) {
}
BrowserViewLayout::~BrowserViewLayout() {
}
gfx::Size BrowserViewLayout::GetMinimumSize() {
gfx::Size tabstrip_size(
browser()->SupportsWindowFeature(Browser::FEATURE_TABSTRIP) ?
tabstrip_->GetMinimumSize() : gfx::Size());
BrowserNonClientFrameView::TabStripInsets tab_strip_insets(
browser_view_->frame()->GetTabStripInsets(false));
gfx::Size toolbar_size(
(browser()->SupportsWindowFeature(Browser::FEATURE_TOOLBAR) ||
browser()->SupportsWindowFeature(Browser::FEATURE_LOCATIONBAR)) ?
toolbar_->GetMinimumSize() : gfx::Size());
if (tabstrip_size.height() && toolbar_size.height())
toolbar_size.Enlarge(0, -kToolbarTabStripVerticalOverlap);
gfx::Size bookmark_bar_size;
if (active_bookmark_bar_ &&
browser()->SupportsWindowFeature(Browser::FEATURE_BOOKMARKBAR)) {
bookmark_bar_size = active_bookmark_bar_->GetMinimumSize();
bookmark_bar_size.Enlarge(0,
-(views::NonClientFrameView::kClientEdgeThickness +
active_bookmark_bar_->GetToolbarOverlap(true)));
}
gfx::Size contents_size(contents_split_->GetMinimumSize());
int min_height = tabstrip_size.height() + toolbar_size.height() +
bookmark_bar_size.height() + contents_size.height();
int widths[] = {
tabstrip_size.width() + tab_strip_insets.left + tab_strip_insets.right,
toolbar_size.width(),
bookmark_bar_size.width(),
contents_size.width() };
int min_width = *std::max_element(&widths[0], &widths[arraysize(widths)]);
return gfx::Size(min_width, min_height);
}
gfx::Rect BrowserViewLayout::GetFindBarBoundingBox() const {
// This function returns the area the Find Bar can be laid out
// within. This basically implies the "user-perceived content
// area" of the browser window excluding the vertical
// scrollbar. This is not quite so straightforward as positioning
// based on the TabContentsContainer since the BookmarkBarView may
// be visible but not persistent (in the New Tab case) and we
// position the Find Bar over the top of it in that case since the
// BookmarkBarView is not _visually_ connected to the Toolbar.
// First determine the bounding box of the content area in Widget
// coordinates.
gfx::Rect bounding_box = contents_container_->ConvertRectToWidget(
contents_container_->GetLocalBounds());
// Adjust the position and size of the bounding box by the find bar offset
// calculated during the last Layout.
int height_delta = find_bar_y_ - bounding_box.y();
bounding_box.set_y(find_bar_y_);
bounding_box.set_height(std::max(0, bounding_box.height() + height_delta));
// Finally decrease the width of the bounding box by the width of
// the vertical scroll bar.
int scrollbar_width = gfx::scrollbar_size();
bounding_box.set_width(std::max(0, bounding_box.width() - scrollbar_width));
if (base::i18n::IsRTL())
bounding_box.set_x(bounding_box.x() + scrollbar_width);
return bounding_box;
}
bool BrowserViewLayout::IsPositionInWindowCaption(
const gfx::Point& point) {
gfx::Point tabstrip_point(point);
views::View::ConvertPointToTarget(browser_view_, tabstrip_, &tabstrip_point);
return tabstrip_->IsPositionInWindowCaption(tabstrip_point);
}
int BrowserViewLayout::NonClientHitTest(
const gfx::Point& point) {
// Since the TabStrip only renders in some parts of the top of the window,
// the un-obscured area is considered to be part of the non-client caption
// area of the window. So we need to treat hit-tests in these regions as
// hit-tests of the titlebar.
views::View* parent = browser_view_->parent();
gfx::Point point_in_browser_view_coords(point);
views::View::ConvertPointToTarget(
parent, browser_view_, &point_in_browser_view_coords);
gfx::Point test_point(point);
// Determine if the TabStrip exists and is capable of being clicked on. We
// might be a popup window without a TabStrip.
if (browser_view_->IsTabStripVisible()) {
// See if the mouse pointer is within the bounds of the TabStrip.
if (ConvertedHitTest(parent, tabstrip_, &test_point)) {
if (tabstrip_->IsPositionInWindowCaption(test_point))
return HTCAPTION;
return HTCLIENT;
}
// The top few pixels of the TabStrip are a drop-shadow - as we're pretty
// starved of dragable area, let's give it to window dragging (this also
// makes sense visually).
if (!browser_view_->IsMaximized() &&
(point_in_browser_view_coords.y() <
(tabstrip_->y() + kTabShadowSize))) {
// We return HTNOWHERE as this is a signal to our containing
// NonClientView that it should figure out what the correct hit-test
// code is given the mouse position...
return HTNOWHERE;
}
}
// If the point's y coordinate is below the top of the toolbar and otherwise
// within the bounds of this view, the point is considered to be within the
// client area.
gfx::Rect bv_bounds = browser_view_->bounds();
bv_bounds.Offset(0, toolbar_->y());
bv_bounds.set_height(bv_bounds.height() - toolbar_->y());
if (bv_bounds.Contains(point))
return HTCLIENT;
// If the point's y coordinate is above the top of the toolbar, but not in
// the tabstrip (per previous checking in this function), then we consider it
// in the window caption (e.g. the area to the right of the tabstrip
// underneath the window controls). However, note that we DO NOT return
// HTCAPTION here, because when the window is maximized the window controls
// will fall into this space (since the BrowserView is sized to entire size
// of the window at that point), and the HTCAPTION value will cause the
// window controls not to work. So we return HTNOWHERE so that the caller
// will hit-test the window controls before finally falling back to
// HTCAPTION.
bv_bounds = browser_view_->bounds();
bv_bounds.set_height(toolbar_->y());
if (bv_bounds.Contains(point))
return HTNOWHERE;
// If the point is somewhere else, delegate to the default implementation.
return browser_view_->views::ClientView::NonClientHitTest(point);
}
//////////////////////////////////////////////////////////////////////////////
// BrowserViewLayout, views::LayoutManager implementation:
void BrowserViewLayout::Installed(views::View* host) {
toolbar_ = NULL;
contents_split_ = NULL;
contents_container_ = NULL;
infobar_container_ = NULL;
download_shelf_ = NULL;
active_bookmark_bar_ = NULL;
tabstrip_ = NULL;
browser_view_ = static_cast<BrowserView*>(host);
}
void BrowserViewLayout::Uninstalled(views::View* host) {}
void BrowserViewLayout::ViewAdded(views::View* host, views::View* view) {
switch (view->id()) {
case VIEW_ID_CONTENTS_SPLIT: {
contents_split_ = static_cast<views::SingleSplitView*>(view);
// We're installed as the LayoutManager before BrowserView creates the
// contents, so we have to set contents_container_ here rather than in
// Installed.
contents_container_ = browser_view_->contents_;
break;
}
case VIEW_ID_INFO_BAR_CONTAINER:
infobar_container_ = view;
break;
case VIEW_ID_DOWNLOAD_SHELF:
download_shelf_ = static_cast<DownloadShelfView*>(view);
break;
case VIEW_ID_BOOKMARK_BAR:
active_bookmark_bar_ = static_cast<BookmarkBarView*>(view);
break;
case VIEW_ID_TOOLBAR:
toolbar_ = static_cast<ToolbarView*>(view);
break;
case VIEW_ID_TAB_STRIP:
tabstrip_ = static_cast<TabStrip*>(view);
break;
}
}
void BrowserViewLayout::ViewRemoved(views::View* host, views::View* view) {
switch (view->id()) {
case VIEW_ID_BOOKMARK_BAR:
active_bookmark_bar_ = NULL;
break;
}
}
void BrowserViewLayout::Layout(views::View* host) {
vertical_layout_rect_ = browser_view_->GetLocalBounds();
int top = LayoutTabStripRegion();
if (browser_view_->IsTabStripVisible()) {
int x = tabstrip_->GetMirroredX() + browser_view_->GetMirroredX() +
browser_view_->frame()->GetThemeBackgroundXInset();
tabstrip_->SetBackgroundOffset(gfx::Point(x,
browser_view_->frame()->GetTabStripInsets(false).top));
}
top = LayoutToolbar(top);
top = LayoutBookmarkAndInfoBars(top);
int bottom = LayoutDownloadShelf(browser_view_->height());
int active_top_margin = GetTopMarginForActiveContent();
top -= active_top_margin;
contents_container_->SetActiveTopMargin(active_top_margin);
LayoutTabContents(top, bottom);
// This must be done _after_ we lay out the WebContents since this
// code calls back into us to find the bounding box the find bar
// must be laid out within, and that code depends on the
// TabContentsContainer's bounds being up to date.
if (browser()->HasFindBarController()) {
browser()->GetFindBarController()->find_bar()->MoveWindowIfNecessary(
gfx::Rect(), true);
}
// NTP needs to layout the search box now that we have the contents bounds.
toolbar_->LayoutForSearch();
}
// Return the preferred size which is the size required to give each
// children their respective preferred size.
gfx::Size BrowserViewLayout::GetPreferredSize(views::View* host) {
return gfx::Size();
}
//////////////////////////////////////////////////////////////////////////////
// BrowserViewLayout, private:
Browser* BrowserViewLayout::browser() {
return browser_view_->browser();
}
const Browser* BrowserViewLayout::browser() const {
return browser_view_->browser();
}
int BrowserViewLayout::LayoutTabStripRegion() {
if (!browser_view_->IsTabStripVisible()) {
tabstrip_->SetVisible(false);
tabstrip_->SetBounds(0, 0, 0, 0);
return 0;
}
// This retrieves the bounds for the tab strip based on whether or not we show
// anything to the left of it, like the incognito avatar.
gfx::Rect tabstrip_bounds(
browser_view_->frame()->GetBoundsForTabStrip(tabstrip_));
gfx::Point tabstrip_origin(tabstrip_bounds.origin());
views::View::ConvertPointToTarget(browser_view_->parent(), browser_view_,
&tabstrip_origin);
tabstrip_bounds.set_origin(tabstrip_origin);
tabstrip_->SetVisible(true);
tabstrip_->SetBoundsRect(tabstrip_bounds);
int bottom = tabstrip_bounds.bottom();
// The metro window switcher sits at the far right edge of the tabstrip
// a |kWindowSwitcherOffsetX| pixels from the right edge.
// Only visible if there is more than one type of window to switch between.
// TODO(mad): update this code when more window types than just incognito
// and regular are available.
views::Button* switcher_button = browser_view_->window_switcher_button_;
if (switcher_button) {
if (browser()->profile()->HasOffTheRecordProfile() &&
browser::FindBrowserWithProfile(
browser()->profile()->GetOriginalProfile(),
browser()->host_desktop_type()) != NULL) {
switcher_button->SetVisible(true);
int width = browser_view_->width();
gfx::Size ps = switcher_button->GetPreferredSize();
if (width > ps.width()) {
switcher_button->SetBounds(width - ps.width() - kWindowSwitcherOffsetX,
0,
ps.width(),
ps.height());
}
} else {
// We hide the button if the incognito profile is not alive.
// Note that Layout() is not called to all browser windows automatically
// when a profile goes away but we rely in the metro_driver.dll to call
// ::SetWindowPos( , .. SWP_SHOWWINDOW) which causes this function to
// be called again. This works both in showing or hidding the button.
switcher_button->SetVisible(false);
}
}
return bottom;
}
int BrowserViewLayout::LayoutToolbar(int top) {
int browser_view_width = vertical_layout_rect_.width();
bool toolbar_visible = browser_view_->IsToolbarVisible();
toolbar_->location_bar()->SetLocationEntryFocusable(toolbar_visible);
int y = top;
y -= (toolbar_visible && browser_view_->IsTabStripVisible()) ?
kToolbarTabStripVerticalOverlap : 0;
int height = toolbar_visible ? toolbar_->GetPreferredSize().height() : 0;
toolbar_->SetVisible(toolbar_visible);
toolbar_->location_bar_container()->SetVisible(toolbar_visible);
toolbar_->SetBounds(vertical_layout_rect_.x(), y, browser_view_width, height);
return y + height;
}
int BrowserViewLayout::LayoutBookmarkAndInfoBars(int top) {
find_bar_y_ = top + browser_view_->y() - 1;
if (active_bookmark_bar_) {
// If we're showing the Bookmark bar in detached style, then we
// need to show any Info bar _above_ the Bookmark bar, since the
// Bookmark bar is styled to look like it's part of the page.
if (active_bookmark_bar_->IsDetached())
return LayoutBookmarkBar(LayoutInfoBar(top));
// Otherwise, Bookmark bar first, Info bar second.
top = std::max(toolbar_->bounds().bottom(), LayoutBookmarkBar(top));
}
find_bar_y_ = top + browser_view_->y() - 1;
return LayoutInfoBar(top);
}
int BrowserViewLayout::LayoutBookmarkBar(int top) {
DCHECK(active_bookmark_bar_);
int y = top;
if (!browser_view_->IsBookmarkBarVisible()) {
active_bookmark_bar_->SetVisible(false);
active_bookmark_bar_->SetBounds(0, y, browser_view_->width(), 0);
return y;
}
active_bookmark_bar_->set_infobar_visible(InfobarVisible());
int bookmark_bar_height = active_bookmark_bar_->GetPreferredSize().height();
y -= views::NonClientFrameView::kClientEdgeThickness +
active_bookmark_bar_->GetToolbarOverlap(false);
active_bookmark_bar_->SetVisible(true);
active_bookmark_bar_->SetBounds(vertical_layout_rect_.x(), y,
vertical_layout_rect_.width(),
bookmark_bar_height);
return y + bookmark_bar_height;
}
int BrowserViewLayout::LayoutInfoBar(int top) {
// Raise the |infobar_container_| by its vertical overlap.
infobar_container_->SetVisible(InfobarVisible());
int height;
int overlapped_top = top -
static_cast<InfoBarContainerView*>(infobar_container_)->
GetVerticalOverlap(&height);
infobar_container_->SetBounds(vertical_layout_rect_.x(),
overlapped_top,
vertical_layout_rect_.width(),
height);
return overlapped_top + height;
}
void BrowserViewLayout::LayoutTabContents(int top, int bottom) {
// The ultimate idea is to calculate bounds and reserved areas for all
// contents views first and then resize them all, so every view
// (and its contents) is resized and laid out only once.
// The views hierarcy (see browser_view.h for more details):
// contents_split_ -> [contents_container_ | devtools]
gfx::Rect contents_bounds;
gfx::Rect devtools_bounds;
gfx::Rect contents_split_bounds(vertical_layout_rect_.x(), top,
vertical_layout_rect_.width(),
std::max(0, bottom - top));
gfx::Point contents_split_offset(
contents_split_bounds.x() - contents_split_->bounds().x(),
contents_split_bounds.y() - contents_split_->bounds().y());
// Layout resize corner and calculate reserved contents rects here as all
// contents view bounds are already determined, but not yet set at this point,
// so contents will be laid out once at most.
gfx::Rect browser_reserved_rect;
if (!browser_view_->frame_->IsMaximized() &&
!browser_view_->frame_->IsFullscreen()) {
gfx::Size resize_corner_size = browser_view_->GetResizeCornerSize();
if (!resize_corner_size.IsEmpty()) {
gfx::Rect bounds = browser_view_->GetContentsBounds();
gfx::Point resize_corner_origin(
bounds.right() - resize_corner_size.width(),
bounds.bottom() - resize_corner_size.height());
browser_reserved_rect =
gfx::Rect(resize_corner_origin, resize_corner_size);
}
}
// Now it's safe to actually resize all contents views in the hierarchy.
contents_split_->SetBoundsRect(contents_split_bounds);
}
int BrowserViewLayout::GetTopMarginForActiveContent() {
if (!active_bookmark_bar_ || !browser_view_->IsBookmarkBarVisible() ||
!active_bookmark_bar_->IsDetached()) {
return 0;
}
if (contents_split_->child_at(1) && contents_split_->child_at(1)->visible())
return 0;
// Adjust for separator.
return active_bookmark_bar_->height() -
views::NonClientFrameView::kClientEdgeThickness;
}
int BrowserViewLayout::LayoutDownloadShelf(int bottom) {
// Re-layout the shelf either if it is visible or if its close animation
// is currently running.
if (browser_view_->IsDownloadShelfVisible() ||
(download_shelf_ && download_shelf_->IsClosing())) {
bool visible = browser()->SupportsWindowFeature(
Browser::FEATURE_DOWNLOADSHELF);
DCHECK(download_shelf_);
int height = visible ? download_shelf_->GetPreferredSize().height() : 0;
download_shelf_->SetVisible(visible);
download_shelf_->SetBounds(vertical_layout_rect_.x(), bottom - height,
vertical_layout_rect_.width(), height);
download_shelf_->Layout();
bottom -= height;
}
return bottom;
}
bool BrowserViewLayout::InfobarVisible() const {
// NOTE: Can't check if the size IsEmpty() since it's always 0-width.
return browser()->SupportsWindowFeature(Browser::FEATURE_INFOBAR) &&
(infobar_container_->GetPreferredSize().height() != 0);
}