blob: 572d235c6d1f14dce21228919353b7344a903ce2 [file] [log] [blame]
/*
* Copyright (C) 2010-2020 Apple Inc. All rights reserved.
* Portions Copyright (c) 2011 Motorola Mobility, Inc. All rights reserved.
*
* 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.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. 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 INC. 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 "WebInspectorUIProxy.h"
#include "APIInspectorClient.h"
#include "APINavigation.h"
#include "APIProcessPoolConfiguration.h"
#include "APIUIClient.h"
#include "InspectorBrowserAgent.h"
#include "WebAutomationSession.h"
#include "WebFrameProxy.h"
#include "WebInspectorInterruptDispatcherMessages.h"
#include "WebInspectorMessages.h"
#include "WebInspectorUIExtensionControllerProxy.h"
#include "WebInspectorUIMessages.h"
#include "WebInspectorUIProxyMessages.h"
#include "WebPageGroup.h"
#include "WebPageInspectorController.h"
#include "WebPageProxy.h"
#include "WebPreferences.h"
#include "WebProcessPool.h"
#include "WebProcessProxy.h"
#include <WebCore/CertificateInfo.h>
#include <WebCore/MockRealtimeMediaSourceCenter.h>
#include <WebCore/NotImplemented.h>
#include <WebCore/TextEncoding.h>
#include <wtf/SetForScope.h>
#if PLATFORM(GTK)
#include "WebInspectorUIProxyClient.h"
#endif
namespace WebKit {
using namespace WebCore;
const unsigned WebInspectorUIProxy::minimumWindowWidth = 500;
const unsigned WebInspectorUIProxy::minimumWindowHeight = 400;
const unsigned WebInspectorUIProxy::initialWindowWidth = 1000;
const unsigned WebInspectorUIProxy::initialWindowHeight = 650;
WebInspectorUIProxy::WebInspectorUIProxy(WebPageProxy& inspectedPage)
: m_inspectedPage(&inspectedPage)
, m_inspectorClient(makeUnique<API::InspectorClient>())
#if PLATFORM(MAC)
, m_closeFrontendAfterInactivityTimer(RunLoop::main(), this, &WebInspectorUIProxy::closeFrontendAfterInactivityTimerFired)
#endif
{
m_inspectedPage->process().addMessageReceiver(Messages::WebInspectorUIProxy::messageReceiverName(), m_inspectedPage->webPageID(), *this);
}
WebInspectorUIProxy::~WebInspectorUIProxy()
{
}
void WebInspectorUIProxy::setInspectorClient(std::unique_ptr<API::InspectorClient>&& inspectorClient)
{
if (!inspectorClient) {
m_inspectorClient = nullptr;
return;
}
m_inspectorClient = WTFMove(inspectorClient);
}
unsigned WebInspectorUIProxy::inspectionLevel() const
{
return inspectorLevelForPage(inspectedPage());
}
WebPreferences& WebInspectorUIProxy::inspectorPagePreferences() const
{
ASSERT(m_inspectorPage);
return m_inspectorPage->pageGroup().preferences();
}
void WebInspectorUIProxy::invalidate()
{
closeFrontendPageAndWindow();
platformInvalidate();
reset();
}
void WebInspectorUIProxy::sendMessageToFrontend(const String& message)
{
if (!m_inspectorPage)
return;
m_inspectorPage->send(Messages::WebInspectorUI::SendMessageToFrontend(message));
}
// Public APIs
bool WebInspectorUIProxy::isFront()
{
if (!m_inspectedPage)
return false;
return platformIsFront();
}
void WebInspectorUIProxy::connect()
{
if (!m_inspectedPage)
return;
if (!m_inspectedPage->preferences().developerExtrasEnabled())
return;
if (m_showMessageSent)
return;
m_showMessageSent = true;
m_ignoreFirstBringToFront = true;
createFrontendPage();
m_inspectedPage->send(Messages::WebInspectorInterruptDispatcher::NotifyNeedDebuggerBreak(), 0);
m_inspectedPage->send(Messages::WebInspector::Show());
}
void WebInspectorUIProxy::show()
{
if (!m_inspectedPage)
return;
if (isConnected()) {
bringToFront();
return;
}
connect();
// Don't ignore the first bringToFront so it opens the Inspector.
m_ignoreFirstBringToFront = false;
}
void WebInspectorUIProxy::hide()
{
if (!m_inspectedPage)
return;
m_isVisible = false;
platformHide();
}
void WebInspectorUIProxy::close()
{
if (!m_inspectedPage)
return;
m_inspectedPage->send(Messages::WebInspector::Close());
closeFrontendPageAndWindow();
}
void WebInspectorUIProxy::closeForCrash()
{
close();
platformDidCloseForCrash();
}
void WebInspectorUIProxy::reopen()
{
if (!m_inspectedPage)
return;
close();
show();
}
void WebInspectorUIProxy::resetState()
{
ASSERT(m_inspectedPage);
ASSERT(m_inspectorPage);
if (!m_inspectedPage || !m_inspectorPage)
return;
inspectorPagePreferences().deleteInspectorAttachedHeight();
inspectorPagePreferences().deleteInspectorAttachedWidth();
inspectorPagePreferences().deleteInspectorAttachmentSide();
inspectorPagePreferences().deleteInspectorStartsAttached();
inspectorPagePreferences().deleteInspectorWindowFrame();
platformResetState();
}
void WebInspectorUIProxy::reset()
{
if (m_inspectedPage) {
m_inspectedPage->process().removeMessageReceiver(Messages::WebInspectorUIProxy::messageReceiverName(), m_inspectedPage->webPageID());
m_inspectedPage = nullptr;
}
}
void WebInspectorUIProxy::updateForNewPageProcess(WebPageProxy& inspectedPage)
{
ASSERT(!m_inspectedPage);
m_inspectedPage = &inspectedPage;
m_inspectedPage->process().addMessageReceiver(Messages::WebInspectorUIProxy::messageReceiverName(), m_inspectedPage->webPageID(), *this);
if (m_inspectorPage)
m_inspectorPage->send(Messages::WebInspectorUI::UpdateConnection());
}
void WebInspectorUIProxy::setFrontendConnection(IPC::Attachment connectionIdentifier)
{
if (!m_inspectedPage)
return;
m_inspectedPage->send(Messages::WebInspector::SetFrontendConnection(connectionIdentifier));
}
void WebInspectorUIProxy::showConsole()
{
if (!m_inspectedPage)
return;
createFrontendPage();
m_inspectedPage->send(Messages::WebInspector::ShowConsole());
}
void WebInspectorUIProxy::showResources()
{
if (!m_inspectedPage)
return;
createFrontendPage();
m_inspectedPage->send(Messages::WebInspector::ShowResources());
}
void WebInspectorUIProxy::showMainResourceForFrame(WebFrameProxy* frame)
{
if (!m_inspectedPage)
return;
createFrontendPage();
m_inspectedPage->send(Messages::WebInspector::ShowMainResourceForFrame(frame->frameID()));
}
void WebInspectorUIProxy::attachBottom()
{
attach(AttachmentSide::Bottom);
}
void WebInspectorUIProxy::attachRight()
{
attach(AttachmentSide::Right);
}
void WebInspectorUIProxy::attachLeft()
{
attach(AttachmentSide::Left);
}
void WebInspectorUIProxy::attach(AttachmentSide side)
{
ASSERT(m_inspectorPage);
if (!m_inspectedPage || !m_inspectorPage || (!m_isAttached && !platformCanAttach(m_canAttach)))
return;
m_isAttached = true;
m_attachmentSide = side;
inspectorPagePreferences().setInspectorAttachmentSide(static_cast<uint32_t>(side));
if (m_isVisible)
inspectorPagePreferences().setInspectorStartsAttached(true);
m_inspectedPage->send(Messages::WebInspector::SetAttached(true));
switch (m_attachmentSide) {
case AttachmentSide::Bottom:
m_inspectorPage->send(Messages::WebInspectorUI::AttachedBottom());
break;
case AttachmentSide::Right:
m_inspectorPage->send(Messages::WebInspectorUI::AttachedRight());
break;
case AttachmentSide::Left:
m_inspectorPage->send(Messages::WebInspectorUI::AttachedLeft());
break;
}
platformAttach();
}
void WebInspectorUIProxy::detach()
{
if (!m_inspectedPage || !m_inspectorPage || (!m_isAttached && m_isVisible))
return;
m_isAttached = false;
if (m_isVisible)
inspectorPagePreferences().setInspectorStartsAttached(false);
m_inspectedPage->send(Messages::WebInspector::SetAttached(false));
m_inspectorPage->send(Messages::WebInspectorUI::Detached());
platformDetach();
}
void WebInspectorUIProxy::setAttachedWindowHeight(unsigned height)
{
ASSERT(m_inspectorPage);
if (!m_inspectorPage)
return;
inspectorPagePreferences().setInspectorAttachedHeight(height);
platformSetAttachedWindowHeight(height);
}
void WebInspectorUIProxy::setAttachedWindowWidth(unsigned width)
{
ASSERT(m_inspectorPage);
if (!m_inspectorPage)
return;
inspectorPagePreferences().setInspectorAttachedWidth(width);
platformSetAttachedWindowWidth(width);
}
void WebInspectorUIProxy::setSheetRect(const FloatRect& rect)
{
platformSetSheetRect(rect);
}
void WebInspectorUIProxy::startWindowDrag()
{
platformStartWindowDrag();
}
void WebInspectorUIProxy::togglePageProfiling()
{
if (!m_inspectedPage)
return;
if (m_isProfilingPage)
m_inspectedPage->send(Messages::WebInspector::StopPageProfiling());
else
m_inspectedPage->send(Messages::WebInspector::StartPageProfiling());
}
void WebInspectorUIProxy::toggleElementSelection()
{
if (!m_inspectedPage)
return;
if (m_elementSelectionActive) {
m_ignoreElementSelectionChange = true;
m_inspectedPage->send(Messages::WebInspector::StopElementSelection());
} else {
connect();
m_inspectedPage->send(Messages::WebInspector::StartElementSelection());
}
}
bool WebInspectorUIProxy::isMainOrTestInspectorPage(const URL& url)
{
// Use URL so we can compare the paths and protocols.
URL mainPageURL(URL(), WebInspectorUIProxy::inspectorPageURL());
if (url.protocol() == mainPageURL.protocol() && decodeURLEscapeSequences(url.path()) == decodeURLEscapeSequences(mainPageURL.path()))
return true;
// We might not have a Test URL in Production builds.
String testPageURLString = WebInspectorUIProxy::inspectorTestPageURL();
if (testPageURLString.isNull())
return false;
URL testPageURL(URL(), testPageURLString);
return url.protocol() == testPageURL.protocol() && decodeURLEscapeSequences(url.path()) == decodeURLEscapeSequences(testPageURL.path());
}
void WebInspectorUIProxy::createFrontendPage()
{
if (m_inspectorPage)
return;
m_inspectorPage = platformCreateFrontendPage();
ASSERT(m_inspectorPage);
if (!m_inspectorPage)
return;
trackInspectorPage(m_inspectorPage, m_inspectedPage);
// Make sure the inspected page has a running WebProcess so we can inspect it.
m_inspectedPage->launchInitialProcessIfNecessary();
m_inspectorPage->process().addMessageReceiver(Messages::WebInspectorUIProxy::messageReceiverName(), m_inspectedPage->identifier(), *this);
#if ENABLE(INSPECTOR_EXTENSIONS)
m_extensionController = WebInspectorUIExtensionControllerProxy::create(*m_inspectorPage);
#endif
}
void WebInspectorUIProxy::openLocalInspectorFrontend(bool canAttach, bool underTest)
{
if (!m_inspectedPage)
return;
if (!m_inspectedPage->preferences().developerExtrasEnabled())
return;
if (m_inspectedPage->inspectorController().hasLocalFrontend()) {
show();
return;
}
m_underTest = underTest;
createFrontendPage();
ASSERT(m_inspectorPage);
if (!m_inspectorPage)
return;
m_inspectorPage->send(Messages::WebInspectorUI::EstablishConnection(m_inspectedPage->identifier(), infoForLocalDebuggable(), m_underTest, inspectionLevel()));
ASSERT(!m_isActiveFrontend);
m_isActiveFrontend = true;
m_inspectedPage->inspectorController().connectFrontend(*this);
if (!m_underTest) {
m_canAttach = platformCanAttach(canAttach);
m_isAttached = shouldOpenAttached();
m_attachmentSide = static_cast<AttachmentSide>(inspectorPagePreferences().inspectorAttachmentSide());
m_inspectedPage->send(Messages::WebInspector::SetAttached(m_isAttached));
if (m_isAttached) {
switch (m_attachmentSide) {
case AttachmentSide::Bottom:
m_inspectorPage->send(Messages::WebInspectorUI::AttachedBottom());
break;
case AttachmentSide::Right:
m_inspectorPage->send(Messages::WebInspectorUI::AttachedRight());
break;
case AttachmentSide::Left:
m_inspectorPage->send(Messages::WebInspectorUI::AttachedLeft());
break;
}
} else
m_inspectorPage->send(Messages::WebInspectorUI::Detached());
m_inspectorPage->send(Messages::WebInspectorUI::SetDockingUnavailable(!m_canAttach));
}
// Notify WebKit client when a local inspector attaches so that it may install delegates prior to the _WKInspector loading its frontend.
m_inspectedPage->uiClient().didAttachLocalInspector(*m_inspectedPage, *this);
// Bail out if the client closed the inspector from the delegate method.
if (!m_inspectorPage)
return;
m_inspectorPage->loadRequest(URL(URL(), m_underTest ? WebInspectorUIProxy::inspectorTestPageURL() : WebInspectorUIProxy::inspectorPageURL()));
}
void WebInspectorUIProxy::open()
{
if (m_underTest)
return;
if (!m_inspectorPage)
return;
#if PLATFORM(GTK)
SetForScope<bool> isOpening(m_isOpening, true);
#endif
m_isVisible = true;
m_inspectorPage->send(Messages::WebInspectorUI::SetIsVisible(m_isVisible));
if (m_isAttached)
platformAttach();
else
platformCreateFrontendWindow();
platformBringToFront();
}
void WebInspectorUIProxy::didClose()
{
closeFrontendPageAndWindow();
}
void WebInspectorUIProxy::closeFrontendPageAndWindow()
{
if (!m_inspectorPage)
return;
// Guard against calls to close() made by the client while already closing.
if (m_closing)
return;
SetForScope<bool> reentrancyProtector(m_closing, true);
// Notify WebKit client when a local inspector closes so it can clear _WKInspectorDelegate and perform other cleanup.
m_inspectedPage->uiClient().willCloseLocalInspector(*m_inspectedPage, *this);
m_isVisible = false;
m_isProfilingPage = false;
m_showMessageSent = false;
m_ignoreFirstBringToFront = false;
untrackInspectorPage(m_inspectorPage);
m_inspectorPage->send(Messages::WebInspectorUI::SetIsVisible(m_isVisible));
m_inspectorPage->process().removeMessageReceiver(Messages::WebInspectorUIProxy::messageReceiverName(), m_inspectedPage->identifier());
if (m_isActiveFrontend) {
m_isActiveFrontend = false;
m_inspectedPage->inspectorController().disconnectFrontend(*this);
}
if (m_isAttached)
platformDetach();
#if ENABLE(INSPECTOR_EXTENSIONS)
// This extension controller may be kept alive by the IPC dispatcher beyond the point
// when m_inspectorPage is cleared below. Notify the controller so it can clean up before then.
m_extensionController->inspectorFrontendWillClose();
m_extensionController = nullptr;
#endif
// Null out m_inspectorPage after platformDetach(), so the views can be cleaned up correctly.
m_inspectorPage = nullptr;
m_isAttached = false;
m_canAttach = false;
m_underTest = false;
platformCloseFrontendPageAndWindow();
}
void WebInspectorUIProxy::sendMessageToBackend(const String& message)
{
if (!m_inspectedPage)
return;
m_inspectedPage->inspectorController().dispatchMessageFromFrontend(message);
}
void WebInspectorUIProxy::frontendLoaded()
{
if (!m_inspectedPage)
return;
if (auto* automationSession = m_inspectedPage->process().processPool().automationSession())
automationSession->inspectorFrontendLoaded(*m_inspectedPage);
#if ENABLE(INSPECTOR_EXTENSIONS)
if (m_extensionController)
m_extensionController->inspectorFrontendLoaded();
#endif
if (m_inspectorClient)
m_inspectorClient->frontendLoaded(*this);
}
void WebInspectorUIProxy::bringToFront()
{
// WebCore::InspectorFrontendClientLocal tells us to do this on load. We want to
// ignore it once if we only wanted to connect. This allows the Inspector to later
// request to be brought to the front when a breakpoint is hit or some other action.
if (m_ignoreFirstBringToFront) {
m_ignoreFirstBringToFront = false;
return;
}
if (m_isVisible)
platformBringToFront();
else
open();
}
void WebInspectorUIProxy::bringInspectedPageToFront()
{
platformBringInspectedPageToFront();
}
void WebInspectorUIProxy::attachAvailabilityChanged(bool available)
{
bool previousCanAttach = m_canAttach;
m_canAttach = m_isAttached || platformCanAttach(available);
if (previousCanAttach == m_canAttach)
return;
if (m_inspectorPage && !m_underTest)
m_inspectorPage->send(Messages::WebInspectorUI::SetDockingUnavailable(!m_canAttach));
platformAttachAvailabilityChanged(m_canAttach);
}
void WebInspectorUIProxy::setForcedAppearance(InspectorFrontendClient::Appearance appearance)
{
platformSetForcedAppearance(appearance);
}
void WebInspectorUIProxy::openURLExternally(const String& url)
{
if (m_inspectorClient)
m_inspectorClient->openURLExternally(*this, url);
}
void WebInspectorUIProxy::inspectedURLChanged(const String& urlString)
{
platformInspectedURLChanged(urlString);
}
void WebInspectorUIProxy::showCertificate(const CertificateInfo& certificateInfo)
{
platformShowCertificate(certificateInfo);
}
void WebInspectorUIProxy::elementSelectionChanged(bool active)
{
m_elementSelectionActive = active;
if (m_ignoreElementSelectionChange) {
m_ignoreElementSelectionChange = false;
if (!m_isVisible)
close();
return;
}
if (active)
platformBringInspectedPageToFront();
else if (isConnected())
bringToFront();
}
void WebInspectorUIProxy::timelineRecordingChanged(bool active)
{
m_isProfilingPage = active;
}
void WebInspectorUIProxy::setDeveloperPreferenceOverride(WebCore::InspectorClient::DeveloperPreference developerPreference, std::optional<bool> overrideValue)
{
switch (developerPreference) {
case InspectorClient::DeveloperPreference::PrivateClickMeasurementDebugModeEnabled:
if (m_inspectedPage)
m_inspectedPage->websiteDataStore().setPrivateClickMeasurementDebugMode(overrideValue && overrideValue.value());
return;
case InspectorClient::DeveloperPreference::ITPDebugModeEnabled:
if (m_inspectedPage)
m_inspectedPage->websiteDataStore().setResourceLoadStatisticsDebugMode(overrideValue && overrideValue.value());
return;
case InspectorClient::DeveloperPreference::MockCaptureDevicesEnabled:
#if ENABLE(MEDIA_STREAM)
if (m_inspectedPage)
m_inspectedPage->setMockCaptureDevicesEnabledOverride(overrideValue);
#endif // ENABLE(MEDIA_STREAM)
return;
}
ASSERT_NOT_REACHED();
}
void WebInspectorUIProxy::setDiagnosticLoggingAvailable(bool available)
{
#if ENABLE(INSPECTOR_TELEMETRY)
m_inspectorPage->send(Messages::WebInspectorUI::SetDiagnosticLoggingAvailable(available));
#else
UNUSED_PARAM(available);
#endif
}
void WebInspectorUIProxy::save(const String& filename, const String& content, bool base64Encoded, bool forceSaveAs)
{
if (!m_inspectedPage->preferences().developerExtrasEnabled())
return;
ASSERT(!filename.isEmpty());
if (filename.isEmpty())
return;
platformSave(filename, content, base64Encoded, forceSaveAs);
}
void WebInspectorUIProxy::append(const String& filename, const String& content)
{
if (!m_inspectedPage->preferences().developerExtrasEnabled())
return;
ASSERT(!filename.isEmpty());
if (filename.isEmpty())
return;
platformAppend(filename, content);
}
bool WebInspectorUIProxy::shouldOpenAttached()
{
return inspectorPagePreferences().inspectorStartsAttached() && canAttach();
}
void WebInspectorUIProxy::evaluateInFrontendForTesting(const String& expression)
{
if (!m_inspectorPage)
return;
m_inspectorPage->send(Messages::WebInspectorUI::EvaluateInFrontendForTesting(expression));
}
// Unsupported configurations can use the stubs provided here.
#if !PLATFORM(MAC) && !PLATFORM(GTK) && !PLATFORM(WIN)
WebPageProxy* WebInspectorUIProxy::platformCreateFrontendPage()
{
notImplemented();
return nullptr;
}
void WebInspectorUIProxy::platformCreateFrontendWindow()
{
notImplemented();
}
void WebInspectorUIProxy::platformCloseFrontendPageAndWindow()
{
notImplemented();
}
void WebInspectorUIProxy::platformDidCloseForCrash()
{
notImplemented();
}
void WebInspectorUIProxy::platformInvalidate()
{
notImplemented();
}
void WebInspectorUIProxy::platformResetState()
{
notImplemented();
}
void WebInspectorUIProxy::platformBringToFront()
{
notImplemented();
}
void WebInspectorUIProxy::platformBringInspectedPageToFront()
{
notImplemented();
}
void WebInspectorUIProxy::platformHide()
{
notImplemented();
}
bool WebInspectorUIProxy::platformIsFront()
{
notImplemented();
return false;
}
void WebInspectorUIProxy::platformSetForcedAppearance(InspectorFrontendClient::Appearance)
{
notImplemented();
}
void WebInspectorUIProxy::platformInspectedURLChanged(const String&)
{
notImplemented();
}
void WebInspectorUIProxy::platformShowCertificate(const CertificateInfo&)
{
notImplemented();
}
void WebInspectorUIProxy::platformSave(const String& suggestedURL, const String& content, bool base64Encoded, bool forceSaveDialog)
{
notImplemented();
}
void WebInspectorUIProxy::platformAppend(const String& suggestedURL, const String& content)
{
notImplemented();
}
unsigned WebInspectorUIProxy::platformInspectedWindowHeight()
{
notImplemented();
return 0;
}
unsigned WebInspectorUIProxy::platformInspectedWindowWidth()
{
notImplemented();
return 0;
}
void WebInspectorUIProxy::platformAttach()
{
notImplemented();
}
void WebInspectorUIProxy::platformDetach()
{
notImplemented();
}
void WebInspectorUIProxy::platformSetAttachedWindowHeight(unsigned)
{
notImplemented();
}
void WebInspectorUIProxy::platformSetSheetRect(const FloatRect&)
{
notImplemented();
}
void WebInspectorUIProxy::platformStartWindowDrag()
{
notImplemented();
}
String WebInspectorUIProxy::inspectorPageURL()
{
notImplemented();
return String();
}
String WebInspectorUIProxy::inspectorTestPageURL()
{
notImplemented();
return String();
}
DebuggableInfoData WebInspectorUIProxy::infoForLocalDebuggable()
{
notImplemented();
return DebuggableInfoData::empty();
}
void WebInspectorUIProxy::platformSetAttachedWindowWidth(unsigned)
{
notImplemented();
}
void WebInspectorUIProxy::platformAttachAvailabilityChanged(bool)
{
notImplemented();
}
#endif // !PLATFORM(MAC) && !PLATFORM(GTK) && !PLATFORM(WIN)
} // namespace WebKit