/* * 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 "config.h" #include "InspectorBackend.h" #if ENABLE(INSPECTOR) #if ENABLE(DATABASE) #include "Database.h" #endif #include "Element.h" #include "Frame.h" #include "FrameLoader.h" #include "HTMLFrameOwnerElement.h" #include "InjectedScript.h" #include "InjectedScriptHost.h" #include "InspectorClient.h" #include "InspectorController.h" #include "InspectorDOMAgent.h" #include "InspectorFrontend.h" #include "InspectorResource.h" #include "Pasteboard.h" #include "ScriptArray.h" #include "SerializedScriptValue.h" #if ENABLE(DOM_STORAGE) #include "Storage.h" #endif #if ENABLE(JAVASCRIPT_DEBUGGER) && USE(JSC) #include "JavaScriptCallFrame.h" #include "JavaScriptDebugServer.h" using namespace JSC; #endif #include "markup.h" #include <wtf/RefPtr.h> #include <wtf/StdLibExtras.h> using namespace std; namespace WebCore { InspectorBackend::InspectorBackend(InspectorController* inspectorController) : m_inspectorController(inspectorController) { } InspectorBackend::~InspectorBackend() { } void InspectorBackend::saveFrontendSettings(const String& settings) { if (m_inspectorController) m_inspectorController->setSetting(InspectorController::FrontendSettingsSettingName, settings); } void InspectorBackend::storeLastActivePanel(const String& panelName) { if (m_inspectorController) m_inspectorController->storeLastActivePanel(panelName); } void InspectorBackend::toggleNodeSearch() { if (m_inspectorController) m_inspectorController->toggleSearchForNodeInPage(); } bool InspectorBackend::searchingForNode() { if (m_inspectorController) return m_inspectorController->searchingForNodeInPage(); return false; } bool InspectorBackend::resourceTrackingEnabled() const { if (m_inspectorController) return m_inspectorController->resourceTrackingEnabled(); return false; } void InspectorBackend::enableResourceTracking(bool always) { if (m_inspectorController) m_inspectorController->enableResourceTracking(always); } void InspectorBackend::disableResourceTracking(bool always) { if (m_inspectorController) m_inspectorController->disableResourceTracking(always); } void InspectorBackend::getResourceContent(long callId, unsigned long identifier) { InspectorFrontend* frontend = inspectorFrontend(); if (!frontend) return; RefPtr<InspectorResource> resource = m_inspectorController->resources().get(identifier); if (resource) frontend->didGetResourceContent(callId, resource->sourceString()); else frontend->didGetResourceContent(callId, ""); } void InspectorBackend::startTimelineProfiler() { if (m_inspectorController) m_inspectorController->startTimelineProfiler(); } void InspectorBackend::stopTimelineProfiler() { if (m_inspectorController) m_inspectorController->stopTimelineProfiler(); } #if ENABLE(JAVASCRIPT_DEBUGGER) && USE(JSC) bool InspectorBackend::debuggerEnabled() const { if (m_inspectorController) return m_inspectorController->debuggerEnabled(); return false; } void InspectorBackend::enableDebugger(bool always) { if (m_inspectorController) m_inspectorController->enableDebuggerFromFrontend(always); } void InspectorBackend::disableDebugger(bool always) { if (m_inspectorController) m_inspectorController->disableDebugger(always); } void InspectorBackend::addBreakpoint(const String& sourceID, unsigned lineNumber, const String& condition) { intptr_t sourceIDValue = sourceID.toIntPtr(); JavaScriptDebugServer::shared().addBreakpoint(sourceIDValue, lineNumber, condition); } void InspectorBackend::updateBreakpoint(const String& sourceID, unsigned lineNumber, const String& condition) { intptr_t sourceIDValue = sourceID.toIntPtr(); JavaScriptDebugServer::shared().updateBreakpoint(sourceIDValue, lineNumber, condition); } void InspectorBackend::removeBreakpoint(const String& sourceID, unsigned lineNumber) { intptr_t sourceIDValue = sourceID.toIntPtr(); JavaScriptDebugServer::shared().removeBreakpoint(sourceIDValue, lineNumber); } void InspectorBackend::pauseInDebugger() { JavaScriptDebugServer::shared().pauseProgram(); } void InspectorBackend::resumeDebugger() { if (m_inspectorController) m_inspectorController->resumeDebugger(); } void InspectorBackend::stepOverStatementInDebugger() { JavaScriptDebugServer::shared().stepOverStatement(); } void InspectorBackend::stepIntoStatementInDebugger() { JavaScriptDebugServer::shared().stepIntoStatement(); } void InspectorBackend::stepOutOfFunctionInDebugger() { JavaScriptDebugServer::shared().stepOutOfFunction(); } long InspectorBackend::pauseOnExceptionsState() { return JavaScriptDebugServer::shared().pauseOnExceptionsState(); } void InspectorBackend::setPauseOnExceptionsState(long pauseState) { JavaScriptDebugServer::shared().setPauseOnExceptionsState(static_cast<JavaScriptDebugServer::PauseOnExceptionsState>(pauseState)); } JavaScriptCallFrame* InspectorBackend::currentCallFrame() const { return JavaScriptDebugServer::shared().currentCallFrame(); } #endif #if ENABLE(JAVASCRIPT_DEBUGGER) bool InspectorBackend::profilerEnabled() { if (m_inspectorController) return m_inspectorController->profilerEnabled(); return false; } void InspectorBackend::enableProfiler(bool always) { if (m_inspectorController) m_inspectorController->enableProfiler(always); } void InspectorBackend::disableProfiler(bool always) { if (m_inspectorController) m_inspectorController->disableProfiler(always); } void InspectorBackend::startProfiling() { if (m_inspectorController) m_inspectorController->startUserInitiatedProfiling(); } void InspectorBackend::stopProfiling() { if (m_inspectorController) m_inspectorController->stopUserInitiatedProfiling(); } void InspectorBackend::getProfileHeaders(long callId) { if (m_inspectorController) m_inspectorController->getProfileHeaders(callId); } void InspectorBackend::getProfile(long callId, unsigned uid) { if (m_inspectorController) m_inspectorController->getProfile(callId, uid); } #endif void InspectorBackend::setInjectedScriptSource(const String& source) { if (m_inspectorController) m_inspectorController->injectedScriptHost()->setInjectedScriptSource(source); } void InspectorBackend::dispatchOnInjectedScript(long callId, long injectedScriptId, const String& methodName, const String& arguments, bool async) { InspectorFrontend* frontend = inspectorFrontend(); if (!frontend) return; // FIXME: explicitly pass injectedScriptId along with node id to the frontend. bool injectedScriptIdIsNodeId = injectedScriptId <= 0; InjectedScript injectedScript; if (injectedScriptIdIsNodeId) injectedScript = m_inspectorController->injectedScriptForNodeId(-injectedScriptId); else injectedScript = m_inspectorController->injectedScriptHost()->injectedScriptForId(injectedScriptId); if (injectedScript.hasNoValue()) return; RefPtr<SerializedScriptValue> result; bool hadException = false; injectedScript.dispatch(callId, methodName, arguments, async, &result, &hadException); if (async) return; // InjectedScript will return result asynchronously by means of ::reportDidDispatchOnInjectedScript. frontend->didDispatchOnInjectedScript(callId, result.get(), hadException); } void InspectorBackend::getChildNodes(long callId, long nodeId) { if (InspectorDOMAgent* domAgent = inspectorDOMAgent()) domAgent->getChildNodes(callId, nodeId); } void InspectorBackend::setAttribute(long callId, long elementId, const String& name, const String& value) { if (InspectorDOMAgent* domAgent = inspectorDOMAgent()) domAgent->setAttribute(callId, elementId, name, value); } void InspectorBackend::removeAttribute(long callId, long elementId, const String& name) { if (InspectorDOMAgent* domAgent = inspectorDOMAgent()) domAgent->removeAttribute(callId, elementId, name); } void InspectorBackend::setTextNodeValue(long callId, long nodeId, const String& value) { if (InspectorDOMAgent* domAgent = inspectorDOMAgent()) domAgent->setTextNodeValue(callId, nodeId, value); } void InspectorBackend::getEventListenersForNode(long callId, long nodeId) { if (InspectorDOMAgent* domAgent = inspectorDOMAgent()) domAgent->getEventListenersForNode(callId, nodeId); } void InspectorBackend::copyNode(long nodeId) { Node* node = nodeForId(nodeId); if (!node) return; String markup = createMarkup(node); Pasteboard::generalPasteboard()->writePlainText(markup); } void InspectorBackend::removeNode(long callId, long nodeId) { InspectorFrontend* frontend = inspectorFrontend(); if (!frontend) return; Node* node = nodeForId(nodeId); if (!node) { // Use -1 to denote an error condition. frontend->didRemoveNode(callId, -1); return; } Node* parentNode = node->parentNode(); if (!parentNode) { frontend->didRemoveNode(callId, -1); return; } ExceptionCode code; parentNode->removeChild(node, code); if (code) { frontend->didRemoveNode(callId, -1); return; } frontend->didRemoveNode(callId, nodeId); } void InspectorBackend::highlightDOMNode(long nodeId) { if (Node* node = nodeForId(nodeId)) m_inspectorController->highlight(node); } void InspectorBackend::hideDOMNodeHighlight() { if (m_inspectorController) m_inspectorController->hideHighlight(); } void InspectorBackend::getCookies(long callId) { if (!m_inspectorController) return; m_inspectorController->getCookies(callId); } void InspectorBackend::deleteCookie(const String& cookieName, const String& domain) { if (!m_inspectorController) return; m_inspectorController->deleteCookie(cookieName, domain); } void InspectorBackend::releaseWrapperObjectGroup(long injectedScriptId, const String& objectGroup) { if (!m_inspectorController) return; m_inspectorController->injectedScriptHost()->releaseWrapperObjectGroup(injectedScriptId, objectGroup); } void InspectorBackend::didEvaluateForTestInFrontend(long callId, const String& jsonResult) { if (m_inspectorController) m_inspectorController->didEvaluateForTestInFrontend(callId, jsonResult); } #if ENABLE(DATABASE) void InspectorBackend::getDatabaseTableNames(long callId, long databaseId) { if (InspectorFrontend* frontend = inspectorFrontend()) { ScriptArray result = frontend->newScriptArray(); Database* database = m_inspectorController->databaseForId(databaseId); if (database) { Vector<String> tableNames = database->tableNames(); unsigned length = tableNames.size(); for (unsigned i = 0; i < length; ++i) result.set(i, tableNames[i]); } frontend->didGetDatabaseTableNames(callId, result); } } #endif #if ENABLE(DOM_STORAGE) void InspectorBackend::getDOMStorageEntries(long callId, long storageId) { if (m_inspectorController) m_inspectorController->getDOMStorageEntries(callId, storageId); } void InspectorBackend::setDOMStorageItem(long callId, long storageId, const String& key, const String& value) { if (m_inspectorController) m_inspectorController->setDOMStorageItem(callId, storageId, key, value); } void InspectorBackend::removeDOMStorageItem(long callId, long storageId, const String& key) { if (m_inspectorController) m_inspectorController->removeDOMStorageItem(callId, storageId, key); } #endif InspectorDOMAgent* InspectorBackend::inspectorDOMAgent() { if (!m_inspectorController) return 0; return m_inspectorController->domAgent(); } InspectorFrontend* InspectorBackend::inspectorFrontend() { if (!m_inspectorController) return 0; return m_inspectorController->m_frontend.get(); } Node* InspectorBackend::nodeForId(long nodeId) { if (InspectorDOMAgent* domAgent = inspectorDOMAgent()) return domAgent->nodeForId(nodeId); return 0; } } // namespace WebCore #endif // ENABLE(INSPECTOR)