// Copyright (c) 2015 GitHub, Inc. // Use of this source code is governed by the MIT license that can be // found in the LICENSE file. #include "shell/common/gin_converters/blink_converter.h" #include #include #include #include #include "base/strings/string_util.h" #include "base/strings/utf_string_conversions.h" #include "gin/converter.h" #include "mojo/public/cpp/base/values_mojom_traits.h" #include "mojo/public/mojom/base/values.mojom.h" #include "shell/common/deprecate_util.h" #include "shell/common/gin_converters/value_converter.h" #include "shell/common/gin_helper/dictionary.h" #include "shell/common/keyboard_util.h" #include "third_party/blink/public/platform/web_input_event.h" #include "third_party/blink/public/platform/web_keyboard_event.h" #include "third_party/blink/public/platform/web_mouse_event.h" #include "third_party/blink/public/platform/web_mouse_wheel_event.h" #include "third_party/blink/public/web/web_device_emulation_params.h" #include "ui/base/clipboard/clipboard.h" #include "ui/events/keycodes/dom/keycode_converter.h" #include "ui/events/keycodes/keyboard_code_conversion.h" namespace { template int VectorToBitArray(const std::vector& vec) { int bits = 0; for (const T& item : vec) bits |= item; return bits; } } // namespace namespace gin { template <> struct Converter { static bool FromV8(v8::Isolate* isolate, v8::Handle val, base::char16* out) { base::string16 code = base::UTF8ToUTF16(gin::V8ToString(isolate, val)); if (code.length() != 1) return false; *out = code[0]; return true; } }; template <> struct Converter { static bool FromV8(v8::Isolate* isolate, v8::Handle val, blink::WebInputEvent::Type* out) { std::string type = base::ToLowerASCII(gin::V8ToString(isolate, val)); if (type == "mousedown") *out = blink::WebInputEvent::kMouseDown; else if (type == "mouseup") *out = blink::WebInputEvent::kMouseUp; else if (type == "mousemove") *out = blink::WebInputEvent::kMouseMove; else if (type == "mouseenter") *out = blink::WebInputEvent::kMouseEnter; else if (type == "mouseleave") *out = blink::WebInputEvent::kMouseLeave; else if (type == "contextmenu") *out = blink::WebInputEvent::kContextMenu; else if (type == "mousewheel") *out = blink::WebInputEvent::kMouseWheel; else if (type == "keydown") *out = blink::WebInputEvent::kRawKeyDown; else if (type == "keyup") *out = blink::WebInputEvent::kKeyUp; else if (type == "char") *out = blink::WebInputEvent::kChar; else if (type == "touchstart") *out = blink::WebInputEvent::kTouchStart; else if (type == "touchmove") *out = blink::WebInputEvent::kTouchMove; else if (type == "touchend") *out = blink::WebInputEvent::kTouchEnd; else if (type == "touchcancel") *out = blink::WebInputEvent::kTouchCancel; return true; } }; template <> struct Converter { static bool FromV8(v8::Isolate* isolate, v8::Handle val, blink::WebMouseEvent::Button* out) { std::string button = base::ToLowerASCII(gin::V8ToString(isolate, val)); if (button == "left") *out = blink::WebMouseEvent::Button::kLeft; else if (button == "middle") *out = blink::WebMouseEvent::Button::kMiddle; else if (button == "right") *out = blink::WebMouseEvent::Button::kRight; else return false; return true; } }; template <> struct Converter { static bool FromV8(v8::Isolate* isolate, v8::Handle val, blink::WebInputEvent::Modifiers* out) { std::string modifier = base::ToLowerASCII(gin::V8ToString(isolate, val)); if (modifier == "shift") *out = blink::WebInputEvent::kShiftKey; else if (modifier == "control" || modifier == "ctrl") *out = blink::WebInputEvent::kControlKey; else if (modifier == "alt") *out = blink::WebInputEvent::kAltKey; else if (modifier == "meta" || modifier == "command" || modifier == "cmd") *out = blink::WebInputEvent::kMetaKey; else if (modifier == "iskeypad") *out = blink::WebInputEvent::kIsKeyPad; else if (modifier == "isautorepeat") *out = blink::WebInputEvent::kIsAutoRepeat; else if (modifier == "leftbuttondown") *out = blink::WebInputEvent::kLeftButtonDown; else if (modifier == "middlebuttondown") *out = blink::WebInputEvent::kMiddleButtonDown; else if (modifier == "rightbuttondown") *out = blink::WebInputEvent::kRightButtonDown; else if (modifier == "capslock") *out = blink::WebInputEvent::kCapsLockOn; else if (modifier == "numlock") *out = blink::WebInputEvent::kNumLockOn; else if (modifier == "left") *out = blink::WebInputEvent::kIsLeft; else if (modifier == "right") *out = blink::WebInputEvent::kIsRight; return true; } }; blink::WebInputEvent::Type GetWebInputEventType(v8::Isolate* isolate, v8::Local val) { blink::WebInputEvent::Type type = blink::WebInputEvent::kUndefined; gin_helper::Dictionary dict; ConvertFromV8(isolate, val, &dict) && dict.Get("type", &type); return type; } bool Converter::FromV8(v8::Isolate* isolate, v8::Local val, blink::WebInputEvent* out) { gin_helper::Dictionary dict; if (!ConvertFromV8(isolate, val, &dict)) return false; blink::WebInputEvent::Type type; if (!dict.Get("type", &type)) return false; out->SetType(type); std::vector modifiers; if (dict.Get("modifiers", &modifiers)) out->SetModifiers(VectorToBitArray(modifiers)); out->SetTimeStamp(base::TimeTicks::Now()); return true; } bool Converter::FromV8(v8::Isolate* isolate, v8::Local val, blink::WebKeyboardEvent* out) { gin_helper::Dictionary dict; if (!ConvertFromV8(isolate, val, &dict)) return false; if (!ConvertFromV8(isolate, val, static_cast(out))) return false; std::string str; if (!dict.Get("keyCode", &str)) return false; bool shifted = false; ui::KeyboardCode keyCode = electron::KeyboardCodeFromStr(str, &shifted); out->windows_key_code = keyCode; if (shifted) out->SetModifiers(out->GetModifiers() | blink::WebInputEvent::kShiftKey); ui::DomCode domCode = ui::UsLayoutKeyboardCodeToDomCode(keyCode); out->dom_code = static_cast(domCode); ui::DomKey domKey; ui::KeyboardCode dummy_code; int flags = electron::WebEventModifiersToEventFlags(out->GetModifiers()); if (ui::DomCodeToUsLayoutDomKey(domCode, flags, &domKey, &dummy_code)) out->dom_key = static_cast(domKey); if ((out->GetType() == blink::WebInputEvent::kChar || out->GetType() == blink::WebInputEvent::kRawKeyDown)) { // Make sure to not read beyond the buffer in case some bad code doesn't // NULL-terminate it (this is called from plugins). size_t text_length_cap = blink::WebKeyboardEvent::kTextLengthCap; base::string16 text16 = base::UTF8ToUTF16(str); memset(out->text, 0, text_length_cap); memset(out->unmodified_text, 0, text_length_cap); for (size_t i = 0; i < std::min(text_length_cap, text16.size()); ++i) { out->text[i] = text16[i]; out->unmodified_text[i] = text16[i]; } } return true; } bool Converter::FromV8(v8::Isolate* isolate, v8::Local val, blink::WebMouseEvent* out) { gin_helper::Dictionary dict; if (!ConvertFromV8(isolate, val, &dict)) return false; if (!ConvertFromV8(isolate, val, static_cast(out))) return false; float x = 0.f; float y = 0.f; if (!dict.Get("x", &x) || !dict.Get("y", &y)) return false; out->SetPositionInWidget(x, y); if (!dict.Get("button", &out->button)) out->button = blink::WebMouseEvent::Button::kLeft; float global_x = 0.f; float global_y = 0.f; dict.Get("globalX", &global_x); dict.Get("globalY", &global_y); out->SetPositionInScreen(global_x, global_y); dict.Get("movementX", &out->movement_x); dict.Get("movementY", &out->movement_y); dict.Get("clickCount", &out->click_count); return true; } bool Converter::FromV8( v8::Isolate* isolate, v8::Local val, blink::WebMouseWheelEvent* out) { gin_helper::Dictionary dict; if (!ConvertFromV8(isolate, val, &dict)) return false; if (!ConvertFromV8(isolate, val, static_cast(out))) return false; dict.Get("deltaX", &out->delta_x); dict.Get("deltaY", &out->delta_y); dict.Get("wheelTicksX", &out->wheel_ticks_x); dict.Get("wheelTicksY", &out->wheel_ticks_y); dict.Get("accelerationRatioX", &out->acceleration_ratio_x); dict.Get("accelerationRatioY", &out->acceleration_ratio_y); bool has_precise_scrolling_deltas = false; dict.Get("hasPreciseScrollingDeltas", &has_precise_scrolling_deltas); if (has_precise_scrolling_deltas) { out->delta_units = ui::input_types::ScrollGranularity::kScrollByPrecisePixel; } else { out->delta_units = ui::input_types::ScrollGranularity::kScrollByPixel; } #if defined(USE_AURA) // Matches the behavior of ui/events/blink/web_input_event_traits.cc: bool can_scroll = true; if (dict.Get("canScroll", &can_scroll) && !can_scroll) { out->delta_units = ui::input_types::ScrollGranularity::kScrollByPage; out->SetModifiers(out->GetModifiers() & ~blink::WebInputEvent::kControlKey); } #endif return true; } bool Converter::FromV8(v8::Isolate* isolate, v8::Local val, blink::WebFloatPoint* out) { gin_helper::Dictionary dict; if (!ConvertFromV8(isolate, val, &dict)) return false; return dict.Get("x", &out->x) && dict.Get("y", &out->y); } template <> struct Converter> { static bool FromV8(v8::Isolate* isolate, v8::Local val, base::Optional* out) { gin_helper::Dictionary dict; if (!ConvertFromV8(isolate, val, &dict)) return false; blink::WebPoint point; bool success = dict.Get("x", &point.x) && dict.Get("y", &point.y); if (!success) return false; out->emplace(point); return true; } }; bool Converter::FromV8(v8::Isolate* isolate, v8::Local val, blink::WebSize* out) { gin_helper::Dictionary dict; if (!ConvertFromV8(isolate, val, &dict)) return false; return dict.Get("width", &out->width) && dict.Get("height", &out->height); } bool Converter::FromV8( v8::Isolate* isolate, v8::Local val, blink::WebDeviceEmulationParams* out) { gin_helper::Dictionary dict; if (!ConvertFromV8(isolate, val, &dict)) return false; std::string screen_position; if (dict.Get("screenPosition", &screen_position)) { screen_position = base::ToLowerASCII(screen_position); if (screen_position == "mobile") out->screen_position = blink::WebDeviceEmulationParams::kMobile; else if (screen_position == "desktop") out->screen_position = blink::WebDeviceEmulationParams::kDesktop; else return false; } dict.Get("screenSize", &out->screen_size); dict.Get("viewPosition", &out->view_position); dict.Get("deviceScaleFactor", &out->device_scale_factor); dict.Get("viewSize", &out->view_size); dict.Get("scale", &out->scale); return true; } // static v8::Local Converter::ToV8( v8::Isolate* isolate, const blink::WebContextMenuData::MediaType& in) { switch (in) { case blink::WebContextMenuData::kMediaTypeImage: return StringToV8(isolate, "image"); case blink::WebContextMenuData::kMediaTypeVideo: return StringToV8(isolate, "video"); case blink::WebContextMenuData::kMediaTypeAudio: return StringToV8(isolate, "audio"); case blink::WebContextMenuData::kMediaTypeCanvas: return StringToV8(isolate, "canvas"); case blink::WebContextMenuData::kMediaTypeFile: return StringToV8(isolate, "file"); case blink::WebContextMenuData::kMediaTypePlugin: return StringToV8(isolate, "plugin"); default: return StringToV8(isolate, "none"); } } // static v8::Local Converter::ToV8( v8::Isolate* isolate, const blink::WebContextMenuData::InputFieldType& in) { switch (in) { case blink::WebContextMenuData::kInputFieldTypePlainText: return StringToV8(isolate, "plainText"); case blink::WebContextMenuData::kInputFieldTypePassword: return StringToV8(isolate, "password"); case blink::WebContextMenuData::kInputFieldTypeOther: return StringToV8(isolate, "other"); default: return StringToV8(isolate, "none"); } } v8::Local EditFlagsToV8(v8::Isolate* isolate, int editFlags) { gin_helper::Dictionary dict = gin::Dictionary::CreateEmpty(isolate); dict.Set("canUndo", !!(editFlags & blink::WebContextMenuData::kCanUndo)); dict.Set("canRedo", !!(editFlags & blink::WebContextMenuData::kCanRedo)); dict.Set("canCut", !!(editFlags & blink::WebContextMenuData::kCanCut)); dict.Set("canCopy", !!(editFlags & blink::WebContextMenuData::kCanCopy)); bool pasteFlag = false; if (editFlags & blink::WebContextMenuData::kCanPaste) { std::vector types; bool ignore; ui::Clipboard::GetForCurrentThread()->ReadAvailableTypes( ui::ClipboardBuffer::kCopyPaste, &types, &ignore); pasteFlag = !types.empty(); } dict.Set("canPaste", pasteFlag); dict.Set("canDelete", !!(editFlags & blink::WebContextMenuData::kCanDelete)); dict.Set("canSelectAll", !!(editFlags & blink::WebContextMenuData::kCanSelectAll)); return ConvertToV8(isolate, dict); } v8::Local MediaFlagsToV8(v8::Isolate* isolate, int mediaFlags) { gin_helper::Dictionary dict = gin::Dictionary::CreateEmpty(isolate); dict.Set("inError", !!(mediaFlags & blink::WebContextMenuData::kMediaInError)); dict.Set("isPaused", !!(mediaFlags & blink::WebContextMenuData::kMediaPaused)); dict.Set("isMuted", !!(mediaFlags & blink::WebContextMenuData::kMediaMuted)); dict.Set("hasAudio", !!(mediaFlags & blink::WebContextMenuData::kMediaHasAudio)); dict.Set("isLooping", (mediaFlags & blink::WebContextMenuData::kMediaLoop) != 0); dict.Set("isControlsVisible", (mediaFlags & blink::WebContextMenuData::kMediaControls) != 0); dict.Set("canToggleControls", !!(mediaFlags & blink::WebContextMenuData::kMediaCanToggleControls)); dict.Set("canRotate", !!(mediaFlags & blink::WebContextMenuData::kMediaCanRotate)); return ConvertToV8(isolate, dict); } v8::Local Converter::ToV8( v8::Isolate* isolate, const blink::WebCacheResourceTypeStat& stat) { gin_helper::Dictionary dict = gin::Dictionary::CreateEmpty(isolate); dict.Set("count", static_cast(stat.count)); dict.Set("size", static_cast(stat.size)); dict.Set("liveSize", static_cast(stat.decoded_size)); return dict.GetHandle(); } v8::Local Converter::ToV8( v8::Isolate* isolate, const blink::WebCacheResourceTypeStats& stats) { gin_helper::Dictionary dict = gin::Dictionary::CreateEmpty(isolate); dict.Set("images", stats.images); dict.Set("scripts", stats.scripts); dict.Set("cssStyleSheets", stats.css_style_sheets); dict.Set("xslStyleSheets", stats.xsl_style_sheets); dict.Set("fonts", stats.fonts); dict.Set("other", stats.other); return dict.GetHandle(); } // static v8::Local Converter::ToV8( v8::Isolate* isolate, const network::mojom::ReferrerPolicy& in) { switch (in) { case network::mojom::ReferrerPolicy::kDefault: return StringToV8(isolate, "default"); case network::mojom::ReferrerPolicy::kAlways: return StringToV8(isolate, "unsafe-url"); case network::mojom::ReferrerPolicy::kNoReferrerWhenDowngrade: return StringToV8(isolate, "no-referrer-when-downgrade"); case network::mojom::ReferrerPolicy::kNever: return StringToV8(isolate, "no-referrer"); case network::mojom::ReferrerPolicy::kOrigin: return StringToV8(isolate, "origin"); case network::mojom::ReferrerPolicy:: kNoReferrerWhenDowngradeOriginWhenCrossOrigin: return StringToV8(isolate, "strict-origin-when-cross-origin"); case network::mojom::ReferrerPolicy::kSameOrigin: return StringToV8(isolate, "same-origin"); case network::mojom::ReferrerPolicy::kStrictOrigin: return StringToV8(isolate, "strict-origin"); default: return StringToV8(isolate, "no-referrer"); } } // static bool Converter::FromV8( v8::Isolate* isolate, v8::Handle val, network::mojom::ReferrerPolicy* out) { std::string policy = base::ToLowerASCII(gin::V8ToString(isolate, val)); if (policy == "default") *out = network::mojom::ReferrerPolicy::kDefault; else if (policy == "unsafe-url") *out = network::mojom::ReferrerPolicy::kAlways; else if (policy == "no-referrer-when-downgrade") *out = network::mojom::ReferrerPolicy::kNoReferrerWhenDowngrade; else if (policy == "no-referrer") *out = network::mojom::ReferrerPolicy::kNever; else if (policy == "origin") *out = network::mojom::ReferrerPolicy::kOrigin; else if (policy == "strict-origin-when-cross-origin") *out = network::mojom::ReferrerPolicy:: kNoReferrerWhenDowngradeOriginWhenCrossOrigin; else if (policy == "same-origin") *out = network::mojom::ReferrerPolicy::kSameOrigin; else if (policy == "strict-origin") *out = network::mojom::ReferrerPolicy::kStrictOrigin; else return false; return true; } namespace { constexpr uint8_t kNewSerializationTag = 0; constexpr uint8_t kOldSerializationTag = 1; class V8Serializer : public v8::ValueSerializer::Delegate { public: explicit V8Serializer(v8::Isolate* isolate, bool use_old_serialization = false) : isolate_(isolate), serializer_(isolate, this), use_old_serialization_(use_old_serialization) {} ~V8Serializer() override = default; bool Serialize(v8::Local value, blink::CloneableMessage* out) { serializer_.WriteHeader(); if (use_old_serialization_) { WriteTag(kOldSerializationTag); if (!WriteBaseValue(value)) { isolate_->ThrowException( StringToV8(isolate_, "An object could not be cloned.")); return false; } } else { WriteTag(kNewSerializationTag); bool wrote_value; v8::TryCatch try_catch(isolate_); if (!serializer_.WriteValue(isolate_->GetCurrentContext(), value) .To(&wrote_value)) { try_catch.Reset(); if (!V8Serializer(isolate_, true).Serialize(value, out)) { try_catch.ReThrow(); return false; } return true; } DCHECK(wrote_value); } std::pair buffer = serializer_.Release(); DCHECK_EQ(buffer.first, data_.data()); out->encoded_message = base::make_span(buffer.first, buffer.second); out->owned_encoded_message = std::move(data_); return true; } bool WriteBaseValue(v8::Local object) { node::Environment* env = node::Environment::GetCurrent(isolate_); if (env) { electron::EmitDeprecationWarning( env, "Passing functions, DOM objects and other non-cloneable JavaScript " "objects to IPC methods is deprecated and will throw an exception " "beginning with Electron 9.", "DeprecationWarning"); } base::Value value; if (!ConvertFromV8(isolate_, object, &value)) { return false; } mojo::Message message = mojo_base::mojom::Value::SerializeAsMessage(&value); serializer_.WriteUint32(message.data_num_bytes()); serializer_.WriteRawBytes(message.data(), message.data_num_bytes()); return true; } void WriteTag(uint8_t tag) { serializer_.WriteRawBytes(&tag, 1); } // v8::ValueSerializer::Delegate void* ReallocateBufferMemory(void* old_buffer, size_t size, size_t* actual_size) override { DCHECK_EQ(old_buffer, data_.data()); data_.resize(size); *actual_size = data_.capacity(); return data_.data(); } void FreeBufferMemory(void* buffer) override { DCHECK_EQ(buffer, data_.data()); data_ = {}; } void ThrowDataCloneError(v8::Local message) override { isolate_->ThrowException(v8::Exception::Error(message)); } private: v8::Isolate* isolate_; std::vector data_; v8::ValueSerializer serializer_; bool use_old_serialization_; }; class V8Deserializer : public v8::ValueDeserializer::Delegate { public: V8Deserializer(v8::Isolate* isolate, const blink::CloneableMessage& message) : isolate_(isolate), deserializer_(isolate, message.encoded_message.data(), message.encoded_message.size(), this) {} v8::Local Deserialize() { v8::EscapableHandleScope scope(isolate_); auto context = isolate_->GetCurrentContext(); bool read_header; if (!deserializer_.ReadHeader(context).To(&read_header)) return v8::Null(isolate_); DCHECK(read_header); uint8_t tag; if (!ReadTag(&tag)) return v8::Null(isolate_); switch (tag) { case kNewSerializationTag: { v8::Local value; if (!deserializer_.ReadValue(context).ToLocal(&value)) { return v8::Null(isolate_); } return scope.Escape(value); } case kOldSerializationTag: { v8::Local value; if (!ReadBaseValue(&value)) { return v8::Null(isolate_); } return scope.Escape(value); } default: NOTREACHED() << "Invalid tag: " << tag; return v8::Null(isolate_); } } bool ReadTag(uint8_t* tag) { const void* tag_bytes; if (!deserializer_.ReadRawBytes(1, &tag_bytes)) return false; *tag = *reinterpret_cast(tag_bytes); return true; } bool ReadBaseValue(v8::Local* value) { uint32_t length; const void* data; if (!deserializer_.ReadUint32(&length) || !deserializer_.ReadRawBytes(length, &data)) { return false; } mojo::Message message( base::make_span(reinterpret_cast(data), length), {}); base::Value out; if (!mojo_base::mojom::Value::DeserializeFromMessage(std::move(message), &out)) { return false; } *value = ConvertToV8(isolate_, out); return true; } private: v8::Isolate* isolate_; v8::ValueDeserializer deserializer_; }; } // namespace v8::Local Converter::ToV8( v8::Isolate* isolate, const blink::CloneableMessage& in) { return V8Deserializer(isolate, in).Deserialize(); } bool Converter::FromV8(v8::Isolate* isolate, v8::Handle val, blink::CloneableMessage* out) { return V8Serializer(isolate).Serialize(val, out); } } // namespace gin