Move content/shell/$process/web_test to content/web_test/$process

This separates the web test code out of content/shell/. While it is
built into content_shell, it is only part of it when --run-web-tests
is used, and is not conceptually part of the content shell library
being embedded in other shell apps like extensions's shell, ash shell,
or the views example app.

[email protected], [email protected], [email protected]
TBR=

Bug: 866140
Change-Id: I1787d7bbd3be27d1232ab6ddab64db2b5df1d3a4
Reviewed-on: https://chromium-review.googlesource.com/c/chromium/src/+/2389045
Commit-Queue: danakj <[email protected]>
Reviewed-by: Michael Moss <[email protected]>
Reviewed-by: Avi Drissman <[email protected]>
Reviewed-by: Ken Buchanan <[email protected]>
Cr-Commit-Position: refs/heads/master@{#803914}
diff --git a/content/web_test/renderer/text_input_controller.cc b/content/web_test/renderer/text_input_controller.cc
new file mode 100644
index 0000000..c40d88d
--- /dev/null
+++ b/content/web_test/renderer/text_input_controller.cc
@@ -0,0 +1,424 @@
+// Copyright 2020 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/web_test/renderer/text_input_controller.h"
+
+#include "base/macros.h"
+#include "content/web_test/renderer/web_view_test_proxy.h"
+#include "gin/arguments.h"
+#include "gin/handle.h"
+#include "gin/object_template_builder.h"
+#include "gin/wrappable.h"
+#include "third_party/blink/public/common/input/web_coalesced_input_event.h"
+#include "third_party/blink/public/common/input/web_keyboard_event.h"
+#include "third_party/blink/public/platform/web_input_event_result.h"
+#include "third_party/blink/public/web/blink.h"
+#include "third_party/blink/public/web/web_frame_widget.h"
+#include "third_party/blink/public/web/web_input_method_controller.h"
+#include "third_party/blink/public/web/web_local_frame.h"
+#include "third_party/blink/public/web/web_range.h"
+#include "third_party/blink/public/web/web_view.h"
+#include "third_party/skia/include/core/SkColor.h"
+#include "ui/base/ime/ime_text_span.h"
+#include "ui/events/base_event_utils.h"
+#include "v8/include/v8.h"
+
+namespace content {
+
+class TextInputControllerBindings
+    : public gin::Wrappable<TextInputControllerBindings> {
+ public:
+  static gin::WrapperInfo kWrapperInfo;
+
+  static void Install(base::WeakPtr<TextInputController> controller,
+                      blink::WebLocalFrame* frame);
+
+ private:
+  explicit TextInputControllerBindings(
+      base::WeakPtr<TextInputController> controller);
+  ~TextInputControllerBindings() override;
+
+  // gin::Wrappable:
+  gin::ObjectTemplateBuilder GetObjectTemplateBuilder(
+      v8::Isolate* isolate) override;
+
+  void InsertText(const std::string& text);
+  void UnmarkText();
+  void UnmarkAndUnselectText();
+  void DoCommand(const std::string& text);
+  void ExtendSelectionAndDelete(int before, int after);
+  void DeleteSurroundingText(int before, int after);
+  void SetMarkedText(const std::string& text, int start, int length);
+  void SetMarkedTextFromExistingText(int start, int length);
+  bool HasMarkedText();
+  std::vector<int> MarkedRange();
+  std::vector<int> SelectedRange();
+  std::vector<int> FirstRectForCharacterRange(unsigned location,
+                                              unsigned length);
+  void SetComposition(const std::string& text);
+  void ForceTextInputStateUpdate();
+
+  base::WeakPtr<TextInputController> controller_;
+
+  DISALLOW_COPY_AND_ASSIGN(TextInputControllerBindings);
+};
+
+gin::WrapperInfo TextInputControllerBindings::kWrapperInfo = {
+    gin::kEmbedderNativeGin};
+
+// static
+void TextInputControllerBindings::Install(
+    base::WeakPtr<TextInputController> controller,
+    blink::WebLocalFrame* frame) {
+  v8::Isolate* isolate = blink::MainThreadIsolate();
+  v8::HandleScope handle_scope(isolate);
+  v8::Local<v8::Context> context = frame->MainWorldScriptContext();
+  if (context.IsEmpty())
+    return;
+
+  v8::Context::Scope context_scope(context);
+
+  gin::Handle<TextInputControllerBindings> bindings =
+      gin::CreateHandle(isolate, new TextInputControllerBindings(controller));
+  if (bindings.IsEmpty())
+    return;
+  v8::Local<v8::Object> global = context->Global();
+  global
+      ->Set(context, gin::StringToV8(isolate, "textInputController"),
+            bindings.ToV8())
+      .Check();
+}
+
+TextInputControllerBindings::TextInputControllerBindings(
+    base::WeakPtr<TextInputController> controller)
+    : controller_(controller) {}
+
+TextInputControllerBindings::~TextInputControllerBindings() {}
+
+gin::ObjectTemplateBuilder
+TextInputControllerBindings::GetObjectTemplateBuilder(v8::Isolate* isolate) {
+  return gin::Wrappable<TextInputControllerBindings>::GetObjectTemplateBuilder(
+             isolate)
+      .SetMethod("insertText", &TextInputControllerBindings::InsertText)
+      .SetMethod("unmarkText", &TextInputControllerBindings::UnmarkText)
+      .SetMethod("unmarkAndUnselectText",
+                 &TextInputControllerBindings::UnmarkAndUnselectText)
+      .SetMethod("doCommand", &TextInputControllerBindings::DoCommand)
+      .SetMethod("extendSelectionAndDelete",
+                 &TextInputControllerBindings::ExtendSelectionAndDelete)
+      .SetMethod("deleteSurroundingText",
+                 &TextInputControllerBindings::DeleteSurroundingText)
+      .SetMethod("setMarkedText", &TextInputControllerBindings::SetMarkedText)
+      .SetMethod("setMarkedTextFromExistingText",
+                 &TextInputControllerBindings::SetMarkedTextFromExistingText)
+      .SetMethod("hasMarkedText", &TextInputControllerBindings::HasMarkedText)
+      .SetMethod("markedRange", &TextInputControllerBindings::MarkedRange)
+      .SetMethod("selectedRange", &TextInputControllerBindings::SelectedRange)
+      .SetMethod("firstRectForCharacterRange",
+                 &TextInputControllerBindings::FirstRectForCharacterRange)
+      .SetMethod("setComposition", &TextInputControllerBindings::SetComposition)
+      .SetMethod("forceTextInputStateUpdate",
+                 &TextInputControllerBindings::ForceTextInputStateUpdate);
+}
+
+void TextInputControllerBindings::InsertText(const std::string& text) {
+  if (controller_)
+    controller_->InsertText(text);
+}
+
+void TextInputControllerBindings::UnmarkText() {
+  if (controller_)
+    controller_->UnmarkText();
+}
+
+void TextInputControllerBindings::UnmarkAndUnselectText() {
+  if (controller_)
+    controller_->UnmarkAndUnselectText();
+}
+
+void TextInputControllerBindings::DoCommand(const std::string& text) {
+  if (controller_)
+    controller_->DoCommand(text);
+}
+
+void TextInputControllerBindings::ExtendSelectionAndDelete(int before,
+                                                           int after) {
+  if (controller_)
+    controller_->ExtendSelectionAndDelete(before, after);
+}
+
+void TextInputControllerBindings::DeleteSurroundingText(int before, int after) {
+  if (controller_)
+    controller_->DeleteSurroundingText(before, after);
+}
+
+void TextInputControllerBindings::SetMarkedText(const std::string& text,
+                                                int start,
+                                                int length) {
+  if (controller_)
+    controller_->SetMarkedText(text, start, length);
+}
+
+void TextInputControllerBindings::SetMarkedTextFromExistingText(int start,
+                                                                int end) {
+  if (controller_)
+    controller_->SetMarkedTextFromExistingText(start, end);
+}
+
+bool TextInputControllerBindings::HasMarkedText() {
+  return controller_ ? controller_->HasMarkedText() : false;
+}
+
+std::vector<int> TextInputControllerBindings::MarkedRange() {
+  return controller_ ? controller_->MarkedRange() : std::vector<int>();
+}
+
+std::vector<int> TextInputControllerBindings::SelectedRange() {
+  return controller_ ? controller_->SelectedRange() : std::vector<int>();
+}
+
+std::vector<int> TextInputControllerBindings::FirstRectForCharacterRange(
+    unsigned location,
+    unsigned length) {
+  return controller_ ? controller_->FirstRectForCharacterRange(location, length)
+                     : std::vector<int>();
+}
+
+void TextInputControllerBindings::SetComposition(const std::string& text) {
+  if (controller_)
+    controller_->SetComposition(text);
+}
+void TextInputControllerBindings::ForceTextInputStateUpdate() {
+  if (controller_)
+    controller_->ForceTextInputStateUpdate();
+}
+// TextInputController ---------------------------------------------------------
+
+TextInputController::TextInputController(WebViewTestProxy* web_view_test_proxy)
+    : web_view_test_proxy_(web_view_test_proxy) {}
+
+TextInputController::~TextInputController() {}
+
+void TextInputController::Install(blink::WebLocalFrame* frame) {
+  TextInputControllerBindings::Install(weak_factory_.GetWeakPtr(), frame);
+}
+
+void TextInputController::InsertText(const std::string& text) {
+  if (auto* controller = GetInputMethodController()) {
+    controller->CommitText(blink::WebString::FromUTF8(text),
+                           std::vector<ui::ImeTextSpan>(), blink::WebRange(),
+                           0);
+  }
+}
+
+void TextInputController::UnmarkText() {
+  if (auto* controller = GetInputMethodController()) {
+    controller->FinishComposingText(
+        blink::WebInputMethodController::kKeepSelection);
+  }
+}
+
+void TextInputController::UnmarkAndUnselectText() {
+  if (auto* controller = GetInputMethodController()) {
+    controller->FinishComposingText(
+        blink::WebInputMethodController::kDoNotKeepSelection);
+  }
+}
+
+void TextInputController::DoCommand(const std::string& text) {
+  if (view()->MainFrame()) {
+    CHECK(view()->MainFrame()->ToWebLocalFrame()) << "This function cannot be "
+                                                     "called if the main frame "
+                                                     "is not a local frame.";
+    view()->MainFrame()->ToWebLocalFrame()->ExecuteCommand(
+        blink::WebString::FromUTF8(text));
+  }
+}
+
+void TextInputController::ExtendSelectionAndDelete(int before, int after) {
+  if (view()->MainFrame()) {
+    CHECK(view()->MainFrame()->ToWebLocalFrame()) << "This function cannot be "
+                                                     "called if the main frame "
+                                                     "is not a local frame.";
+    view()->MainFrame()->ToWebLocalFrame()->ExtendSelectionAndDelete(before,
+                                                                     after);
+  }
+}
+
+void TextInputController::DeleteSurroundingText(int before, int after) {
+  if (view()->MainFrame()) {
+    CHECK(view()->MainFrame()->ToWebLocalFrame()) << "This function cannot be "
+                                                     "called if the main frame "
+                                                     "is not a local frame.";
+    view()->MainFrame()->ToWebLocalFrame()->DeleteSurroundingText(before,
+                                                                  after);
+  }
+}
+
+void TextInputController::SetMarkedText(const std::string& text,
+                                        int start,
+                                        int length) {
+  blink::WebString web_text(blink::WebString::FromUTF8(text));
+
+  // Split underline into up to 3 elements (before, selection, and after).
+  std::vector<ui::ImeTextSpan> ime_text_spans;
+  ui::ImeTextSpan ime_text_span;
+  if (!start) {
+    ime_text_span.end_offset = length;
+  } else {
+    ime_text_span.end_offset = start;
+    ime_text_spans.push_back(ime_text_span);
+    ime_text_span.start_offset = start;
+    ime_text_span.end_offset = start + length;
+  }
+  ime_text_span.thickness = ui::ImeTextSpan::Thickness::kThick;
+  ime_text_span.underline_style = ui::ImeTextSpan::UnderlineStyle::kSolid;
+  ime_text_spans.push_back(ime_text_span);
+  if (start + length < static_cast<int>(web_text.length())) {
+    ime_text_span.start_offset = ime_text_span.end_offset;
+    ime_text_span.end_offset = web_text.length();
+    ime_text_span.thickness = ui::ImeTextSpan::Thickness::kThin;
+    ime_text_span.underline_style = ui::ImeTextSpan::UnderlineStyle::kSolid;
+    ime_text_spans.push_back(ime_text_span);
+  }
+
+  if (auto* controller = GetInputMethodController()) {
+    controller->SetComposition(web_text, ime_text_spans, blink::WebRange(),
+                               start, start + length);
+  }
+}
+
+void TextInputController::SetMarkedTextFromExistingText(int start, int end) {
+  if (!view()->MainFrame())
+    return;
+
+  CHECK(view()->MainFrame()->ToWebLocalFrame()) << "This function cannot be "
+                                                   "called if the main frame "
+                                                   "is not a local frame.";
+
+  view()->MainFrame()->ToWebLocalFrame()->SetCompositionFromExistingText(
+      start, end, std::vector<ui::ImeTextSpan>());
+}
+
+bool TextInputController::HasMarkedText() {
+  if (!view()->MainFrame())
+    return false;
+
+  CHECK(view()->MainFrame()->ToWebLocalFrame()) << "This function cannot be "
+                                                   "called if the main frame "
+                                                   "is not a local frame.";
+
+  return view()->MainFrame()->ToWebLocalFrame()->HasMarkedText();
+}
+
+std::vector<int> TextInputController::MarkedRange() {
+  if (!view()->MainFrame())
+    return std::vector<int>();
+
+  CHECK(view()->MainFrame()->ToWebLocalFrame()) << "This function cannot be "
+                                                   "called if the main frame "
+                                                   "is not a local frame.";
+
+  blink::WebRange range = view()->MainFrame()->ToWebLocalFrame()->MarkedRange();
+  std::vector<int> int_array(2);
+  int_array[0] = range.StartOffset();
+  int_array[1] = range.EndOffset();
+
+  return int_array;
+}
+
+std::vector<int> TextInputController::SelectedRange() {
+  if (!view()->MainFrame())
+    return std::vector<int>();
+
+  CHECK(view()->MainFrame()->ToWebLocalFrame()) << "This function cannot be "
+                                                   "called if the main frame "
+                                                   "is not a local frame.";
+
+  blink::WebRange range =
+      view()->MainFrame()->ToWebLocalFrame()->SelectionRange();
+  if (range.IsNull())
+    return std::vector<int>();
+  std::vector<int> int_array(2);
+  int_array[0] = range.StartOffset();
+  int_array[1] = range.EndOffset();
+
+  return int_array;
+}
+
+std::vector<int> TextInputController::FirstRectForCharacterRange(
+    unsigned location,
+    unsigned length) {
+  blink::WebRect rect;
+  if (!view()->FocusedFrame() ||
+      !view()->FocusedFrame()->FirstRectForCharacterRange(location, length,
+                                                          rect)) {
+    return std::vector<int>();
+  }
+
+  std::vector<int> int_array(4);
+  int_array[0] = rect.x;
+  int_array[1] = rect.y;
+  int_array[2] = rect.width;
+  int_array[3] = rect.height;
+
+  return int_array;
+}
+
+void TextInputController::SetComposition(const std::string& text) {
+  // Sends a keydown event with key code = 0xE5 to emulate input method
+  // behavior.
+  blink::WebKeyboardEvent key_down(blink::WebInputEvent::Type::kRawKeyDown,
+                                   blink::WebInputEvent::kNoModifiers,
+                                   ui::EventTimeForNow());
+
+  key_down.windows_key_code = 0xE5;  // VKEY_PROCESSKEY
+  view()->MainFrameWidget()->HandleInputEvent(
+      blink::WebCoalescedInputEvent(key_down, ui::LatencyInfo()));
+
+  // The value returned by std::string::length() may not correspond to the
+  // actual number of encoded characters in sequences of multi-byte or
+  // variable-length characters.
+  blink::WebString newText = blink::WebString::FromUTF8(text);
+  size_t textLength = newText.length();
+
+  std::vector<ui::ImeTextSpan> ime_text_spans;
+  ime_text_spans.push_back(ui::ImeTextSpan(
+      ui::ImeTextSpan::Type::kComposition, 0, textLength,
+      ui::ImeTextSpan::Thickness::kThin,
+      ui::ImeTextSpan::UnderlineStyle::kSolid, SK_ColorTRANSPARENT));
+  if (auto* controller = GetInputMethodController()) {
+    controller->SetComposition(
+        newText, blink::WebVector<ui::ImeTextSpan>(std::move(ime_text_spans)),
+        blink::WebRange(), textLength, textLength);
+  }
+}
+
+void TextInputController::ForceTextInputStateUpdate() {
+  // TODO(lukasza): Finish adding OOPIF support to the web tests harness.
+  RenderFrameImpl* main_frame = web_view_test_proxy_->GetMainRenderFrame();
+  CHECK(main_frame) << "WebView does not have a local main frame and"
+                    << " cannot handle input method controller tasks.";
+  RenderWidget* main_widget = main_frame->GetLocalRootRenderWidget();
+  main_widget->GetWebWidget()->ShowVirtualKeyboard();
+}
+
+blink::WebView* TextInputController::view() {
+  return web_view_test_proxy_->GetWebView();
+}
+
+blink::WebInputMethodController*
+TextInputController::GetInputMethodController() {
+  if (!view()->MainFrame())
+    return nullptr;
+
+  // TODO(lukasza): Finish adding OOPIF support to the web tests harness.
+  CHECK(view()->MainFrame()->IsWebLocalFrame())
+      << "WebView does not have a local main frame and"
+         " cannot handle input method controller tasks.";
+
+  return view()->MainFrameWidget()->GetActiveWebInputMethodController();
+}
+
+}  // namespace content