blob: f77f34515dd190739e49d64309bea88f202821c0 [file] [log] [blame]
// Copyright 2014 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/contents_web_view.h"
#include "chrome/browser/themes/theme_properties.h"
#include "chrome/browser/ui/views/status_bubble_views.h"
#include "content/public/browser/web_contents.h"
#include "ui/base/theme_provider.h"
#include "ui/compositor/layer_tree_owner.h"
#include "ui/views/background.h"
#if defined(USE_AURA)
#include "ui/aura/window.h"
#include "ui/wm/core/window_util.h"
#endif
ContentsWebView::ContentsWebView(content::BrowserContext* browser_context)
: views::WebView(browser_context),
status_bubble_(nullptr) {
}
ContentsWebView::~ContentsWebView() {
}
void ContentsWebView::SetStatusBubble(StatusBubbleViews* status_bubble) {
status_bubble_ = status_bubble;
DCHECK(!status_bubble_ || status_bubble_->base_view() == this);
if (status_bubble_)
status_bubble_->Reposition();
}
bool ContentsWebView::GetNeedsNotificationWhenVisibleBoundsChange() const {
return true;
}
void ContentsWebView::OnVisibleBoundsChanged() {
if (status_bubble_)
status_bubble_->Reposition();
}
void ContentsWebView::ViewHierarchyChanged(
const ViewHierarchyChangedDetails& details) {
WebView::ViewHierarchyChanged(details);
if (details.is_add)
OnThemeChanged();
}
void ContentsWebView::OnThemeChanged() {
const ui::ThemeProvider* const theme = GetThemeProvider();
if (!theme)
return;
// Set the background color to a dark tint of the new tab page's background
// color. This is the color filled within the WebView's bounds when its child
// view is sized specially for fullscreen tab capture. See WebView header
// file comments for more details.
const int kBackgroundBrightness = 0x33; // 20%
const SkColor ntp_background =
theme->GetColor(ThemeProperties::COLOR_NTP_BACKGROUND);
set_background(views::Background::CreateSolidBackground(
SkColorGetR(ntp_background) * kBackgroundBrightness / 0xFF,
SkColorGetG(ntp_background) * kBackgroundBrightness / 0xFF,
SkColorGetB(ntp_background) * kBackgroundBrightness / 0xFF,
SkColorGetA(ntp_background)));
}
void ContentsWebView::OnLayerRecreated(ui::Layer* old_layer,
ui::Layer* new_layer) {
if (!cloned_layer_tree_)
return;
// Our layer has been recreated and we have a clone of the WebContents
// layer. Combined this means we're about to be destroyed and an animation is
// in effect. The animation cloned our layer, but it won't create another
// clone of the WebContents layer (|cloned_layer_tree_|). Another clone
// is not created as the clone has no owner (see CloneChildren()). Because we
// want the WebContents layer clone to be animated we move it to the
// old_layer, which is the layer the animation happens on. This animation ends
// up owning the layer (and all its descendants).
old_layer->Add(cloned_layer_tree_->release());
cloned_layer_tree_.reset();
}
void ContentsWebView::CloneWebContentsLayer() {
if (!web_contents())
return;
#if defined(USE_AURA)
// We don't need to clone the layers on non-Aura (Mac), because closing an
// NSWindow does not animate.
cloned_layer_tree_ = wm::RecreateLayers(web_contents()->GetNativeView());
#endif
if (!cloned_layer_tree_ || !cloned_layer_tree_->root()) {
cloned_layer_tree_.reset();
return;
}
SetPaintToLayer(true);
set_layer_owner_delegate(this);
// The cloned layer is in a different coordinate system them our layer (which
// is now the new parent of the cloned layer). Convert coordinates so that the
// cloned layer appears at the right location.
gfx::Point origin;
ui::Layer::ConvertPointToLayer(cloned_layer_tree_->root(), layer(), &origin);
cloned_layer_tree_->root()->SetBounds(
gfx::Rect(origin, cloned_layer_tree_->root()->bounds().size()));
layer()->Add(cloned_layer_tree_->root());
}
void ContentsWebView::DestroyClonedLayer() {
cloned_layer_tree_.reset();
SetPaintToLayer(false);
set_layer_owner_delegate(nullptr);
}