blob: 6f0d3bf04bd51b1a334e52cd85fda883aba283b7 [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 "config.h"
#include "core/inspector/InspectorFrontendHost.h"
#include "bindings/v8/ScriptFunctionCall.h"
#include "core/dom/UserGestureIndicator.h"
#include "core/inspector/InspectorController.h"
#include "core/inspector/InspectorFrontendClient.h"
#include "core/loader/FrameLoader.h"
#include "core/loader/TextResourceDecoder.h"
#include "core/page/ContextMenuController.h"
#include "core/page/ContextMenuProvider.h"
#include "core/page/Frame.h"
#include "core/page/Page.h"
#include "core/platform/ContextMenu.h"
#include "core/platform/ContextMenuItem.h"
#include "core/platform/Pasteboard.h"
#include "core/platform/network/ResourceError.h"
#include "core/platform/network/ResourceRequest.h"
#include "core/platform/network/ResourceResponse.h"
#include "core/rendering/RenderTheme.h"
#include "modules/filesystem/DOMFileSystem.h"
namespace WebCore {
class FrontendMenuProvider : public ContextMenuProvider {
public:
static PassRefPtr<FrontendMenuProvider> create(InspectorFrontendHost* frontendHost, ScriptObject frontendApiObject, const Vector<ContextMenuItem>& items)
{
return adoptRef(new FrontendMenuProvider(frontendHost, frontendApiObject, items));
}
void disconnect()
{
m_frontendApiObject = ScriptObject();
m_frontendHost = 0;
}
private:
FrontendMenuProvider(InspectorFrontendHost* frontendHost, ScriptObject frontendApiObject, const Vector<ContextMenuItem>& items)
: m_frontendHost(frontendHost)
, m_frontendApiObject(frontendApiObject)
, m_items(items)
{
}
virtual ~FrontendMenuProvider()
{
contextMenuCleared();
}
virtual void populateContextMenu(ContextMenu* menu)
{
for (size_t i = 0; i < m_items.size(); ++i)
menu->appendItem(m_items[i]);
}
virtual void contextMenuItemSelected(const ContextMenuItem* item)
{
if (m_frontendHost) {
UserGestureIndicator gestureIndicator(DefinitelyProcessingNewUserGesture);
int itemNumber = item->action() - ContextMenuItemBaseCustomTag;
ScriptFunctionCall function(m_frontendApiObject, "contextMenuItemSelected");
function.appendArgument(itemNumber);
function.call();
}
}
virtual void contextMenuCleared()
{
if (m_frontendHost) {
ScriptFunctionCall function(m_frontendApiObject, "contextMenuCleared");
function.call();
m_frontendHost->m_menuProvider = 0;
}
m_items.clear();
}
InspectorFrontendHost* m_frontendHost;
ScriptObject m_frontendApiObject;
Vector<ContextMenuItem> m_items;
};
InspectorFrontendHost::InspectorFrontendHost(InspectorFrontendClient* client, Page* frontendPage)
: m_client(client)
, m_frontendPage(frontendPage)
, m_menuProvider(0)
{
ScriptWrappable::init(this);
}
InspectorFrontendHost::~InspectorFrontendHost()
{
ASSERT(!m_client);
}
void InspectorFrontendHost::disconnectClient()
{
m_client = 0;
if (m_menuProvider)
m_menuProvider->disconnect();
m_frontendPage = 0;
}
void InspectorFrontendHost::loaded()
{
}
void InspectorFrontendHost::requestSetDockSide(const String& side)
{
if (!m_client)
return;
if (side == "undocked")
m_client->requestSetDockSide(InspectorFrontendClient::Undocked);
else if (side == "right")
m_client->requestSetDockSide(InspectorFrontendClient::DockedToRight);
else if (side == "bottom")
m_client->requestSetDockSide(InspectorFrontendClient::DockedToBottom);
}
void InspectorFrontendHost::closeWindow()
{
if (m_client) {
m_client->closeWindow();
disconnectClient(); // Disconnect from client.
}
}
void InspectorFrontendHost::bringToFront()
{
if (m_client)
m_client->bringToFront();
}
void InspectorFrontendHost::setZoomFactor(float zoom)
{
m_frontendPage->mainFrame()->setPageAndTextZoomFactors(zoom, 1);
}
void InspectorFrontendHost::inspectedURLChanged(const String& newURL)
{
if (m_client)
m_client->inspectedURLChanged(newURL);
}
void InspectorFrontendHost::setAttachedWindowHeight(unsigned height)
{
if (m_client)
m_client->changeAttachedWindowHeight(height);
}
void InspectorFrontendHost::moveWindowBy(float x, float y) const
{
if (m_client)
m_client->moveWindowBy(x, y);
}
void InspectorFrontendHost::setInjectedScriptForOrigin(const String& origin, const String& script)
{
ASSERT(m_frontendPage->inspectorController());
m_frontendPage->inspectorController()->setInjectedScriptForOrigin(origin, script);
}
String InspectorFrontendHost::localizedStringsURL()
{
return "";
}
void InspectorFrontendHost::copyText(const String& text)
{
Pasteboard::generalPasteboard()->writePlainText(text, Pasteboard::CannotSmartReplace);
}
void InspectorFrontendHost::openInNewTab(const String& url)
{
if (m_client)
m_client->openInNewTab(url);
}
bool InspectorFrontendHost::canSave()
{
return true;
}
void InspectorFrontendHost::save(const String& url, const String& content, bool forceSaveAs)
{
if (m_client)
m_client->save(url, content, 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);
}
void InspectorFrontendHost::showContextMenu(Event* event, const Vector<ContextMenuItem>& items)
{
if (!event)
return;
ASSERT(m_frontendPage);
ScriptState* frontendScriptState = mainWorldScriptState(m_frontendPage->mainFrame());
ScriptObject frontendApiObject;
if (!ScriptGlobalObject::get(frontendScriptState, "InspectorFrontendAPI", frontendApiObject)) {
ASSERT_NOT_REACHED();
return;
}
RefPtr<FrontendMenuProvider> menuProvider = FrontendMenuProvider::create(this, frontendApiObject, items);
m_frontendPage->contextMenuController().showContextMenu(event, menuProvider);
m_menuProvider = menuProvider.get();
}
String InspectorFrontendHost::loadResourceSynchronously(const String& url)
{
ResourceRequest request(url);
request.setHTTPMethod("GET");
Vector<char> data;
ResourceError error;
ResourceResponse response;
m_frontendPage->mainFrame()->loader()->loadResourceSynchronously(request, DoNotAllowStoredCredentials, error, response, data);
WTF::TextEncoding textEncoding(response.textEncodingName());
bool useDetector = false;
if (!textEncoding.isValid()) {
textEncoding = UTF8Encoding();
useDetector = true;
}
RefPtr<TextResourceDecoder> decoder = TextResourceDecoder::create("text/plain", textEncoding, useDetector);
return decoder->decode(data.data(), data.size()) + decoder->flush();
}
String InspectorFrontendHost::getSelectionBackgroundColor()
{
Color color = m_frontendPage->theme()->activeSelectionBackgroundColor();
return color != Color::transparent ? color.serialized() : "";
}
String InspectorFrontendHost::getSelectionForegroundColor()
{
Color color = m_frontendPage->theme()->activeSelectionForegroundColor();
return color != Color::transparent ? color.serialized() : "";
}
bool InspectorFrontendHost::supportsFileSystems()
{
return true;
}
void InspectorFrontendHost::requestFileSystems()
{
if (m_client)
m_client->requestFileSystems();
}
void InspectorFrontendHost::addFileSystem()
{
if (m_client)
m_client->addFileSystem();
}
void InspectorFrontendHost::removeFileSystem(const String& fileSystemPath)
{
if (m_client)
m_client->removeFileSystem(fileSystemPath);
}
PassRefPtr<DOMFileSystem> InspectorFrontendHost::isolatedFileSystem(const String& fileSystemName, const String& rootURL)
{
ScriptExecutionContext* context = m_frontendPage->mainFrame()->document();
return DOMFileSystem::create(context, fileSystemName, FileSystemTypeIsolated, KURL(ParsedURLString, rootURL), AsyncFileSystem::create());
}
void InspectorFrontendHost::indexPath(int requestId, const String& fileSystemPath)
{
if (m_client)
m_client->indexPath(requestId, fileSystemPath);
}
void InspectorFrontendHost::stopIndexing(int requestId)
{
if (m_client)
m_client->stopIndexing(requestId);
}
void InspectorFrontendHost::searchInPath(int requestId, const String& fileSystemPath, const String& query)
{
if (m_client)
m_client->searchInPath(requestId, fileSystemPath, query);
}
bool InspectorFrontendHost::isUnderTest()
{
return m_client && m_client->isUnderTest();
}
bool InspectorFrontendHost::canSaveAs()
{
return false;
}
bool InspectorFrontendHost::canInspectWorkers()
{
return false;
}
String InspectorFrontendHost::hiddenPanels()
{
return "";
}
} // namespace WebCore