blob: 4b454e0be043ef7ee452d5c65374e84de667a969 [file] [log] [blame]
/*
* Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
* Copyright (C) 2008 Matt Lilek <webkit@mattlilek.com>
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
#include "third_party/blink/renderer/core/inspector/dev_tools_host.h"
#include "third_party/blink/public/web/web_menu_item_info.h"
#include "third_party/blink/renderer/bindings/core/v8/script_source_code.h"
#include "third_party/blink/renderer/bindings/core/v8/v8_binding_for_core.h"
#include "third_party/blink/renderer/bindings/core/v8/v8_script_runner.h"
#include "third_party/blink/renderer/core/clipboard/system_clipboard.h"
#include "third_party/blink/renderer/core/dom/events/event.h"
#include "third_party/blink/renderer/core/dom/events/event_target.h"
#include "third_party/blink/renderer/core/dom/user_gesture_indicator.h"
#include "third_party/blink/renderer/core/execution_context/execution_context.h"
#include "third_party/blink/renderer/core/frame/local_dom_window.h"
#include "third_party/blink/renderer/core/frame/local_frame.h"
#include "third_party/blink/renderer/core/frame/local_frame_view.h"
#include "third_party/blink/renderer/core/html/parser/text_resource_decoder.h"
#include "third_party/blink/renderer/core/input/context_menu_allowed_scope.h"
#include "third_party/blink/renderer/core/inspector/inspector_frontend_client.h"
#include "third_party/blink/renderer/core/loader/frame_loader.h"
#include "third_party/blink/renderer/core/page/chrome_client.h"
#include "third_party/blink/renderer/core/page/context_menu_controller.h"
#include "third_party/blink/renderer/core/page/context_menu_provider.h"
#include "third_party/blink/renderer/core/page/page.h"
#include "third_party/blink/renderer/platform/bindings/script_forbidden_scope.h"
#include "third_party/blink/renderer/platform/bindings/script_state.h"
#include "third_party/blink/renderer/platform/loader/fetch/resource_error.h"
#include "third_party/blink/renderer/platform/loader/fetch/resource_fetcher.h"
#include "third_party/blink/renderer/platform/loader/fetch/resource_request.h"
#include "third_party/blink/renderer/platform/loader/fetch/resource_response.h"
#include "third_party/blink/renderer/platform/shared_buffer.h"
#include "third_party/blink/renderer/platform/wtf/text/string_builder.h"
namespace blink {
class FrontendMenuProvider final : public ContextMenuProvider {
public:
static FrontendMenuProvider* Create(DevToolsHost* devtools_host,
WebVector<WebMenuItemInfo> items) {
return MakeGarbageCollected<FrontendMenuProvider>(devtools_host,
std::move(items));
}
FrontendMenuProvider(DevToolsHost* devtools_host,
WebVector<WebMenuItemInfo> items)
: devtools_host_(devtools_host), items_(std::move(items)) {}
~FrontendMenuProvider() override {
// Verify that this menu provider has been detached.
DCHECK(!devtools_host_);
}
void Trace(blink::Visitor* visitor) override {
visitor->Trace(devtools_host_);
ContextMenuProvider::Trace(visitor);
}
void Disconnect() { devtools_host_ = nullptr; }
void ContextMenuCleared() override {
if (devtools_host_) {
devtools_host_->EvaluateScript("DevToolsAPI.contextMenuCleared()");
devtools_host_->ClearMenuProvider();
devtools_host_ = nullptr;
}
items_.Clear();
}
WebVector<WebMenuItemInfo> PopulateContextMenu() override {
return std::move(items_);
}
void ContextMenuItemSelected(unsigned action) override {
if (!devtools_host_ || action >= DevToolsHost::kMaxContextMenuAction)
return;
devtools_host_->EvaluateScript("DevToolsAPI.contextMenuItemSelected(" +
String::Number(action) + ")");
}
private:
Member<DevToolsHost> devtools_host_;
WebVector<WebMenuItemInfo> items_;
};
DevToolsHost::DevToolsHost(InspectorFrontendClient* client,
LocalFrame* frontend_frame)
: client_(client),
frontend_frame_(frontend_frame),
menu_provider_(nullptr) {}
DevToolsHost::~DevToolsHost() = default;
void DevToolsHost::Trace(blink::Visitor* visitor) {
visitor->Trace(client_);
visitor->Trace(frontend_frame_);
visitor->Trace(menu_provider_);
ScriptWrappable::Trace(visitor);
}
void DevToolsHost::EvaluateScript(const String& expression) {
if (ScriptForbiddenScope::IsScriptForbidden())
return;
if (!frontend_frame_)
return;
ScriptState* script_state = ToScriptStateForMainWorld(frontend_frame_);
if (!script_state)
return;
ScriptState::Scope scope(script_state);
std::unique_ptr<UserGestureIndicator> gesture_indicator =
LocalFrame::NotifyUserActivation(frontend_frame_);
v8::MicrotasksScope microtasks(script_state->GetIsolate(),
v8::MicrotasksScope::kRunMicrotasks);
ScriptSourceCode source_code(expression, ScriptSourceLocationType::kInternal,
nullptr, KURL(), TextPosition());
V8ScriptRunner::CompileAndRunInternalScript(script_state->GetIsolate(),
script_state, source_code);
}
void DevToolsHost::DisconnectClient() {
client_ = nullptr;
if (menu_provider_) {
menu_provider_->Disconnect();
menu_provider_ = nullptr;
}
frontend_frame_ = nullptr;
}
float DevToolsHost::zoomFactor() {
if (!frontend_frame_)
return 1;
float zoom_factor = frontend_frame_->PageZoomFactor();
// Cancel the device scale factor applied to the zoom factor in
// use-zoom-for-dsf mode.
const ChromeClient* client =
frontend_frame_->View()->GetChromeClient();
float window_to_viewport_ratio = client->WindowToViewportScalar(1.0f);
return zoom_factor / window_to_viewport_ratio;
}
void DevToolsHost::copyText(const String& text) {
SystemClipboard::GetInstance().WritePlainText(text);
}
static String EscapeUnicodeNonCharacters(const String& str) {
const UChar kNonChar = 0xD800;
unsigned i = 0;
while (i < str.length() && str[i] < kNonChar)
++i;
if (i == str.length())
return str;
StringBuilder dst;
dst.Append(str, 0, i);
for (; i < str.length(); ++i) {
UChar c = str[i];
if (c >= kNonChar) {
unsigned symbol = static_cast<unsigned>(c);
String symbol_code = String::Format("\\u%04X", symbol);
dst.Append(symbol_code);
} else {
dst.Append(c);
}
}
return dst.ToString();
}
void DevToolsHost::sendMessageToEmbedder(const String& message) {
if (client_)
client_->SendMessageToEmbedder(EscapeUnicodeNonCharacters(message));
}
void DevToolsHost::ShowContextMenu(LocalFrame* target_frame,
float x,
float y,
WebVector<WebMenuItemInfo> items) {
DCHECK(frontend_frame_);
FrontendMenuProvider* menu_provider =
FrontendMenuProvider::Create(this, std::move(items));
menu_provider_ = menu_provider;
float zoom = target_frame->PageZoomFactor();
{
ContextMenuAllowedScope scope;
target_frame->GetPage()->GetContextMenuController().ClearContextMenu();
target_frame->GetPage()->GetContextMenuController().ShowContextMenuAtPoint(
target_frame, x * zoom, y * zoom, menu_provider);
}
}
bool DevToolsHost::isHostedMode() {
return false;
}
} // namespace blink