910e62b5创建于 1月15日历史提交
// Copyright 2012 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "ui/views/controls/native/native_view_host.h"

#include <utility>

#include "base/check.h"
#include "build/buildflag.h"
#include "ui/accessibility/ax_enums.mojom.h"
#include "ui/base/cursor/cursor.h"
#include "ui/base/metadata/metadata_impl_macros.h"
#include "ui/gfx/canvas.h"
#include "ui/gfx/native_ui_types.h"
#include "ui/views/controls/native/native_view_host_wrapper.h"
#include "ui/views/painter.h"
#include "ui/views/view_utils.h"
#include "ui/views/widget/widget.h"

namespace views {

////////////////////////////////////////////////////////////////////////////////
// NativeViewHost, public:

NativeViewHost::NativeViewHost() {
  set_suppress_default_focus_handling();
}

NativeViewHost::~NativeViewHost() {
  // As part of deleting NativeViewHostWrapper the native view is unparented.
  // Make sure the FocusManager is updated.
  ClearFocus();
}

void NativeViewHost::Attach(gfx::NativeView native_view) {
  DCHECK(native_view);
  DCHECK(!native_view_);
  native_view_ = native_view;
  native_wrapper_->AttachNativeView();
  InvalidateLayout();
  // The call to InvalidateLayout() triggers an async call to Layout(), which
  // updates the visibility of the NativeView. The call to Layout() only happens
  // if |this| is drawn. Call hide if not drawn as otherwise the NativeView
  // could be visible when |this| is not.
  if (!IsDrawn()) {
    native_wrapper_->HideWidget();
  }

  Widget* widget = Widget::GetWidgetForNativeView(native_view);
  if (widget) {
    widget->SetNativeWindowProperty(kWidgetNativeViewHostKey, this);
  }
}

void NativeViewHost::Detach() {
  Detach(false);
}

void NativeViewHost::SetParentAccessible(gfx::NativeViewAccessible accessible) {
  if (!native_wrapper_) {
    return;
  }
  native_wrapper_->SetParentAccessible(accessible);
}

gfx::NativeViewAccessible NativeViewHost::GetParentAccessible() {
  if (!native_wrapper_) {
    return gfx::NativeViewAccessible();
  }
  return native_wrapper_->GetParentAccessible();
}

bool NativeViewHost::SetCornerRadii(const gfx::RoundedCornersF& corner_radii) {
  if (!native_wrapper_) {
    return false;
  }
  return native_wrapper_->SetCornerRadii(corner_radii);
}

void NativeViewHost::SetHitTestTopInset(int top_inset) {
  native_wrapper_->SetHitTestTopInset(top_inset);
}

int NativeViewHost::GetHitTestTopInset() const {
  return native_wrapper_->GetHitTestTopInset();
}

void NativeViewHost::SetNativeViewSize(const gfx::Size& size) {
  if (native_view_size_ == size) {
    return;
  }
  native_view_size_ = size;
  InvalidateLayout();
}

gfx::NativeView NativeViewHost::GetNativeViewContainer() const {
  return native_view_ ? native_wrapper_->GetNativeViewContainer()
                      : gfx::NativeView();
}

void NativeViewHost::NativeViewDestroyed() {
  // Detach so we can clear our state and notify the native_wrapper_ to release
  // ref on the native view.
  Detach(true);
}

void NativeViewHost::SetBackgroundColorWhenClipped(
    std::optional<SkColor> color) {
  background_color_when_clipped_ = color;
}

ui::Layer* NativeViewHost::GetUILayer() {
  return native_wrapper_->GetUILayer();
}

////////////////////////////////////////////////////////////////////////////////
// NativeViewHost, View overrides:

void NativeViewHost::Layout(PassKey) {
  if (!native_view_ || !native_wrapper_.get()) {
    return;
  }

  gfx::Rect vis_bounds = GetVisibleBounds();
  bool visible = !vis_bounds.IsEmpty();

#if !BUILDFLAG(IS_MAC)
  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();
    }
  }
