⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 pluginview.cpp

📁 linux下开源浏览器WebKit的源码,市面上的很多商用浏览器都是移植自WebKit
💻 CPP
📖 第 1 页 / 共 2 页
字号:
/* * Copyright (C) 2006, 2007, 2008 Apple Inc. All rights reserved. * Copyright (C) 2008 Collabora Ltd. 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 COMPUTER, INC. ``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 COMPUTER, INC. OR * 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 "PluginView.h"#include "Document.h"#include "DocumentLoader.h"#include "Element.h"#include "FrameLoader.h"#include "FrameTree.h"#include "Frame.h"#include "FrameView.h"#include "GraphicsContext.h"#include "Image.h"#include "HTMLNames.h"#include "HTMLPlugInElement.h"#include "JSDOMWindow.h"#include "KeyboardEvent.h"#include "MIMETypeRegistry.h"#include "MouseEvent.h"#include "NotImplemented.h"#include "Page.h"#include "FocusController.h"#include "PlatformMouseEvent.h"#if PLATFORM(WIN_OS) && !PLATFORM(WX) && ENABLE(NETSCAPE_PLUGIN_API)#include "PluginMessageThrottlerWin.h"#endif#include "PluginPackage.h"#include "JSDOMBinding.h"#include "ScriptController.h"#include "ScriptValue.h"#include "PluginDatabase.h"#include "PluginDebug.h"#include "PluginMainThreadScheduler.h"#include "PluginPackage.h"#include "RenderBox.h"#include "RenderObject.h"#include "c_instance.h"#include "npruntime_impl.h"#include "runtime_root.h"#include "Settings.h"#include "runtime.h"#include <runtime/JSLock.h>#include <runtime/JSValue.h>#include <wtf/ASCIICType.h>using JSC::ExecState;using JSC::JSLock;using JSC::JSObject;using JSC::JSValuePtr;using JSC::UString;using std::min;using namespace WTF;namespace WebCore {using namespace HTMLNames;static int s_callingPlugin;static String scriptStringIfJavaScriptURL(const KURL& url){    if (!url.protocolIs("javascript"))        return String();    // This returns an unescaped string    return decodeURLEscapeSequences(url.string().substring(11));}PluginView* PluginView::s_currentPluginView = 0;void PluginView::popPopupsStateTimerFired(Timer<PluginView>*){    popPopupsEnabledState();}IntRect PluginView::windowClipRect() const{    // Start by clipping to our bounds.    IntRect clipRect(m_windowRect);        // Take our element and get the clip rect from the enclosing layer and frame view.    RenderLayer* layer = m_element->renderer()->enclosingLayer();    FrameView* parentView = m_element->document()->view();    clipRect.intersect(parentView->windowClipRectForLayer(layer, true));    return clipRect;}void PluginView::setFrameRect(const IntRect& rect){    if (m_element->document()->printing())        return;    if (rect != frameRect())        Widget::setFrameRect(rect);    updatePluginWidget();#if PLATFORM(WIN_OS)    // On Windows, always call plugin to change geometry.    setNPWindowRect(rect);#elif XP_UNIX    // On Unix, only call plugin if it's full-page.    if (m_mode == NP_FULL)        setNPWindowRect(rect);#endif}void PluginView::frameRectsChanged(){    updatePluginWidget();}void PluginView::handleEvent(Event* event){    if (!m_plugin || m_isWindowed)        return;    if (event->isMouseEvent())        handleMouseEvent(static_cast<MouseEvent*>(event));    else if (event->isKeyboardEvent())        handleKeyboardEvent(static_cast<KeyboardEvent*>(event));}bool PluginView::start(){    if (m_isStarted)        return false;    PluginMainThreadScheduler::scheduler().registerPlugin(m_instance);    ASSERT(m_plugin);    ASSERT(m_plugin->pluginFuncs()->newp);    NPError npErr;    {        PluginView::setCurrentPluginView(this);        JSC::JSLock::DropAllLocks dropAllLocks(false);        setCallingPlugin(true);        npErr = m_plugin->pluginFuncs()->newp((NPMIMEType)m_mimeType.data(), m_instance, m_mode, m_paramCount, m_paramNames, m_paramValues, NULL);        setCallingPlugin(false);        LOG_NPERROR(npErr);        PluginView::setCurrentPluginView(0);    }    if (npErr != NPERR_NO_ERROR)        return false;    m_isStarted = true;    if (!m_url.isEmpty() && !m_loadManually) {        FrameLoadRequest frameLoadRequest;        frameLoadRequest.resourceRequest().setHTTPMethod("GET");        frameLoadRequest.resourceRequest().setURL(m_url);        load(frameLoadRequest, false, 0);    }    return true;}void PluginView::setCurrentPluginView(PluginView* pluginView){    s_currentPluginView = pluginView;}PluginView* PluginView::currentPluginView(){    return s_currentPluginView;}static char* createUTF8String(const String& str){    CString cstr = str.utf8();    char* result = reinterpret_cast<char*>(fastMalloc(cstr.length() + 1));    strncpy(result, cstr.data(), cstr.length() + 1);    return result;}static bool getString(ScriptController* proxy, JSValuePtr result, String& string){    if (!proxy || !result || result.isUndefined())        return false;    JSLock lock(false);    ExecState* exec = proxy->globalObject()->globalExec();    UString ustring = result.toString(exec);    exec->clearException();    string = ustring;    return true;}void PluginView::performRequest(PluginRequest* request){    // don't let a plugin start any loads if it is no longer part of a document that is being     // displayed unless the loads are in the same frame as the plugin.    const String& targetFrameName = request->frameLoadRequest().frameName();    if (m_parentFrame->loader()->documentLoader() != m_parentFrame->loader()->activeDocumentLoader() &&        (targetFrameName.isNull() || m_parentFrame->tree()->find(targetFrameName) != m_parentFrame))        return;    KURL requestURL = request->frameLoadRequest().resourceRequest().url();    String jsString = scriptStringIfJavaScriptURL(requestURL);    if (jsString.isNull()) {        // if this is not a targeted request, create a stream for it. otherwise,        // just pass it off to the loader        if (targetFrameName.isEmpty()) {            RefPtr<PluginStream> stream = PluginStream::create(this, m_parentFrame, request->frameLoadRequest().resourceRequest(), request->sendNotification(), request->notifyData(), plugin()->pluginFuncs(), instance(), m_plugin->quirks());            m_streams.add(stream);            stream->start();        } else {            m_parentFrame->loader()->load(request->frameLoadRequest().resourceRequest(), targetFrameName, false);                  // FIXME: <rdar://problem/4807469> This should be sent when the document has finished loading            if (request->sendNotification()) {                PluginView::setCurrentPluginView(this);                JSC::JSLock::DropAllLocks dropAllLocks(false);                setCallingPlugin(true);                m_plugin->pluginFuncs()->urlnotify(m_instance, requestURL.string().utf8().data(), NPRES_DONE, request->notifyData());                setCallingPlugin(false);                PluginView::setCurrentPluginView(0);            }        }        return;    }    // Targeted JavaScript requests are only allowed on the frame that contains the JavaScript plugin    // and this has been made sure in ::load.    ASSERT(targetFrameName.isEmpty() || m_parentFrame->tree()->find(targetFrameName) == m_parentFrame);        // Executing a script can cause the plugin view to be destroyed, so we keep a reference to the parent frame.    RefPtr<Frame> parentFrame = m_parentFrame;    JSValuePtr result = m_parentFrame->loader()->executeScript(jsString, request->shouldAllowPopups()).jsValue();    if (targetFrameName.isNull()) {        String resultString;        CString cstr;        if (getString(parentFrame->script(), result, resultString))            cstr = resultString.utf8();        RefPtr<PluginStream> stream = PluginStream::create(this, m_parentFrame, request->frameLoadRequest().resourceRequest(), request->sendNotification(), request->notifyData(), plugin()->pluginFuncs(), instance(), m_plugin->quirks());        m_streams.add(stream);        stream->sendJavaScriptStream(requestURL, cstr);    }}void PluginView::requestTimerFired(Timer<PluginView>* timer){    ASSERT(timer == &m_requestTimer);    ASSERT(m_requests.size() > 0);    ASSERT(!m_isJavaScriptPaused);    PluginRequest* request = m_requests[0];    m_requests.remove(0);        // Schedule a new request before calling performRequest since the call to    // performRequest can cause the plugin view to be deleted.    if (m_requests.size() > 0)        m_requestTimer.startOneShot(0);    performRequest(request);    delete request;}void PluginView::scheduleRequest(PluginRequest* request){    m_requests.append(request);    if (!m_isJavaScriptPaused)        m_requestTimer.startOneShot(0);}NPError PluginView::load(const FrameLoadRequest& frameLoadRequest, bool sendNotification, void* notifyData){    ASSERT(frameLoadRequest.resourceRequest().httpMethod() == "GET" || frameLoadRequest.resourceRequest().httpMethod() == "POST");    KURL url = frameLoadRequest.resourceRequest().url();        if (url.isEmpty())        return NPERR_INVALID_URL;    // Don't allow requests to be made when the document loader is stopping all loaders.    if (m_parentFrame->loader()->documentLoader()->isStopping())        return NPERR_GENERIC_ERROR;    const String& targetFrameName = frameLoadRequest.frameName();    String jsString = scriptStringIfJavaScriptURL(url);    if (!jsString.isNull()) {        Settings* settings = m_parentFrame->settings();        // Return NPERR_GENERIC_ERROR if JS is disabled. This is what Mozilla does.        if (!settings || !settings->isJavaScriptEnabled())            return NPERR_GENERIC_ERROR;                // For security reasons, only allow JS requests to be made on the frame that contains the plug-in.        if (!targetFrameName.isNull() && m_parentFrame->tree()->find(targetFrameName) != m_parentFrame)            return NPERR_INVALID_PARAM;    } else if (!FrameLoader::canLoad(url, String(), m_parentFrame->document())) {            return NPERR_GENERIC_ERROR;    }    PluginRequest* request = new PluginRequest(frameLoadRequest, sendNotification, notifyData, arePopupsAllowed());    scheduleRequest(request);    return NPERR_NO_ERROR;}static KURL makeURL(const KURL& baseURL, const char* relativeURLString){    String urlString = relativeURLString;    // Strip return characters.    urlString.replace('\n', "");    urlString.replace('\r', "");    return KURL(baseURL, urlString);}NPError PluginView::getURLNotify(const char* url, const char* target, void* notifyData){    FrameLoadRequest frameLoadRequest;    frameLoadRequest.setFrameName(target);    frameLoadRequest.resourceRequest().setHTTPMethod("GET");    frameLoadRequest.resourceRequest().setURL(makeURL(m_baseURL, url));    return load(frameLoadRequest, true, notifyData);}NPError PluginView::getURL(const char* url, const char* target){    FrameLoadRequest frameLoadRequest;    frameLoadRequest.setFrameName(target);    frameLoadRequest.resourceRequest().setHTTPMethod("GET");    frameLoadRequest.resourceRequest().setURL(makeURL(m_baseURL, url));    return load(frameLoadRequest, false, 0);}NPError PluginView::postURLNotify(const char* url, const char* target, uint32 len, const char* buf, NPBool file, void* notifyData){    return handlePost(url, target, len, buf, file, notifyData, true, true);}NPError PluginView::postURL(const char* url, const char* target, uint32 len, const char* buf, NPBool file){    // As documented, only allow headers to be specified via NPP_PostURL when using a file.    return handlePost(url, target, len, buf, file, 0, false, file);}NPError PluginView::newStream(NPMIMEType type, const char* target, NPStream** stream){    notImplemented();    // Unsupported    return NPERR_GENERIC_ERROR;}int32 PluginView::write(NPStream* stream, int32 len, void* buffer){    notImplemented();    // Unsupported    return -1;}NPError PluginView::destroyStream(NPStream* stream, NPReason reason){    if (!stream || PluginStream::ownerForStream(stream) != m_instance)        return NPERR_INVALID_INSTANCE_ERROR;    PluginStream* browserStream = static_cast<PluginStream*>(stream->ndata);    browserStream->cancelAndDestroyStream(reason);    return NPERR_NO_ERROR;}void PluginView::status(const char* message){    if (Page* page = m_parentFrame->page())        page->chrome()->setStatusbarText(m_parentFrame, String(message));}NPError PluginView::setValue(NPPVariable variable, void* value){    switch (variable) {    case NPPVpluginWindowBool:        m_isWindowed = value;        return NPERR_NO_ERROR;    case NPPVpluginTransparentBool:        m_isTransparent = value;        return NPERR_NO_ERROR;#if defined(XP_MACOSX)    case NPPVpluginDrawingModel:        return NPERR_NO_ERROR;    case NPPVpluginEventModel:        return NPERR_NO_ERROR;#endif    default:        notImplemented();        return NPERR_GENERIC_ERROR;    }}void PluginView::invalidateTimerFired(Timer<PluginView>* timer){    ASSERT(timer == &m_invalidateTimer);    for (unsigned i = 0; i < m_invalidRects.size(); i++)        invalidateRect(m_invalidRects[i]);    m_invalidRects.clear();}void PluginView::pushPopupsEnabledState(bool state){    m_popupStateStack.append(state);} void PluginView::popPopupsEnabledState(){    m_popupStateStack.removeLast();}bool PluginView::arePopupsAllowed() const{    if (!m_popupStateStack.isEmpty())        return m_popupStateStack.last();    return false;}void PluginView::setJavaScriptPaused(bool paused){    if (m_isJavaScriptPaused == paused)        return;    m_isJavaScriptPaused = paused;    if (m_isJavaScriptPaused)        m_requestTimer.stop();

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -