blob: bf405d07c23cf3f618db3c17bf47f908a8864178 [file] [log] [blame]
// Copyright 2015 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/devtools/protocol/emulation_handler.h"
#include <utility>
#include "base/numerics/safe_conversions.h"
#include "base/strings/string_number_conversions.h"
#include "base/strings/string_util.h"
#include "build/build_config.h"
#include "content/browser/devtools/devtools_agent_host_impl.h"
#include "content/browser/frame_host/render_frame_host_impl.h"
#include "content/browser/renderer_host/input/touch_emulator.h"
#include "content/browser/renderer_host/render_widget_host_impl.h"
#include "content/browser/web_contents/web_contents_impl.h"
#include "content/common/widget_messages.h"
#include "content/public/common/url_constants.h"
#include "net/http/http_util.h"
#include "services/device/public/cpp/geolocation/geoposition.h"
#include "services/device/public/mojom/geolocation_context.mojom.h"
#include "services/device/public/mojom/geoposition.mojom.h"
#include "third_party/blink/public/mojom/widget/screen_orientation.mojom.h"
#include "ui/events/gesture_detection/gesture_provider_config_helper.h"
namespace content {
namespace protocol {
namespace {
blink::mojom::ScreenOrientation WebScreenOrientationTypeFromString(
const std::string& type) {
if (type == Emulation::ScreenOrientation::TypeEnum::PortraitPrimary)
return blink::mojom::ScreenOrientation::kPortraitPrimary;
if (type == Emulation::ScreenOrientation::TypeEnum::PortraitSecondary)
return blink::mojom::ScreenOrientation::kPortraitSecondary;
if (type == Emulation::ScreenOrientation::TypeEnum::LandscapePrimary)
return blink::mojom::ScreenOrientation::kLandscapePrimary;
if (type == Emulation::ScreenOrientation::TypeEnum::LandscapeSecondary)
return blink::mojom::ScreenOrientation::kLandscapeSecondary;
return blink::mojom::ScreenOrientation::kUndefined;
}
base::Optional<content::DisplayFeature::Orientation>
DisplayFeatureOrientationTypeFromString(const std::string& type) {
if (type == Emulation::DisplayFeature::OrientationEnum::Vertical)
return content::DisplayFeature::Orientation::kVertical;
if (type == Emulation::DisplayFeature::OrientationEnum::Horizontal)
return content::DisplayFeature::Orientation::kHorizontal;
return base::nullopt;
}
ui::GestureProviderConfigType TouchEmulationConfigurationToType(
const std::string& protocol_value) {
ui::GestureProviderConfigType result =
ui::GestureProviderConfigType::CURRENT_PLATFORM;
if (protocol_value ==
Emulation::SetEmitTouchEventsForMouse::ConfigurationEnum::Mobile) {
result = ui::GestureProviderConfigType::GENERIC_MOBILE;
}
if (protocol_value ==
Emulation::SetEmitTouchEventsForMouse::ConfigurationEnum::Desktop) {
result = ui::GestureProviderConfigType::GENERIC_DESKTOP;
}
return result;
}
bool ValidateClientHintString(const std::string& s) {
// Matches definition in structured headers:
// https://tools.ietf.org/html/draft-ietf-httpbis-header-structure-17#section-3.3.3
for (char c : s) {
if (!base::IsAsciiPrintable(c))
return false;
}
return true;
}
} // namespace
EmulationHandler::EmulationHandler()
: DevToolsDomainHandler(Emulation::Metainfo::domainName),
touch_emulation_enabled_(false),
device_emulation_enabled_(false),
focus_emulation_enabled_(false),
host_(nullptr) {}
EmulationHandler::~EmulationHandler() = default;
// static
std::vector<EmulationHandler*> EmulationHandler::ForAgentHost(
DevToolsAgentHostImpl* host) {
return host->HandlersByName<EmulationHandler>(
Emulation::Metainfo::domainName);
}
void EmulationHandler::SetRenderer(int process_host_id,
RenderFrameHostImpl* frame_host) {
if (host_ == frame_host)
return;
host_ = frame_host;
if (touch_emulation_enabled_)
UpdateTouchEventEmulationState();
if (device_emulation_enabled_)
UpdateDeviceEmulationState();
}
void EmulationHandler::Wire(UberDispatcher* dispatcher) {
Emulation::Dispatcher::wire(dispatcher, this);
}
Response EmulationHandler::Disable() {
if (touch_emulation_enabled_) {
touch_emulation_enabled_ = false;
UpdateTouchEventEmulationState();
}
user_agent_ = std::string();
if (device_emulation_enabled_) {
device_emulation_enabled_ = false;
UpdateDeviceEmulationState();
}
if (focus_emulation_enabled_)
SetFocusEmulationEnabled(false);
return Response::Success();
}
Response EmulationHandler::SetGeolocationOverride(
Maybe<double> latitude, Maybe<double> longitude, Maybe<double> accuracy) {
if (!host_)
return Response::InternalError();
auto* geolocation_context = GetWebContents()->GetGeolocationContext();
auto geoposition = device::mojom::Geoposition::New();
if (latitude.isJust() && longitude.isJust() && accuracy.isJust()) {
geoposition->latitude = latitude.fromJust();
geoposition->longitude = longitude.fromJust();
geoposition->accuracy = accuracy.fromJust();
geoposition->timestamp = base::Time::Now();
if (!device::ValidateGeoposition(*geoposition))
return Response::ServerError("Invalid geolocation");
} else {
geoposition->error_code =
device::mojom::Geoposition::ErrorCode::POSITION_UNAVAILABLE;
}
geolocation_context->SetOverride(std::move(geoposition));
return Response::Success();
}
Response EmulationHandler::ClearGeolocationOverride() {
if (!host_)
return Response::InternalError();
auto* geolocation_context = GetWebContents()->GetGeolocationContext();
geolocation_context->ClearOverride();
return Response::Success();
}
Response EmulationHandler::SetEmitTouchEventsForMouse(
bool enabled,
Maybe<std::string> configuration) {
touch_emulation_enabled_ = enabled;
touch_emulation_configuration_ = configuration.fromMaybe("");
UpdateTouchEventEmulationState();
return Response::Success();
}
Response EmulationHandler::CanEmulate(bool* result) {
#if defined(OS_ANDROID)
*result = false;
#else
*result = true;
if (host_) {
if (GetWebContents()->GetVisibleURL().SchemeIs(kChromeDevToolsScheme) ||
host_->GetRenderWidgetHost()->auto_resize_enabled())
*result = false;
}
#endif // defined(OS_ANDROID)
return Response::Success();
}
Response EmulationHandler::SetDeviceMetricsOverride(
int width,
int height,
double device_scale_factor,
bool mobile,
Maybe<double> scale,
Maybe<int> screen_width,
Maybe<int> screen_height,
Maybe<int> position_x,
Maybe<int> position_y,
Maybe<bool> dont_set_visible_size,
Maybe<Emulation::ScreenOrientation> screen_orientation,
Maybe<protocol::Page::Viewport> viewport,
Maybe<protocol::Emulation::DisplayFeature> displayFeature) {
const static int max_size = 10000000;
const static double max_scale = 10;
const static int max_orientation_angle = 360;
if (!host_)
return Response::ServerError("Target does not support metrics override");
if (screen_width.fromMaybe(0) < 0 || screen_height.fromMaybe(0) < 0 ||
screen_width.fromMaybe(0) > max_size ||
screen_height.fromMaybe(0) > max_size) {
return Response::InvalidParams(
"Screen width and height values must be positive, not greater than " +
base::NumberToString(max_size));
}
if (position_x.fromMaybe(0) < 0 || position_y.fromMaybe(0) < 0 ||
position_x.fromMaybe(0) > screen_width.fromMaybe(0) ||
position_y.fromMaybe(0) > screen_height.fromMaybe(0)) {
return Response::InvalidParams("View position should be on the screen");
}
if (width < 0 || height < 0 || width > max_size || height > max_size) {
return Response::InvalidParams(
"Width and height values must be positive, not greater than " +
base::NumberToString(max_size));
}
if (device_scale_factor < 0)
return Response::InvalidParams("deviceScaleFactor must be non-negative");
if (scale.fromMaybe(1) <= 0 || scale.fromMaybe(1) > max_scale) {
return Response::InvalidParams("scale must be positive, not greater than " +
base::NumberToString(max_scale));
}
blink::mojom::ScreenOrientation orientationType =
blink::mojom::ScreenOrientation::kUndefined;
int orientationAngle = 0;
if (screen_orientation.isJust()) {
Emulation::ScreenOrientation* orientation = screen_orientation.fromJust();
orientationType = WebScreenOrientationTypeFromString(
orientation->GetType());
if (orientationType == blink::mojom::ScreenOrientation::kUndefined)
return Response::InvalidParams("Invalid screen orientation type value");
orientationAngle = orientation->GetAngle();
if (orientationAngle < 0 || orientationAngle >= max_orientation_angle) {
return Response::InvalidParams(
"Screen orientation angle must be non-negative, less than " +
base::NumberToString(max_orientation_angle));
}
}
base::Optional<content::DisplayFeature> display_feature = base::nullopt;
if (displayFeature.isJust()) {
protocol::Emulation::DisplayFeature* emu_display_feature =
displayFeature.fromJust();
base::Optional<content::DisplayFeature::Orientation> disp_orientation =
DisplayFeatureOrientationTypeFromString(
emu_display_feature->GetOrientation());
if (!disp_orientation) {
return Response::InvalidParams(
"Invalid display feature orientation type");
}
content::DisplayFeature::ParamErrorEnum error;
display_feature = content::DisplayFeature::Create(
*disp_orientation, emu_display_feature->GetOffset(),
emu_display_feature->GetMaskLength(), width, height, &error);
if (!display_feature) {
switch (error) {
case content::DisplayFeature::ParamErrorEnum::
kDisplayFeatureWithZeroScreenSize:
return Response::InvalidParams(
"Cannot specify a display feature with zero width and height");
case content::DisplayFeature::ParamErrorEnum::
kNegativeDisplayFeatureParams:
return Response::InvalidParams("Negative display feature parameters");
case content::DisplayFeature::ParamErrorEnum::kOutsideScreenWidth:
return Response::InvalidParams(
"Display feature window segments outside screen width");
case content::DisplayFeature::ParamErrorEnum::kOutsideScreenHeight:
return Response::InvalidParams(
"Display feature window segments outside screen height");
}
}
}
blink::WebDeviceEmulationParams params;
params.screen_position = mobile ? blink::WebDeviceEmulationParams::kMobile
: blink::WebDeviceEmulationParams::kDesktop;
params.screen_size =
blink::WebSize(screen_width.fromMaybe(0), screen_height.fromMaybe(0));
if (position_x.isJust() && position_y.isJust()) {
params.view_position =
gfx::Point(position_x.fromMaybe(0), position_y.fromMaybe(0));
}
params.device_scale_factor = device_scale_factor;
params.view_size = blink::WebSize(width, height);
params.scale = scale.fromMaybe(1);
params.screen_orientation_type = orientationType;
params.screen_orientation_angle = orientationAngle;
if (display_feature) {
params.window_segments =
display_feature->ComputeWindowSegments(params.view_size);
}
if (viewport.isJust()) {
params.viewport_offset.SetPoint(viewport.fromJust()->GetX(),
viewport.fromJust()->GetY());
double dpfactor =
device_scale_factor
? device_scale_factor /
host_->GetRenderWidgetHost()->GetDeviceScaleFactor()
: 1;
params.viewport_scale = viewport.fromJust()->GetScale() * dpfactor;
// Resize the RenderWidgetHostView to the size of the overridden viewport.
width =
base::Round(viewport.fromJust()->GetWidth() * params.viewport_scale);
height =
base::Round(viewport.fromJust()->GetHeight() * params.viewport_scale);
}
bool size_changed = false;
if (!dont_set_visible_size.fromMaybe(false) && width > 0 && height > 0) {
if (GetWebContents()) {
size_changed =
GetWebContents()->SetDeviceEmulationSize(gfx::Size(width, height));
} else {
return Response::ServerError("Can't find the associated web contents");
}
}
if (device_emulation_enabled_ && params == device_emulation_params_) {
// Renderer should answer after size was changed, so that the response is
// only sent to the client once updates were applied.
if (size_changed)
return Response::FallThrough();
return Response::Success();
}
device_emulation_enabled_ = true;
device_emulation_params_ = params;
UpdateDeviceEmulationState();
// Renderer should answer after emulation params were updated, so that the
// response is only sent to the client once updates were applied.
// Unless the renderer has crashed.
if (GetWebContents() && GetWebContents()->IsCrashed())
return Response::Success();
return Response::FallThrough();
}
Response EmulationHandler::ClearDeviceMetricsOverride() {
if (!device_emulation_enabled_)
return Response::Success();
if (!host_)
return Response::ServerError("Can't find the associated web contents");
GetWebContents()->ClearDeviceEmulationSize();
device_emulation_enabled_ = false;
device_emulation_params_ = blink::WebDeviceEmulationParams();
UpdateDeviceEmulationState();
// Renderer should answer after emulation was disabled, so that the response
// is only sent to the client once updates were applied.
// Unless the renderer has crashed.
if (GetWebContents()->IsCrashed())
return Response::Success();
return Response::FallThrough();
}
Response EmulationHandler::SetVisibleSize(int width, int height) {
if (width < 0 || height < 0)
return Response::InvalidParams("Width and height must be non-negative");
if (!host_)
return Response::ServerError("Can't find the associated web contents");
GetWebContents()->SetDeviceEmulationSize(gfx::Size(width, height));
return Response::Success();
}
Response EmulationHandler::SetUserAgentOverride(
const std::string& user_agent,
Maybe<std::string> accept_language,
Maybe<std::string> platform,
Maybe<Emulation::UserAgentMetadata> ua_metadata_override) {
if (!user_agent.empty() && !net::HttpUtil::IsValidHeaderValue(user_agent))
return Response::InvalidParams("Invalid characters found in userAgent");
std::string accept_lang = accept_language.fromMaybe(std::string());
if (!accept_lang.empty() && !net::HttpUtil::IsValidHeaderValue(accept_lang)) {
return Response::InvalidParams(
"Invalid characters found in acceptLanguage");
}
user_agent_ = user_agent;
accept_language_ = accept_lang;
user_agent_metadata_ = base::nullopt;
if (!ua_metadata_override.isJust())
return Response::FallThrough();
if (user_agent.empty()) {
return Response::InvalidParams(
"Empty userAgent invalid with userAgentMetadata provided");
}
std::unique_ptr<Emulation::UserAgentMetadata> ua_metadata =
ua_metadata_override.takeJust();
blink::UserAgentMetadata new_ua_metadata;
DCHECK(ua_metadata->GetBrands());
for (const auto& bv : *ua_metadata->GetBrands()) {
blink::UserAgentBrandVersion out_bv;
if (!ValidateClientHintString(bv->GetBrand()))
return Response::InvalidParams("Invalid brand string");
out_bv.brand = bv->GetBrand();
if (!ValidateClientHintString(bv->GetVersion()))
return Response::InvalidParams("Invalid brand version string");
out_bv.major_version = bv->GetVersion();
new_ua_metadata.brand_version_list.push_back(std::move(out_bv));
}
if (!ValidateClientHintString(ua_metadata->GetFullVersion()))
return Response::InvalidParams("Invalid full version string");
new_ua_metadata.full_version = ua_metadata->GetFullVersion();
if (!ValidateClientHintString(ua_metadata->GetPlatform()))
return Response::InvalidParams("Invalid platform string");
new_ua_metadata.platform = ua_metadata->GetPlatform();
if (!ValidateClientHintString(ua_metadata->GetPlatformVersion()))
return Response::InvalidParams("Invalid platform version string");
new_ua_metadata.platform_version = ua_metadata->GetPlatformVersion();
if (!ValidateClientHintString(ua_metadata->GetArchitecture()))
return Response::InvalidParams("Invalid architecture string");
new_ua_metadata.architecture = ua_metadata->GetArchitecture();
if (!ValidateClientHintString(ua_metadata->GetModel()))
return Response::InvalidParams("Invalid model string");
new_ua_metadata.model = ua_metadata->GetModel();
new_ua_metadata.mobile = ua_metadata->GetMobile();
// All checks OK, can update user_agent_metadata_.
user_agent_metadata_.emplace(std::move(new_ua_metadata));
return Response::FallThrough();
}
Response EmulationHandler::SetFocusEmulationEnabled(bool enabled) {
if (enabled == focus_emulation_enabled_)
return Response::FallThrough();
focus_emulation_enabled_ = enabled;
if (enabled) {
GetWebContents()->IncrementCapturerCount(gfx::Size(),
/* stay_hidden */ false);
} else {
GetWebContents()->DecrementCapturerCount(/* stay_hidden */ false);
}
return Response::FallThrough();
}
blink::WebDeviceEmulationParams EmulationHandler::GetDeviceEmulationParams() {
return device_emulation_params_;
}
void EmulationHandler::SetDeviceEmulationParams(
const blink::WebDeviceEmulationParams& params) {
bool enabled = params != blink::WebDeviceEmulationParams();
bool enable_changed = enabled != device_emulation_enabled_;
bool params_changed = params != device_emulation_params_;
if (!device_emulation_enabled_ && !enable_changed)
return; // Still disabled.
if (!enable_changed && !params_changed)
return; // Nothing changed.
device_emulation_enabled_ = enabled;
device_emulation_params_ = params;
UpdateDeviceEmulationState();
}
WebContentsImpl* EmulationHandler::GetWebContents() {
DCHECK(host_); // Only call if |host_| is set.
return static_cast<WebContentsImpl*>(WebContents::FromRenderFrameHost(host_));
}
void EmulationHandler::UpdateTouchEventEmulationState() {
if (!host_)
return;
// We only have a single TouchEmulator for all frames, so let the main frame's
// EmulationHandler enable/disable it.
if (!host_->frame_tree_node()->IsMainFrame())
return;
if (touch_emulation_enabled_) {
if (auto* touch_emulator =
host_->GetRenderWidgetHost()->GetTouchEmulator()) {
touch_emulator->Enable(
TouchEmulator::Mode::kEmulatingTouchFromMouse,
TouchEmulationConfigurationToType(touch_emulation_configuration_));
}
} else {
if (auto* touch_emulator = host_->GetRenderWidgetHost()->GetTouchEmulator())
touch_emulator->Disable();
}
GetWebContents()->SetForceDisableOverscrollContent(touch_emulation_enabled_);
}
void EmulationHandler::UpdateDeviceEmulationState() {
if (!host_)
return;
// Device emulation only happens on the main frame.
if (!host_->frame_tree_node()->IsMainFrame())
return;
// TODO(eseckler): Once we change this to mojo, we should wait for an ack to
// these messages from the renderer. The renderer should send the ack once the
// emulation params were applied. That way, we can avoid having to handle
// Set/ClearDeviceMetricsOverride in the renderer. With the old IPC system,
// this is tricky since we'd have to track the DevTools message id with the
// WidgetMsg and acknowledgment, as well as plump the acknowledgment back to
// the EmulationHandler somehow. Mojo callbacks should make this much simpler.
UpdateDeviceEmulationStateForHost(host_->GetRenderWidgetHost());
// Update portals inside this page.
for (auto* web_contents : GetWebContents()->GetWebContentsAndAllInner()) {
if (web_contents->IsPortal()) {
UpdateDeviceEmulationStateForHost(
web_contents->GetMainFrame()->GetRenderWidgetHost());
}
}
}
void EmulationHandler::UpdateDeviceEmulationStateForHost(
RenderWidgetHostImpl* render_widget_host) {
if (device_emulation_enabled_) {
render_widget_host->Send(new WidgetMsg_EnableDeviceEmulation(
render_widget_host->GetRoutingID(), device_emulation_params_));
} else {
render_widget_host->Send(new WidgetMsg_DisableDeviceEmulation(
render_widget_host->GetRoutingID()));
}
}
void EmulationHandler::ApplyOverrides(net::HttpRequestHeaders* headers) {
if (!user_agent_.empty())
headers->SetHeader(net::HttpRequestHeaders::kUserAgent, user_agent_);
if (!accept_language_.empty()) {
headers->SetHeader(
net::HttpRequestHeaders::kAcceptLanguage,
net::HttpUtil::GenerateAcceptLanguageHeader(accept_language_));
}
}
bool EmulationHandler::ApplyUserAgentMetadataOverrides(
base::Optional<blink::UserAgentMetadata>* override_out) {
// This is conditional on basic user agent override being on; this helps us
// emulate a device not sending any UA client hints.
if (user_agent_.empty())
return false;
*override_out = user_agent_metadata_;
return true;
}
} // namespace protocol
} // namespace content