#endif
  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());
    gfx::Size native_view_size =
        native_view_size_.IsEmpty() ? local_bounds.size() : native_view_size_;
    native_wrapper_->ShowWidget(local_bounds.x(), local_bounds.y(),
                                local_bounds.width(), local_bounds.height(),
                                native_view_size.width(),
                                native_view_size.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.
  //
  // Affected views should set the desired color using
  // SetBackgroundColorWhenClipped(), otherwise the background is left
  // transparent to let the lower layers show through.
  if (native_wrapper_->HasInstalledClip()) {
    if (background_color_when_clipped_) {
      canvas->FillRect(GetLocalBounds(), *background_color_when_clipped_);
    }
  }
}

void NativeViewHost::VisibilityChanged(View* starting_from, bool is_visible) {
  if (is_visible) {
    // This does not use InvalidateLayout() to ensure the visibility state is
    // correctly set.
    DeprecatedLayoutImmediately();
  } else {
    if (native_view_ && native_wrapper_) {
      native_wrapper_->HideWidget();
    }
  }
}

bool NativeViewHost::GetNeedsNotificationWhenVisibleBoundsChange() 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() {
  InvalidateLayout();
}

void NativeViewHost::ViewHierarchyChanged(
    const ViewHierarchyChangedDetails& details) {
  views::Widget* this_widget = GetWidget();

  // A non-NULL |details.move_view| indicates a move operation i.e. |this| is
  // is being reparented.  If the previous and new parents belong to the same
  // widget, don't remove |this| from the widget.  This saves resources from
  // removing from widget and immediately followed by adding to widget; in
  // particular, there wouldn't be spurious visibilitychange events for web
  // contents of |WebView|.
  if (details.move_view && this_widget &&
      details.move_view->GetWidget() == this_widget) {
    return;
  }

  if (details.is_add && this_widget) {
    if (!native_wrapper_.get()) {
      native_wrapper_.reset(NativeViewHostWrapper::CreateWrapper(this));
    }
    native_wrapper_->AddedToWidget();
  } else if (!details.is_add && native_wrapper_) {
    native_wrapper_->RemovedFromWidget();
  }
}

void NativeViewHost::OnFocus() {
  if (native_view_) {
    native_wrapper_->SetFocus();
  }
  NotifyAccessibilityEventDeprecated(ax::mojom::Event::kFocus, 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();
}

ui::Cursor NativeViewHost::GetCursor(const ui::MouseEvent& event) {
  return native_wrapper_->GetCursor(event.x(), event.y());
}

void NativeViewHost::SetVisible(bool visible) {
  if (native_view_) {
    native_wrapper_->SetVisible(visible);
  }
  View::SetVisible(visible);
}

bool NativeViewHost::OnMousePressed(const ui::MouseEvent& event) {
  // In the typical case the attached NativeView receives the events directly
  // from the system and this function is not called. There are scenarios
  // where that may not happen. For example, if the NativeView is configured
  // not to receive events, then this function will be called. An additional
  // scenario is if the WidgetDelegate overrides
  // ShouldDescendIntoChildForEventHandling(). In that case the NativeView
  // will not receive the events, and this function will be called. Regardless,
  // this function does not need to forward to the NativeView, because it is
  // expected to be done by the system, and the only cases where this is called
  // is if the NativeView should not receive events.
  return View::OnMousePressed(event);
}

////////////////////////////////////////////////////////////////////////////////
// NativeViewHost, private:

void NativeViewHost::Detach(bool destroyed) {
  if (native_view_) {
    if (!destroyed) {
      Widget* widget = Widget::GetWidgetForNativeView(native_view_);
      if (widget) {
        widget->SetNativeWindowProperty(kWidgetNativeViewHostKey, nullptr);
      }
      ClearFocus();
    }
    native_wrapper_->NativeViewDetaching(destroyed);
    native_view_ = gfx::NativeView();
  }
}

void NativeViewHost::ClearFocus() {
  FocusManager* focus_manager = GetFocusManager();
  if (!focus_manager || !focus_manager->GetFocusedView()) {
    return;
  }

  Widget::Widgets widgets = Widget::GetAllChildWidgets(native_view());
  for (Widget* widget : widgets) {
    focus_manager->ViewRemoved(widget->GetRootView());
    if (!focus_manager->GetFocusedView()) {
      return;
    }
  }
}

BEGIN_METADATA(NativeViewHost)
END_METADATA

}  // namespace views