blob: 10086ffc8ac8ec668d01023d38f273707e6efd5e [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 "content/browser/web_contents/web_contents_view_android.h"
#include "base/android/build_info.h"
#include "base/android/jni_android.h"
#include "base/android/jni_string.h"
#include "base/logging.h"
#include "cc/layers/layer.h"
#include "content/browser/accessibility/browser_accessibility_manager_android.h"
#include "content/browser/android/content_feature_list.h"
#include "content/browser/android/content_ui_event_handler.h"
#include "content/browser/android/gesture_listener_manager.h"
#include "content/browser/android/select_popup.h"
#include "content/browser/android/selection/selection_popup_controller.h"
#include "content/browser/frame_host/interstitial_page_impl.h"
#include "content/browser/renderer_host/display_util.h"
#include "content/browser/renderer_host/render_view_host_factory.h"
#include "content/browser/renderer_host/render_view_host_impl.h"
#include "content/browser/renderer_host/render_widget_host_view_android.h"
#include "content/browser/web_contents/web_contents_impl.h"
#include "content/public/android/jar_jni/DragEvent_jni.h"
#include "content/public/browser/android/synchronous_compositor.h"
#include "content/public/browser/render_widget_host.h"
#include "content/public/browser/web_contents_delegate.h"
#include "content/public/common/content_features.h"
#include "content/public/common/drop_data.h"
#include "ui/android/overscroll_refresh_handler.h"
#include "ui/base/clipboard/clipboard.h"
#include "ui/base/clipboard/clipboard_constants.h"
#include "ui/display/screen.h"
#include "ui/events/android/drag_event_android.h"
#include "ui/events/android/gesture_event_android.h"
#include "ui/events/android/key_event_android.h"
#include "ui/events/android/motion_event_android.h"
#include "ui/gfx/android/java_bitmap.h"
#include "ui/gfx/image/image_skia.h"
using base::android::AttachCurrentThread;
using base::android::ConvertJavaStringToUTF16;
using base::android::ConvertUTF16ToJavaString;
using base::android::JavaRef;
using base::android::ScopedJavaLocalRef;
namespace content {
namespace {
// True if we want to disable Android native event batching and use
// compositor event queue.
bool ShouldRequestUnbufferedDispatch() {
static bool should_request_unbuffered_dispatch =
base::FeatureList::IsEnabled(features::kRequestUnbufferedDispatch) &&
base::android::BuildInfo::GetInstance()->sdk_int() >=
base::android::SDK_VERSION_LOLLIPOP &&
!content::GetContentClient()->UsingSynchronousCompositing();
return should_request_unbuffered_dispatch;
}
}
// static
void SynchronousCompositor::SetClientForWebContents(
WebContents* contents,
SynchronousCompositorClient* client) {
DCHECK(contents);
DCHECK(client);
WebContentsViewAndroid* wcva = static_cast<WebContentsViewAndroid*>(
static_cast<WebContentsImpl*>(contents)->GetView());
DCHECK(!wcva->synchronous_compositor_client());
wcva->set_synchronous_compositor_client(client);
RenderWidgetHostViewAndroid* rwhv = static_cast<RenderWidgetHostViewAndroid*>(
contents->GetRenderWidgetHostView());
if (rwhv)
rwhv->SetSynchronousCompositorClient(client);
}
WebContentsView* CreateWebContentsView(
WebContentsImpl* web_contents,
WebContentsViewDelegate* delegate,
RenderViewHostDelegateView** render_view_host_delegate_view) {
WebContentsViewAndroid* rv = new WebContentsViewAndroid(
web_contents, delegate);
*render_view_host_delegate_view = rv;
return rv;
}
WebContentsViewAndroid::WebContentsViewAndroid(
WebContentsImpl* web_contents,
WebContentsViewDelegate* delegate)
: web_contents_(web_contents),
delegate_(delegate),
view_(ui::ViewAndroid::LayoutType::NORMAL),
synchronous_compositor_client_(nullptr) {
view_.SetLayer(cc::Layer::Create());
view_.set_event_handler(this);
}
WebContentsViewAndroid::~WebContentsViewAndroid() {
if (view_.GetLayer())
view_.GetLayer()->RemoveFromParent();
view_.set_event_handler(nullptr);
}
void WebContentsViewAndroid::SetContentUiEventHandler(
std::unique_ptr<ContentUiEventHandler> handler) {
content_ui_event_handler_ = std::move(handler);
}
void WebContentsViewAndroid::SetOverscrollRefreshHandler(
std::unique_ptr<ui::OverscrollRefreshHandler> overscroll_refresh_handler) {
overscroll_refresh_handler_ = std::move(overscroll_refresh_handler);
auto* rwhv = web_contents_->GetRenderWidgetHostView();
if (rwhv) {
static_cast<RenderWidgetHostViewAndroid*>(rwhv)
->OnOverscrollRefreshHandlerAvailable();
}
if (web_contents_->ShowingInterstitialPage()) {
rwhv = web_contents_->GetInterstitialPage()
->GetMainFrame()
->GetRenderViewHost()
->GetWidget()
->GetView();
if (rwhv) {
static_cast<RenderWidgetHostViewAndroid*>(rwhv)
->OnOverscrollRefreshHandlerAvailable();
}
}
}
ui::OverscrollRefreshHandler*
WebContentsViewAndroid::GetOverscrollRefreshHandler() const {
return overscroll_refresh_handler_.get();
}
gfx::NativeView WebContentsViewAndroid::GetNativeView() const {
return const_cast<gfx::NativeView>(&view_);
}
gfx::NativeView WebContentsViewAndroid::GetContentNativeView() const {
RenderWidgetHostView* rwhv = web_contents_->GetRenderWidgetHostView();
if (rwhv)
return rwhv->GetNativeView();
// TODO(sievers): This should return null.
return GetNativeView();
}
RenderWidgetHostViewAndroid*
WebContentsViewAndroid::GetRenderWidgetHostViewAndroid() {
RenderWidgetHostView* rwhv = nullptr;
rwhv = web_contents_->GetRenderWidgetHostView();
if (web_contents_->ShowingInterstitialPage()) {
rwhv = web_contents_->GetInterstitialPage()
->GetMainFrame()
->GetRenderViewHost()
->GetWidget()
->GetView();
}
return static_cast<RenderWidgetHostViewAndroid*>(rwhv);
}
gfx::NativeWindow WebContentsViewAndroid::GetTopLevelNativeWindow() const {
return view_.GetWindowAndroid();
}
void WebContentsViewAndroid::GetContainerBounds(gfx::Rect* out) const {
*out = GetViewBounds();
}
void WebContentsViewAndroid::SetPageTitle(const base::string16& title) {
// Do nothing.
}
void WebContentsViewAndroid::SizeContents(const gfx::Size& size) {
// TODO(klobag): Do we need to do anything else?
RenderWidgetHostView* rwhv = web_contents_->GetRenderWidgetHostView();
if (rwhv)
rwhv->SetSize(size);
}
void WebContentsViewAndroid::Focus() {
if (web_contents_->ShowingInterstitialPage()) {
web_contents_->GetInterstitialPage()->Focus();
} else {
auto* rwhv = web_contents_->GetRenderWidgetHostView();
if (rwhv)
static_cast<RenderWidgetHostViewAndroid*>(rwhv)->Focus();
}
}
void WebContentsViewAndroid::SetInitialFocus() {
if (web_contents_->FocusLocationBarByDefault())
web_contents_->SetFocusToLocationBar();
else
Focus();
}
void WebContentsViewAndroid::StoreFocus() {
NOTIMPLEMENTED();
}
void WebContentsViewAndroid::RestoreFocus() {
NOTIMPLEMENTED();
}
void WebContentsViewAndroid::FocusThroughTabTraversal(bool reverse) {
if (web_contents_->ShowingInterstitialPage()) {
web_contents_->GetInterstitialPage()->FocusThroughTabTraversal(reverse);
return;
}
content::RenderWidgetHostView* fullscreen_view =
web_contents_->GetFullscreenRenderWidgetHostView();
if (fullscreen_view) {
fullscreen_view->Focus();
return;
}
web_contents_->GetRenderViewHost()->SetInitialFocus(reverse);
}
DropData* WebContentsViewAndroid::GetDropData() const {
NOTIMPLEMENTED();
return NULL;
}
gfx::Rect WebContentsViewAndroid::GetViewBounds() const {
return gfx::Rect(view_.GetSize());
}
void WebContentsViewAndroid::CreateView(
const gfx::Size& initial_size, gfx::NativeView context) {
}
RenderWidgetHostViewBase* WebContentsViewAndroid::CreateViewForWidget(
RenderWidgetHost* render_widget_host, bool is_guest_view_hack) {
if (render_widget_host->GetView()) {
// During testing, the view will already be set up in most cases to the
// test view, so we don't want to clobber it with a real one. To verify that
// this actually is happening (and somebody isn't accidentally creating the
// view twice), we check for the RVH Factory, which will be set when we're
// making special ones (which go along with the special views).
DCHECK(RenderViewHostFactory::has_factory());
return static_cast<RenderWidgetHostViewBase*>(
render_widget_host->GetView());
}
// Note that while this instructs the render widget host to reference
// |native_view_|, this has no effect without also instructing the
// native view (i.e. ContentView) how to obtain a reference to this widget in
// order to paint it. See ContentView::GetRenderWidgetHostViewAndroid for an
// example of how this is achieved for InterstitialPages.
RenderWidgetHostImpl* rwhi = RenderWidgetHostImpl::From(render_widget_host);
auto* rwhv = new RenderWidgetHostViewAndroid(rwhi, &view_);
rwhv->SetSynchronousCompositorClient(synchronous_compositor_client_);
return rwhv;
}
RenderWidgetHostViewBase* WebContentsViewAndroid::CreateViewForChildWidget(
RenderWidgetHost* render_widget_host) {
RenderWidgetHostImpl* rwhi = RenderWidgetHostImpl::From(render_widget_host);
return new RenderWidgetHostViewAndroid(rwhi, nullptr);
}
void WebContentsViewAndroid::RenderViewCreated(RenderViewHost* host) {
}
void WebContentsViewAndroid::RenderViewReady() {
if (device_orientation_ == 0)
return;
auto* rwhva = GetRenderWidgetHostViewAndroid();
if (rwhva)
rwhva->UpdateScreenInfo(GetNativeView());
web_contents_->OnScreenOrientationChange();
}
void WebContentsViewAndroid::RenderViewHostChanged(RenderViewHost* old_host,
RenderViewHost* new_host) {
if (old_host) {
auto* rwhv = old_host->GetWidget()->GetView();
if (rwhv && rwhv->GetNativeView()) {
static_cast<RenderWidgetHostViewAndroid*>(rwhv)->UpdateNativeViewTree(
nullptr);
}
}
auto* rwhv = new_host->GetWidget()->GetView();
if (rwhv && rwhv->GetNativeView()) {
static_cast<RenderWidgetHostViewAndroid*>(rwhv)->UpdateNativeViewTree(
GetNativeView());
SetFocus(view_.HasFocus());
}
}
void WebContentsViewAndroid::SetFocus(bool focused) {
auto* rwhva = GetRenderWidgetHostViewAndroid();
if (!rwhva)
return;
if (focused)
rwhva->GotFocus();
else
rwhva->LostFocus();
}
void WebContentsViewAndroid::SetOverscrollControllerEnabled(bool enabled) {
}
void WebContentsViewAndroid::ShowContextMenu(
RenderFrameHost* render_frame_host, const ContextMenuParams& params) {
auto* rwhv = static_cast<RenderWidgetHostViewAndroid*>(
web_contents_->GetRenderWidgetHostView());
// See if context menu is handled by SelectionController as a selection menu.
// If not, use the delegate to show it.
if (rwhv && rwhv->ShowSelectionMenu(params))
return;
if (delegate_)
delegate_->ShowContextMenu(render_frame_host, params);
}
SelectPopup* WebContentsViewAndroid::GetSelectPopup() {
if (!select_popup_)
select_popup_ = std::make_unique<SelectPopup>(web_contents_);
return select_popup_.get();
}
void WebContentsViewAndroid::ShowPopupMenu(
RenderFrameHost* render_frame_host,
const gfx::Rect& bounds,
int item_height,
double item_font_size,
int selected_item,
const std::vector<MenuItem>& items,
bool right_aligned,
bool allow_multiple_selection) {
GetSelectPopup()->ShowMenu(render_frame_host, bounds, items, selected_item,
allow_multiple_selection, right_aligned);
}
void WebContentsViewAndroid::HidePopupMenu() {
GetSelectPopup()->HideMenu();
}
void WebContentsViewAndroid::StartDragging(
const DropData& drop_data,
blink::WebDragOperationsMask allowed_ops,
const gfx::ImageSkia& image,
const gfx::Vector2d& image_offset,
const DragEventSourceInfo& event_info,
RenderWidgetHostImpl* source_rwh) {
if (drop_data.text.is_null()) {
// Need to clear drag and drop state in blink.
OnSystemDragEnded();
return;
}
gfx::NativeView native_view = GetNativeView();
if (!native_view) {
// Need to clear drag and drop state in blink.
OnSystemDragEnded();
return;
}
const SkBitmap* bitmap = image.bitmap();
SkBitmap dummy_bitmap;
if (image.size().IsEmpty()) {
// An empty drag image is possible if the Javascript sets an empty drag
// image on purpose.
// Create a dummy 1x1 pixel image to avoid crashes when converting to java
// bitmap.
dummy_bitmap.allocN32Pixels(1, 1);
dummy_bitmap.eraseColor(0);
bitmap = &dummy_bitmap;
}
JNIEnv* env = AttachCurrentThread();
ScopedJavaLocalRef<jstring> jtext =
ConvertUTF16ToJavaString(env, drop_data.text.string());
if (!native_view->StartDragAndDrop(jtext, gfx::ConvertToJavaBitmap(bitmap))) {
// Need to clear drag and drop state in blink.
OnSystemDragEnded();
return;
}
if (selection_popup_controller_) {
selection_popup_controller_->HidePopupsAndPreserveSelection();
// Hide the handles temporarily.
auto* rwhva = GetRenderWidgetHostViewAndroid();
if (rwhva)
rwhva->SetTextHandlesTemporarilyHidden(true);
}
}
void WebContentsViewAndroid::UpdateDragCursor(blink::WebDragOperation op) {
// Intentional no-op because Android does not have cursor.
}
bool WebContentsViewAndroid::OnDragEvent(const ui::DragEventAndroid& event) {
switch (event.action()) {
case JNI_DragEvent::ACTION_DRAG_ENTERED: {
std::vector<DropData::Metadata> metadata;
for (const base::string16& mime_type : event.mime_types()) {
metadata.push_back(DropData::Metadata::CreateForMimeType(
DropData::Kind::STRING, mime_type));
}
OnDragEntered(metadata, event.location_f(), event.screen_location_f());
break;
}
case JNI_DragEvent::ACTION_DRAG_LOCATION:
OnDragUpdated(event.location_f(), event.screen_location_f());
break;
case JNI_DragEvent::ACTION_DROP: {
DropData drop_data;
drop_data.did_originate_from_renderer = false;
JNIEnv* env = AttachCurrentThread();
base::string16 drop_content =
ConvertJavaStringToUTF16(env, event.GetJavaContent());
for (const base::string16& mime_type : event.mime_types()) {
if (base::EqualsASCII(mime_type, ui::kMimeTypeURIList)) {
drop_data.url = GURL(drop_content);
} else if (base::EqualsASCII(mime_type, ui::kMimeTypeText)) {
drop_data.text = base::NullableString16(drop_content, false);
} else {
drop_data.html = base::NullableString16(drop_content, false);
}
}
OnPerformDrop(&drop_data, event.location_f(), event.screen_location_f());
break;
}
case JNI_DragEvent::ACTION_DRAG_EXITED:
OnDragExited();
break;
case JNI_DragEvent::ACTION_DRAG_ENDED:
OnDragEnded();
break;
case JNI_DragEvent::ACTION_DRAG_STARTED:
// Nothing meaningful to do.
break;
}
return true;
}
// TODO(paulmeyer): The drag-and-drop calls on GetRenderViewHost()->GetWidget()
// in the following functions will need to be targeted to specific
// RenderWidgetHosts in order to work with OOPIFs. See crbug.com/647249.
void WebContentsViewAndroid::OnDragEntered(
const std::vector<DropData::Metadata>& metadata,
const gfx::PointF& location,
const gfx::PointF& screen_location) {
blink::WebDragOperationsMask allowed_ops =
static_cast<blink::WebDragOperationsMask>(blink::kWebDragOperationCopy |
blink::kWebDragOperationMove);
web_contents_->GetRenderViewHost()->GetWidget()->
DragTargetDragEnterWithMetaData(metadata, location, screen_location,
allowed_ops, 0);
}
void WebContentsViewAndroid::OnDragUpdated(const gfx::PointF& location,
const gfx::PointF& screen_location) {
drag_location_ = location;
drag_screen_location_ = screen_location;
blink::WebDragOperationsMask allowed_ops =
static_cast<blink::WebDragOperationsMask>(blink::kWebDragOperationCopy |
blink::kWebDragOperationMove);
web_contents_->GetRenderViewHost()->GetWidget()->DragTargetDragOver(
location, screen_location, allowed_ops, 0);
}
void WebContentsViewAndroid::OnDragExited() {
web_contents_->GetRenderViewHost()->GetWidget()->DragTargetDragLeave(
gfx::PointF(), gfx::PointF());
}
void WebContentsViewAndroid::OnPerformDrop(DropData* drop_data,
const gfx::PointF& location,
const gfx::PointF& screen_location) {
web_contents_->GetRenderViewHost()->GetWidget()->FilterDropData(drop_data);
web_contents_->GetRenderViewHost()->GetWidget()->DragTargetDrop(
*drop_data, location, screen_location, 0);
}
void WebContentsViewAndroid::OnSystemDragEnded() {
web_contents_->GetRenderViewHost()->GetWidget()->DragSourceSystemDragEnded();
// Restore the selection popups and the text handles if necessary.
if (selection_popup_controller_) {
selection_popup_controller_->RestoreSelectionPopupsIfNecessary();
auto* rwhva = GetRenderWidgetHostViewAndroid();
if (rwhva)
rwhva->SetTextHandlesTemporarilyHidden(false);
}
}
void WebContentsViewAndroid::OnDragEnded() {
web_contents_->GetRenderViewHost()->GetWidget()->DragSourceEndedAt(
drag_location_, drag_screen_location_, blink::kWebDragOperationNone);
OnSystemDragEnded();
drag_location_ = gfx::PointF();
drag_screen_location_ = gfx::PointF();
}
void WebContentsViewAndroid::GotFocus(
RenderWidgetHostImpl* render_widget_host) {
web_contents_->NotifyWebContentsFocused(render_widget_host);
}
void WebContentsViewAndroid::LostFocus(
RenderWidgetHostImpl* render_widget_host) {
web_contents_->NotifyWebContentsLostFocus(render_widget_host);
}
// This is called when we the renderer asks us to take focus back (i.e., it has
// iterated past the last focusable element on the page).
void WebContentsViewAndroid::TakeFocus(bool reverse) {
if (web_contents_->GetDelegate() &&
web_contents_->GetDelegate()->TakeFocus(web_contents_, reverse))
return;
web_contents_->GetRenderWidgetHostView()->Focus();
}
int WebContentsViewAndroid::GetTopControlsHeight() const {
auto* delegate = web_contents_->GetDelegate();
return delegate ? delegate->GetTopControlsHeight() : 0;
}
int WebContentsViewAndroid::GetBottomControlsHeight() const {
auto* delegate = web_contents_->GetDelegate();
return delegate ? delegate->GetBottomControlsHeight() : 0;
}
bool WebContentsViewAndroid::DoBrowserControlsShrinkRendererSize() const {
auto* delegate = web_contents_->GetDelegate();
return delegate &&
delegate->DoBrowserControlsShrinkRendererSize(web_contents_);
}
bool WebContentsViewAndroid::OnTouchEvent(const ui::MotionEventAndroid& event) {
if (event.GetAction() == ui::MotionEventAndroid::Action::DOWN &&
ShouldRequestUnbufferedDispatch()) {
view_.RequestUnbufferedDispatch(event);
}
return false; // let the children handle the actual event.
}
bool WebContentsViewAndroid::OnMouseEvent(const ui::MotionEventAndroid& event) {
// Hover events can be intercepted when in accessibility mode.
auto action = event.GetAction();
if (action != ui::MotionEventAndroid::Action::HOVER_ENTER &&
action != ui::MotionEventAndroid::Action::HOVER_EXIT &&
action != ui::MotionEventAndroid::Action::HOVER_MOVE)
return false;
auto* manager = static_cast<BrowserAccessibilityManagerAndroid*>(
web_contents_->GetRootBrowserAccessibilityManager());
return manager && manager->OnHoverEvent(event);
}
bool WebContentsViewAndroid::OnGenericMotionEvent(
const ui::MotionEventAndroid& event) {
if (content_ui_event_handler_)
return content_ui_event_handler_->OnGenericMotionEvent(event);
return false;
}
bool WebContentsViewAndroid::OnKeyUp(const ui::KeyEventAndroid& event) {
if (content_ui_event_handler_)
return content_ui_event_handler_->OnKeyUp(event);
return false;
}
bool WebContentsViewAndroid::DispatchKeyEvent(
const ui::KeyEventAndroid& event) {
if (content_ui_event_handler_)
return content_ui_event_handler_->DispatchKeyEvent(event);
return false;
}
bool WebContentsViewAndroid::ScrollBy(float delta_x, float delta_y) {
if (content_ui_event_handler_)
content_ui_event_handler_->ScrollBy(delta_x, delta_y);
return false;
}
bool WebContentsViewAndroid::ScrollTo(float x, float y) {
if (content_ui_event_handler_)
content_ui_event_handler_->ScrollTo(x, y);
return false;
}
void WebContentsViewAndroid::OnSizeChanged() {
auto* rwhv = GetRenderWidgetHostViewAndroid();
if (rwhv) {
web_contents_->SendScreenRects();
rwhv->SynchronizeVisualProperties(cc::DeadlinePolicy::UseDefaultDeadline(),
base::nullopt);
}
}
void WebContentsViewAndroid::OnPhysicalBackingSizeChanged() {
if (web_contents_->GetRenderWidgetHostView())
web_contents_->SendScreenRects();
}
} // namespace content