| /* |
| * 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 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 "config.h" |
| #include "InspectorFrontendHost.h" |
| |
| #include "ContextMenu.h" |
| #include "ContextMenuController.h" |
| #include "ContextMenuItem.h" |
| #include "ContextMenuProvider.h" |
| #include "DOMWrapperWorld.h" |
| #include "Document.h" |
| #include "Editor.h" |
| #include "Event.h" |
| #include "FocusController.h" |
| #include "HitTestResult.h" |
| #include "InspectorFrontendClient.h" |
| #include "JSMainThreadExecState.h" |
| #include "MainFrame.h" |
| #include "MouseEvent.h" |
| #include "Node.h" |
| #include "Page.h" |
| #include "Pasteboard.h" |
| #include "ScriptGlobalObject.h" |
| #include "ScriptState.h" |
| #include "Sound.h" |
| #include "UserGestureIndicator.h" |
| #include <bindings/ScriptFunctionCall.h> |
| #include <wtf/StdLibExtras.h> |
| |
| using namespace Inspector; |
| |
| namespace WebCore { |
| |
| #if ENABLE(CONTEXT_MENUS) |
| class FrontendMenuProvider : public ContextMenuProvider { |
| public: |
| static Ref<FrontendMenuProvider> create(InspectorFrontendHost* frontendHost, Deprecated::ScriptObject frontendApiObject, const Vector<ContextMenuItem>& items) |
| { |
| return adoptRef(*new FrontendMenuProvider(frontendHost, frontendApiObject, items)); |
| } |
| |
| void disconnect() |
| { |
| m_frontendApiObject = { }; |
| m_frontendHost = nullptr; |
| } |
| |
| private: |
| FrontendMenuProvider(InspectorFrontendHost* frontendHost, Deprecated::ScriptObject frontendApiObject, const Vector<ContextMenuItem>& items) |
| : m_frontendHost(frontendHost) |
| , m_frontendApiObject(frontendApiObject) |
| , m_items(items) |
| { |
| } |
| |
| virtual ~FrontendMenuProvider() |
| { |
| contextMenuCleared(); |
| } |
| |
| void populateContextMenu(ContextMenu* menu) override |
| { |
| for (auto& item : m_items) |
| menu->appendItem(item); |
| } |
| |
| void contextMenuItemSelected(ContextMenuAction action, const String&) override |
| { |
| if (m_frontendHost) { |
| UserGestureIndicator gestureIndicator(ProcessingUserGesture); |
| int itemNumber = action - ContextMenuItemBaseCustomTag; |
| |
| Deprecated::ScriptFunctionCall function(m_frontendApiObject, "contextMenuItemSelected", WebCore::functionCallHandlerFromAnyThread); |
| function.appendArgument(itemNumber); |
| function.call(); |
| } |
| } |
| |
| void contextMenuCleared() override |
| { |
| if (m_frontendHost) { |
| Deprecated::ScriptFunctionCall function(m_frontendApiObject, "contextMenuCleared", WebCore::functionCallHandlerFromAnyThread); |
| function.call(); |
| |
| m_frontendHost->m_menuProvider = nullptr; |
| } |
| m_items.clear(); |
| } |
| |
| InspectorFrontendHost* m_frontendHost; |
| Deprecated::ScriptObject m_frontendApiObject; |
| Vector<ContextMenuItem> m_items; |
| }; |
| #endif |
| |
| InspectorFrontendHost::InspectorFrontendHost(InspectorFrontendClient* client, Page* frontendPage) |
| : m_client(client) |
| , m_frontendPage(frontendPage) |
| #if ENABLE(CONTEXT_MENUS) |
| , m_menuProvider(nullptr) |
| #endif |
| { |
| } |
| |
| InspectorFrontendHost::~InspectorFrontendHost() |
| { |
| ASSERT(!m_client); |
| } |
| |
| void InspectorFrontendHost::disconnectClient() |
| { |
| m_client = nullptr; |
| #if ENABLE(CONTEXT_MENUS) |
| if (m_menuProvider) |
| m_menuProvider->disconnect(); |
| #endif |
| m_frontendPage = nullptr; |
| } |
| |
| void InspectorFrontendHost::loaded() |
| { |
| if (m_client) |
| m_client->frontendLoaded(); |
| } |
| |
| void InspectorFrontendHost::requestSetDockSide(const String& side) |
| { |
| if (!m_client) |
| return; |
| if (side == "undocked") |
| m_client->requestSetDockSide(InspectorFrontendClient::DockSide::Undocked); |
| else if (side == "right") |
| m_client->requestSetDockSide(InspectorFrontendClient::DockSide::Right); |
| else if (side == "bottom") |
| m_client->requestSetDockSide(InspectorFrontendClient::DockSide::Bottom); |
| } |
| |
| void InspectorFrontendHost::closeWindow() |
| { |
| if (m_client) { |
| m_client->closeWindow(); |
| disconnectClient(); // Disconnect from client. |
| } |
| } |
| |
| void InspectorFrontendHost::bringToFront() |
| { |
| if (m_client) |
| m_client->bringToFront(); |
| } |
| |
| void InspectorFrontendHost::inspectedURLChanged(const String& newURL) |
| { |
| if (m_client) |
| m_client->inspectedURLChanged(newURL); |
| } |
| |
| void InspectorFrontendHost::setZoomFactor(float zoom) |
| { |
| if (m_frontendPage) |
| m_frontendPage->mainFrame().setPageAndTextZoomFactors(zoom, 1); |
| } |
| |
| float InspectorFrontendHost::zoomFactor() |
| { |
| if (m_frontendPage) |
| return m_frontendPage->mainFrame().pageZoomFactor(); |
| |
| return 1.0; |
| } |
| |
| void InspectorFrontendHost::setAttachedWindowHeight(unsigned height) |
| { |
| if (m_client) |
| m_client->changeAttachedWindowHeight(height); |
| } |
| |
| void InspectorFrontendHost::setAttachedWindowWidth(unsigned width) |
| { |
| if (m_client) |
| m_client->changeAttachedWindowWidth(width); |
| } |
| |
| void InspectorFrontendHost::startWindowDrag() |
| { |
| if (m_client) |
| m_client->startWindowDrag(); |
| } |
| |
| void InspectorFrontendHost::moveWindowBy(float x, float y) const |
| { |
| if (m_client) |
| m_client->moveWindowBy(x, y); |
| } |
| |
| String InspectorFrontendHost::localizedStringsURL() |
| { |
| return m_client ? m_client->localizedStringsURL() : String(); |
| } |
| |
| String InspectorFrontendHost::backendCommandsURL() |
| { |
| return m_client ? m_client->backendCommandsURL() : String(); |
| } |
| |
| String InspectorFrontendHost::debuggableType() |
| { |
| return m_client ? m_client->debuggableType() : String(); |
| } |
| |
| unsigned InspectorFrontendHost::inspectionLevel() |
| { |
| return m_client ? m_client->inspectionLevel() : 1; |
| } |
| |
| String InspectorFrontendHost::platform() |
| { |
| #if PLATFORM(MAC) || PLATFORM(IOS) |
| return ASCIILiteral("mac"); |
| #elif OS(WINDOWS) |
| return ASCIILiteral("windows"); |
| #elif OS(LINUX) |
| return ASCIILiteral("linux"); |
| #elif OS(FREEBSD) |
| return ASCIILiteral("freebsd"); |
| #elif OS(OPENBSD) |
| return ASCIILiteral("openbsd"); |
| #elif OS(SOLARIS) |
| return ASCIILiteral("solaris"); |
| #else |
| return ASCIILiteral("unknown"); |
| #endif |
| } |
| |
| String InspectorFrontendHost::port() |
| { |
| #if PLATFORM(GTK) |
| return ASCIILiteral("gtk"); |
| #elif PLATFORM(EFL) |
| return ASCIILiteral("efl"); |
| #else |
| return ASCIILiteral("unknown"); |
| #endif |
| } |
| |
| void InspectorFrontendHost::copyText(const String& text) |
| { |
| Pasteboard::createForCopyAndPaste()->writePlainText(text, Pasteboard::CannotSmartReplace); |
| } |
| |
| void InspectorFrontendHost::killText(const String& text, bool shouldPrependToKillRing, bool shouldStartNewSequence) |
| { |
| if (!m_frontendPage) |
| return; |
| |
| Editor& editor = m_frontendPage->focusController().focusedOrMainFrame().editor(); |
| editor.setStartNewKillRingSequence(shouldStartNewSequence); |
| Editor::KillRingInsertionMode insertionMode = shouldPrependToKillRing ? Editor::KillRingInsertionMode::PrependText : Editor::KillRingInsertionMode::AppendText; |
| editor.addTextToKillRing(text, insertionMode); |
| } |
| |
| void InspectorFrontendHost::openInNewTab(const String& url) |
| { |
| if (WebCore::protocolIsJavaScript(url)) |
| return; |
| |
| if (m_client) |
| m_client->openInNewTab(url); |
| } |
| |
| bool InspectorFrontendHost::canSave() |
| { |
| if (m_client) |
| return m_client->canSave(); |
| return false; |
| } |
| |
| void InspectorFrontendHost::save(const String& url, const String& content, bool base64Encoded, bool forceSaveAs) |
| { |
| if (m_client) |
| m_client->save(url, content, base64Encoded, forceSaveAs); |
| } |
| |
| void InspectorFrontendHost::append(const String& url, const String& content) |
| { |
| if (m_client) |
| m_client->append(url, content); |
| } |
| |
| void InspectorFrontendHost::close(const String&) |
| { |
| } |
| |
| void InspectorFrontendHost::sendMessageToBackend(const String& message) |
| { |
| if (m_client) |
| m_client->sendMessageToBackend(message); |
| } |
| |
| #if ENABLE(CONTEXT_MENUS) |
| |
| void InspectorFrontendHost::showContextMenu(Event* event, const Vector<ContextMenuItem>& items) |
| { |
| if (!event) |
| return; |
| |
| ASSERT(m_frontendPage); |
| auto& state = *execStateFromPage(debuggerWorld(), m_frontendPage); |
| JSC::JSObject* frontendApiObject; |
| if (!ScriptGlobalObject::get(state, "InspectorFrontendAPI", frontendApiObject)) { |
| ASSERT_NOT_REACHED(); |
| return; |
| } |
| auto menuProvider = FrontendMenuProvider::create(this, { &state, frontendApiObject }, items); |
| m_menuProvider = menuProvider.ptr(); |
| m_frontendPage->contextMenuController().showContextMenu(*event, menuProvider); |
| } |
| |
| #endif |
| |
| void InspectorFrontendHost::dispatchEventAsContextMenuEvent(Event* event) |
| { |
| #if ENABLE(CONTEXT_MENUS) && USE(ACCESSIBILITY_CONTEXT_MENUS) |
| if (!is<MouseEvent>(event)) |
| return; |
| |
| Frame* frame = event->target()->toNode()->document().frame(); |
| MouseEvent& mouseEvent = downcast<MouseEvent>(*event); |
| IntPoint mousePoint = IntPoint(mouseEvent.clientX(), mouseEvent.clientY()); |
| |
| m_frontendPage->contextMenuController().showContextMenuAt(*frame, mousePoint); |
| #else |
| UNUSED_PARAM(event); |
| #endif |
| } |
| |
| bool InspectorFrontendHost::isUnderTest() |
| { |
| return m_client && m_client->isUnderTest(); |
| } |
| |
| void InspectorFrontendHost::unbufferedLog(const String& message) |
| { |
| // This is used only for debugging inspector tests. |
| WTFLogAlways("%s", message.utf8().data()); |
| } |
| |
| void InspectorFrontendHost::beep() |
| { |
| systemBeep(); |
| } |
| |
| } // namespace WebCore |