blob: 728f70c8e0bee763169a20f57b4cb69d551d7693 [file] [log] [blame]
// Copyright (c) 2011 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 "views/controls/native/native_view_host.h"
#include "base/logging.h"
#include "ui/gfx/canvas.h"
#include "views/controls/native/native_view_host_views.h"
#include "views/controls/native/native_view_host_wrapper.h"
#include "views/widget/widget.h"
namespace views {
// static
const char NativeViewHost::kViewClassName[] = "views/NativeViewHost";
#if defined(OS_LINUX)
// GTK renders the focus.
// static
const bool NativeViewHost::kRenderNativeControlFocus = false;
#else
// static
const bool NativeViewHost::kRenderNativeControlFocus = true;
#endif
////////////////////////////////////////////////////////////////////////////////
// NativeViewHost, public:
NativeViewHost::NativeViewHost()
: native_view_(NULL),
views_view_(NULL),
fast_resize_(false),
focus_view_(NULL) {
}
NativeViewHost::~NativeViewHost() {
}
void NativeViewHost::Attach(gfx::NativeView native_view) {
DCHECK(native_view);
DCHECK(!native_view_);
DCHECK(!views_view_);
native_view_ = native_view;
// If set_focus_view() has not been invoked, this view is the one that should
// be seen as focused when the native view receives focus.
if (!focus_view_)
focus_view_ = this;
native_wrapper_->NativeViewAttached();
}
void NativeViewHost::AttachToView(View* view) {
if (view == views_view_)
return;
DCHECK(view);
DCHECK(!native_view_);
DCHECK(!views_view_);
native_wrapper_.reset(new NativeViewHostViews(this));
views_view_ = view;
// If set_focus_view() has not been invoked, this view is the one that should
// be seen as focused when the native view receives focus.
if (!focus_view_)
focus_view_ = this;
native_wrapper_->NativeViewAttached();
}
void NativeViewHost::Detach() {
Detach(false);
}
void NativeViewHost::SetPreferredSize(const gfx::Size& size) {
preferred_size_ = size;
PreferredSizeChanged();
}
void NativeViewHost::NativeViewDestroyed() {
// Detach so we can clear our state and notify the native_wrapper_ to release
// ref on the native view.
Detach(true);
}
////////////////////////////////////////////////////////////////////////////////
// NativeViewHost, View overrides:
gfx::Size NativeViewHost::GetPreferredSize() {
return preferred_size_;
}
void NativeViewHost::Layout() {
if ((!native_view_ && !views_view_) || !native_wrapper_.get())
return;
gfx::Rect vis_bounds = GetVisibleBounds();
bool visible = !vis_bounds.IsEmpty();
if (visible && !fast_resize_) {
if (vis_bounds.size() != size()) {
// Only a portion of the Widget is really visible.
int x = vis_bounds.x();
int y = vis_bounds.y();
native_wrapper_->InstallClip(x, y, vis_bounds.width(),
vis_bounds.height());
} else if (native_wrapper_->HasInstalledClip()) {
// The whole widget is visible but we installed a clip on the widget,
// uninstall it.
native_wrapper_->UninstallClip();
}
}
if (visible) {
// Since widgets know nothing about the View hierarchy (they are direct
// children of the Widget that hosts our View hierarchy) they need to be
// positioned in the coordinate system of the Widget, not the current
// view. Also, they should be positioned respecting the border insets
// of the native view.
gfx::Rect local_bounds = ConvertRectToWidget(GetContentsBounds());
native_wrapper_->ShowWidget(local_bounds.x(), local_bounds.y(),
local_bounds.width(),
local_bounds.height());
} else {
native_wrapper_->HideWidget();
}
}
void NativeViewHost::OnPaint(gfx::Canvas* canvas) {
// Paint background if there is one. NativeViewHost needs to paint
// a background when it is hosted in a TabbedPane. For Gtk implementation,
// NativeTabbedPaneGtk uses a NativeWidgetGtk as page container and because
// NativeWidgetGtk hook "expose" with its root view's paint, we need to
// fill the content. Otherwise, the tab page's background is not properly
// cleared. For Windows case, it appears okay to not paint background because
// we don't have a container window in-between. However if you want to use
// customized background, then this becomes necessary.
OnPaintBackground(canvas);
// The area behind our window is black, so during a fast resize (where our
// content doesn't draw over the full size of our native view, and the native
// view background color doesn't show up), we need to cover that blackness
// with something so that fast resizes don't result in black flash.
//
// It would be nice if this used some approximation of the page's
// current background color.
if (native_wrapper_->HasInstalledClip())
canvas->FillRectInt(SK_ColorWHITE, 0, 0, width(), height());
}
void NativeViewHost::VisibilityChanged(View* starting_from, bool is_visible) {
Layout();
}
bool NativeViewHost::NeedsNotificationWhenVisibleBoundsChange() const {
// The native widget is placed relative to the root. As such, we need to
// know when the position of any ancestor changes, or our visibility relative
// to other views changed as it'll effect our position relative to the root.
return true;
}
void NativeViewHost::OnVisibleBoundsChanged() {
Layout();
}
void NativeViewHost::ViewHierarchyChanged(bool is_add, View* parent,
View* child) {
if (is_add && GetWidget()) {
if (!native_wrapper_.get())
native_wrapper_.reset(NativeViewHostWrapper::CreateWrapper(this));
native_wrapper_->AddedToWidget();
} else if (!is_add) {
native_wrapper_->RemovedFromWidget();
}
}
std::string NativeViewHost::GetClassName() const {
return kViewClassName;
}
void NativeViewHost::OnFocus() {
native_wrapper_->SetFocus();
GetWidget()->NotifyAccessibilityEvent(
this, ui::AccessibilityTypes::EVENT_FOCUS, true);
}
gfx::NativeViewAccessible NativeViewHost::GetNativeViewAccessible() {
if (native_wrapper_.get()) {
gfx::NativeViewAccessible accessible_view =
native_wrapper_->GetNativeViewAccessible();
if (accessible_view)
return accessible_view;
}
return View::GetNativeViewAccessible();
}
////////////////////////////////////////////////////////////////////////////////
// NativeViewHost, private:
void NativeViewHost::Detach(bool destroyed) {
DCHECK(native_view_ || views_view_);
native_wrapper_->NativeViewDetaching(destroyed);
native_view_ = NULL;
views_view_ = NULL;
}
} // namespace views