123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506 |
- /*
- * Copyright (C) 2006, 2007, 2008, 2009, 2010 Apple Inc. All rights reserved.
- * Copyright (C) 2008 Collabora Ltd. All rights reserved.
- * Copyright (C) 2010 Girish Ramakrishnan <girish@forwardbias.in>
- *
- * 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 "BridgeJSC.h"
- #include "Chrome.h"
- #include "CookieJar.h"
- #include "Document.h"
- #include "DocumentLoader.h"
- #include "Element.h"
- #include "FocusController.h"
- #include "Frame.h"
- #include "FrameLoadRequest.h"
- #include "FrameLoader.h"
- #include "FrameLoaderClient.h"
- #include "FrameTree.h"
- #include "FrameView.h"
- #include "GraphicsContext.h"
- #include "HTMLNames.h"
- #include "HTMLPlugInElement.h"
- #include "Image.h"
- #include "JSDOMBinding.h"
- #include "JSDOMWindow.h"
- #include "KeyboardEvent.h"
- #include "MIMETypeRegistry.h"
- #include "MouseEvent.h"
- #include "NotImplemented.h"
- #include "Page.h"
- #include "PlatformMouseEvent.h"
- #include "PluginDatabase.h"
- #include "PluginDebug.h"
- #include "PluginMainThreadScheduler.h"
- #include "PluginPackage.h"
- #include "ProxyServer.h"
- #include "RenderBox.h"
- #include "RenderObject.h"
- #include "ScriptController.h"
- #include "ScriptValue.h"
- #include "SecurityOrigin.h"
- #include "Settings.h"
- #include "WheelEvent.h"
- #include "c_instance.h"
- #include "npruntime_impl.h"
- #include "runtime_root.h"
- #include <runtime/JSCJSValue.h>
- #include <runtime/JSLock.h>
- #include <wtf/ASCIICType.h>
- #include <wtf/text/WTFString.h>
- #if OS(WINDOWS) && ENABLE(NETSCAPE_PLUGIN_API)
- #include "PluginMessageThrottlerWin.h"
- #endif
- using JSC::ExecState;
- using JSC::JSLock;
- using JSC::JSObject;
- using JSC::JSValue;
- #if ENABLE(NETSCAPE_PLUGIN_API)
- using std::min;
- using namespace WTF;
- namespace WebCore {
- using namespace HTMLNames;
- static int s_callingPlugin;
- typedef HashMap<NPP, PluginView*> InstanceMap;
- static InstanceMap& instanceMap()
- {
- static InstanceMap& map = *new InstanceMap;
- return map;
- }
- static String scriptStringIfJavaScriptURL(const KURL& url)
- {
- if (!protocolIsJavaScript(url))
- 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.
- FrameView* parentView = m_element->document()->view();
- clipRect.intersect(parentView->windowClipRectForFrameOwner(m_element, true));
- return clipRect;
- }
- void PluginView::setFrameRect(const IntRect& rect)
- {
- if (m_element->document()->printing())
- return;
- if (rect != frameRect())
- Widget::setFrameRect(rect);
- updatePluginWidget();
- #if OS(WINDOWS)
- // On Windows always call plugin to change geometry.
- setNPWindowRect(rect);
- #elif defined(XP_UNIX)
- // On Unix, multiple calls to setNPWindow() in windowed mode causes Flash to crash
- if (m_mode == NP_FULL || !m_isWindowed)
- setNPWindowRect(rect);
- #endif
- }
- void PluginView::frameRectsChanged()
- {
- updatePluginWidget();
- }
- void PluginView::clipRectChanged()
- {
- updatePluginWidget();
- }
- void PluginView::handleEvent(Event* event)
- {
- if (!m_plugin || m_isWindowed)
- return;
- // Protect the plug-in from deletion while dispatching the event.
- RefPtr<PluginView> protect(this);
- if (event->isMouseEvent())
- handleMouseEvent(static_cast<MouseEvent*>(event));
- else if (event->isKeyboardEvent())
- handleKeyboardEvent(static_cast<KeyboardEvent*>(event));
- #if defined(XP_MACOSX)
- else if (event->type() == eventNames().mousewheelEvent)
- handleWheelEvent(static_cast<WheelEvent*>(event));
- #endif
- else if (event->type() == eventNames().contextmenuEvent)
- event->setDefaultHandled(); // We don't know if the plug-in has handled mousedown event by displaying a context menu, so we never want WebKit to show a default one.
- #if defined(XP_UNIX) && ENABLE(NETSCAPE_PLUGIN_API)
- else if (event->type() == eventNames().focusoutEvent)
- handleFocusOutEvent();
- else if (event->type() == eventNames().focusinEvent)
- handleFocusInEvent();
- #endif
- }
- void PluginView::init()
- {
- if (m_haveInitialized)
- return;
- m_haveInitialized = true;
- if (!m_plugin) {
- ASSERT(m_status == PluginStatusCanNotFindPlugin);
- return;
- }
- LOG(Plugins, "PluginView::init(): Initializing plug-in '%s'", m_plugin->name().utf8().data());
- if (!m_plugin->load()) {
- m_plugin = 0;
- m_status = PluginStatusCanNotLoadPlugin;
- return;
- }
- if (!startOrAddToUnstartedList()) {
- m_status = PluginStatusCanNotLoadPlugin;
- return;
- }
- m_status = PluginStatusLoadedSuccessfully;
- }
- bool PluginView::startOrAddToUnstartedList()
- {
- if (!m_parentFrame->page())
- return false;
- // We only delay starting the plug-in if we're going to kick off the load
- // ourselves. Otherwise, the loader will try to deliver data before we've
- // started the plug-in.
- if (!m_loadManually && !m_parentFrame->page()->canStartMedia()) {
- m_parentFrame->document()->addMediaCanStartListener(this);
- m_isWaitingToStart = true;
- return true;
- }
- return start();
- }
- bool PluginView::start()
- {
- if (m_isStarted)
- return false;
- m_isWaitingToStart = false;
- PluginMainThreadScheduler::scheduler().registerPlugin(m_instance);
- ASSERT(m_plugin);
- ASSERT(m_plugin->pluginFuncs()->newp);
- NPError npErr;
- {
- PluginView::setCurrentPluginView(this);
- JSC::JSLock::DropAllLocks dropAllLocks(JSDOMWindowBase::commonVM());
- setCallingPlugin(true);
- npErr = m_plugin->pluginFuncs()->newp((NPMIMEType)m_mimeType.utf8().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) {
- m_status = PluginStatusCanNotLoadPlugin;
- PluginMainThreadScheduler::scheduler().unregisterPlugin(m_instance);
- return false;
- }
- m_isStarted = true;
- if (!m_url.isEmpty() && !m_loadManually) {
- FrameLoadRequest frameLoadRequest(m_parentFrame->document()->securityOrigin());
- frameLoadRequest.resourceRequest().setHTTPMethod("GET");
- frameLoadRequest.resourceRequest().setURL(m_url);
- load(frameLoadRequest, false, 0);
- }
- m_status = PluginStatusLoadedSuccessfully;
- if (!platformStart())
- m_status = PluginStatusCanNotLoadPlugin;
- if (m_status != PluginStatusLoadedSuccessfully)
- return false;
- return true;
- }
- void PluginView::mediaCanStart()
- {
- ASSERT(!m_isStarted);
- if (!start())
- parentFrame()->loader()->client()->dispatchDidFailToStartPlugin(this);
- }
- PluginView::~PluginView()
- {
- LOG(Plugins, "PluginView::~PluginView()");
- ASSERT(!m_lifeSupportTimer.isActive());
- // If we failed to find the plug-in, we'll return early in our constructor, and
- // m_instance will be 0.
- if (m_instance)
- instanceMap().remove(m_instance);
- if (m_isWaitingToStart)
- m_parentFrame->document()->removeMediaCanStartListener(this);
- stop();
- freeStringArray(m_paramNames, m_paramCount);
- freeStringArray(m_paramValues, m_paramCount);
- platformDestroy();
- m_parentFrame->script()->cleanupScriptObjectsForPlugin(this);
- if (m_plugin && !(m_plugin->quirks().contains(PluginQuirkDontUnloadPlugin)))
- m_plugin->unload();
- }
- void PluginView::stop()
- {
- if (!m_isStarted)
- return;
- LOG(Plugins, "PluginView::stop(): Stopping plug-in '%s'", m_plugin->name().utf8().data());
- HashSet<RefPtr<PluginStream> > streams = m_streams;
- HashSet<RefPtr<PluginStream> >::iterator end = streams.end();
- for (HashSet<RefPtr<PluginStream> >::iterator it = streams.begin(); it != end; ++it) {
- (*it)->stop();
- disconnectStream((*it).get());
- }
- ASSERT(m_streams.isEmpty());
- m_isStarted = false;
- JSC::JSLock::DropAllLocks dropAllLocks(JSDOMWindowBase::commonVM());
- #if ENABLE(NETSCAPE_PLUGIN_API)
- #if defined(XP_WIN) && !PLATFORM(GTK)
- // Unsubclass the window
- if (m_isWindowed) {
- #if OS(WINCE)
- WNDPROC currentWndProc = (WNDPROC)GetWindowLong(platformPluginWidget(), GWL_WNDPROC);
- if (currentWndProc == PluginViewWndProc)
- SetWindowLong(platformPluginWidget(), GWL_WNDPROC, (LONG)m_pluginWndProc);
- #else
- WNDPROC currentWndProc = (WNDPROC)GetWindowLongPtr(platformPluginWidget(), GWLP_WNDPROC);
- if (currentWndProc == PluginViewWndProc)
- SetWindowLongPtr(platformPluginWidget(), GWLP_WNDPROC, (LONG_PTR)m_pluginWndProc);
- #endif
- }
- #endif // !defined(XP_WIN) || PLATFORM(GTK)
- #endif // ENABLE(NETSCAPE_PLUGIN_API)
- #if !defined(XP_MACOSX)
- // Clear the window
- m_npWindow.window = 0;
- if (m_plugin->pluginFuncs()->setwindow && !m_plugin->quirks().contains(PluginQuirkDontSetNullWindowHandleOnDestroy)) {
- PluginView::setCurrentPluginView(this);
- setCallingPlugin(true);
- m_plugin->pluginFuncs()->setwindow(m_instance, &m_npWindow);
- setCallingPlugin(false);
- PluginView::setCurrentPluginView(0);
- }
- #ifdef XP_UNIX
- delete static_cast<NPSetWindowCallbackStruct*>(m_npWindow.ws_info);
- m_npWindow.ws_info = 0;
- #endif
- #endif // !defined(XP_MACOSX)
- PluginMainThreadScheduler::scheduler().unregisterPlugin(m_instance);
- NPSavedData* savedData = 0;
- PluginView::setCurrentPluginView(this);
- setCallingPlugin(true);
- NPError npErr = m_plugin->pluginFuncs()->destroy(m_instance, &savedData);
- setCallingPlugin(false);
- LOG_NPERROR(npErr);
- PluginView::setCurrentPluginView(0);
- #if ENABLE(NETSCAPE_PLUGIN_API)
- if (savedData) {
- // TODO: Actually save this data instead of just discarding it
- if (savedData->buf)
- NPN_MemFree(savedData->buf);
- NPN_MemFree(savedData);
- }
- #endif
- m_instance->pdata = 0;
- }
- void PluginView::setCurrentPluginView(PluginView* pluginView)
- {
- s_currentPluginView = pluginView;
- }
- PluginView* PluginView::currentPluginView()
- {
- return s_currentPluginView;
- }
- static char* createUTF8String(const String& str)
- {
- CString cstr = str.utf8();
- const size_t cstrLength = cstr.length();
- char* result = reinterpret_cast<char*>(fastMalloc(cstrLength + 1));
- memcpy(result, cstr.data(), cstrLength);
- result[cstrLength] = '\0';
- return result;
- }
- void PluginView::performRequest(PluginRequest* request)
- {
- if (!m_isStarted)
- return;
- // 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.get(), request->frameLoadRequest().resourceRequest(), request->sendNotification(), request->notifyData(), plugin()->pluginFuncs(), instance(), m_plugin->quirks());
- m_streams.add(stream);
- stream->start();
- } else {
- // If the target frame is our frame, we could destroy the
- // PluginView, so we protect it. <rdar://problem/6991251>
- RefPtr<PluginView> protect(this);
- FrameLoadRequest frameRequest(m_parentFrame.get(), request->frameLoadRequest().resourceRequest());
- frameRequest.setFrameName(targetFrameName);
- frameRequest.setShouldCheckNewWindowPolicy(true);
- m_parentFrame->loader()->load(frameRequest);
- // FIXME: <rdar://problem/4807469> This should be sent when the document has finished loading
- if (request->sendNotification()) {
- PluginView::setCurrentPluginView(this);
- JSC::JSLock::DropAllLocks dropAllLocks(JSDOMWindowBase::commonVM());
- 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 it.
- RefPtr<PluginView> protector(this);
- ScriptValue result = m_parentFrame->script()->executeScript(jsString, request->shouldAllowPopups());
- if (targetFrameName.isNull()) {
- String resultString;
- ScriptState* scriptState = m_parentFrame->script()->globalObject(pluginWorld())->globalExec();
- CString cstr;
- if (result.getString(scriptState, resultString))
- cstr = resultString.utf8();
- RefPtr<PluginStream> stream = PluginStream::create(this, m_parentFrame.get(), 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_UNUSED(timer, timer == &m_requestTimer);
- ASSERT(!m_requests.isEmpty());
- ASSERT(!m_isJavaScriptPaused);
- OwnPtr<PluginRequest> request = m_requests[0].release();
- 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.isEmpty())
- m_requestTimer.startOneShot(0);
- performRequest(request.get());
- }
- void PluginView::scheduleRequest(PassOwnPtr<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.
- DocumentLoader* loader = m_parentFrame->loader()->documentLoader();
- if (!loader || loader->isStopping())
- return NPERR_GENERIC_ERROR;
- const String& targetFrameName = frameLoadRequest.frameName();
- String jsString = scriptStringIfJavaScriptURL(url);
- if (!jsString.isNull()) {
- // Return NPERR_GENERIC_ERROR if JS is disabled. This is what Mozilla does.
- if (!m_parentFrame->script()->canExecuteScripts(NotAboutToExecuteScript))
- 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 (!m_parentFrame->document()->securityOrigin()->canDisplay(url))
- return NPERR_GENERIC_ERROR;
- scheduleRequest(adoptPtr(new PluginRequest(frameLoadRequest, sendNotification, notifyData, arePopupsAllowed())));
- return NPERR_NO_ERROR;
- }
- static KURL makeURL(const KURL& baseURL, const char* relativeURLString)
- {
- String urlString = relativeURLString;
- // Strip return characters.
- urlString.replaceWithLiteral('\n', "");
- urlString.replaceWithLiteral('\r', "");
- return KURL(baseURL, urlString);
- }
- NPError PluginView::getURLNotify(const char* url, const char* target, void* notifyData)
- {
- FrameLoadRequest frameLoadRequest(m_parentFrame->document()->securityOrigin());
- frameLoadRequest.setFrameName(target);
- frameLoadRequest.resourceRequest().setHTTPMethod("GET");
- frameLoadRequest.resourceRequest().setURL(makeURL(m_parentFrame->document()->baseURL(), url));
- return load(frameLoadRequest, true, notifyData);
- }
- NPError PluginView::getURL(const char* url, const char* target)
- {
- FrameLoadRequest frameLoadRequest(m_parentFrame->document()->securityOrigin());
- frameLoadRequest.setFrameName(target);
- frameLoadRequest.resourceRequest().setHTTPMethod("GET");
- frameLoadRequest.resourceRequest().setURL(makeURL(m_parentFrame->document()->baseURL(), url));
- return load(frameLoadRequest, false, 0);
- }
- NPError PluginView::postURLNotify(const char* url, const char* target, uint32_t 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_t 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, const char* /* target */, NPStream**)
- {
- notImplemented();
- // Unsupported
- return NPERR_GENERIC_ERROR;
- }
- int32_t PluginView::write(NPStream*, int32_t /* 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.get(), String::fromUTF8(message));
- }
- NPError PluginView::setValue(NPPVariable variable, void* value)
- {
- LOG(Plugins, "PluginView::setValue(%s): ", prettyNameForNPPVariable(variable, value).data());
- 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: {
- // Can only set drawing model inside NPP_New()
- if (this != currentPluginView())
- return NPERR_GENERIC_ERROR;
- NPDrawingModel newDrawingModel = NPDrawingModel(uintptr_t(value));
- switch (newDrawingModel) {
- case NPDrawingModelCoreGraphics:
- return NPERR_NO_ERROR;
- case NPDrawingModelCoreAnimation:
- default:
- LOG(Plugins, "Plugin asked for unsupported drawing model: %s",
- prettyNameForDrawingModel(newDrawingModel));
- return NPERR_GENERIC_ERROR;
- }
- }
- case NPPVpluginEventModel: {
- // Can only set event model inside NPP_New()
- if (this != currentPluginView())
- return NPERR_GENERIC_ERROR;
- NPEventModel newEventModel = NPEventModel(uintptr_t(value));
- switch (newEventModel) {
- case NPEventModelCocoa:
- return NPERR_NO_ERROR;
- default:
- LOG(Plugins, "Plugin asked for unsupported event model: %s",
- prettyNameForEventModel(newEventModel));
- return NPERR_GENERIC_ERROR;
- }
- }
- #endif // defined(XP_MACOSX)
- default:
- notImplemented();
- return NPERR_GENERIC_ERROR;
- }
- }
- void PluginView::invalidateTimerFired(Timer<PluginView>* timer)
- {
- ASSERT_UNUSED(timer, 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();
- else if (!m_requests.isEmpty())
- m_requestTimer.startOneShot(0);
- }
- #if ENABLE(NETSCAPE_PLUGIN_API)
- NPObject* PluginView::npObject()
- {
- NPObject* object = 0;
- if (!m_isStarted || !m_plugin || !m_plugin->pluginFuncs()->getvalue)
- return 0;
- // On Windows, calling Java's NPN_GetValue can allow the message loop to
- // run, allowing loading to take place or JavaScript to run. Protect the
- // PluginView from destruction. <rdar://problem/6978804>
- RefPtr<PluginView> protect(this);
- NPError npErr;
- {
- PluginView::setCurrentPluginView(this);
- JSC::JSLock::DropAllLocks dropAllLocks(JSDOMWindowBase::commonVM());
- setCallingPlugin(true);
- npErr = m_plugin->pluginFuncs()->getvalue(m_instance, NPPVpluginScriptableNPObject, &object);
- setCallingPlugin(false);
- PluginView::setCurrentPluginView(0);
- }
- if (npErr != NPERR_NO_ERROR)
- return 0;
- return object;
- }
- #endif
- PassRefPtr<JSC::Bindings::Instance> PluginView::bindingInstance()
- {
- #if ENABLE(NETSCAPE_PLUGIN_API)
- NPObject* object = npObject();
- if (!object)
- return 0;
- if (hasOneRef()) {
- // The renderer for the PluginView was destroyed during the above call, and
- // the PluginView will be destroyed when this function returns, so we
- // return null.
- return 0;
- }
- RefPtr<JSC::Bindings::RootObject> root = m_parentFrame->script()->createRootObject(this);
- RefPtr<JSC::Bindings::Instance> instance = JSC::Bindings::CInstance::create(object, root.release());
- _NPN_ReleaseObject(object);
- return instance.release();
- #else
- return 0;
- #endif
- }
- void PluginView::disconnectStream(PluginStream* stream)
- {
- ASSERT(m_streams.contains(stream));
- m_streams.remove(stream);
- }
- void PluginView::setParameters(const Vector<String>& paramNames, const Vector<String>& paramValues)
- {
- ASSERT(paramNames.size() == paramValues.size());
- unsigned size = paramNames.size();
- unsigned paramCount = 0;
- m_paramNames = reinterpret_cast<char**>(fastMalloc(sizeof(char*) * size));
- m_paramValues = reinterpret_cast<char**>(fastMalloc(sizeof(char*) * size));
- for (unsigned i = 0; i < size; i++) {
- if (m_plugin->quirks().contains(PluginQuirkRemoveWindowlessVideoParam) && equalIgnoringCase(paramNames[i], "windowlessvideo"))
- continue;
- if (paramNames[i] == "pluginspage")
- m_pluginsPage = paramValues[i];
- m_paramNames[paramCount] = createUTF8String(paramNames[i]);
- m_paramValues[paramCount] = createUTF8String(paramValues[i]);
- paramCount++;
- }
- m_paramCount = paramCount;
- }
- PluginView::PluginView(Frame* parentFrame, const IntSize& size, PluginPackage* plugin, HTMLPlugInElement* element, const KURL& url, const Vector<String>& paramNames, const Vector<String>& paramValues, const String& mimeType, bool loadManually)
- : m_parentFrame(parentFrame)
- , m_plugin(plugin)
- , m_element(element)
- , m_isStarted(false)
- , m_url(url)
- , m_status(PluginStatusLoadedSuccessfully)
- , m_requestTimer(this, &PluginView::requestTimerFired)
- , m_invalidateTimer(this, &PluginView::invalidateTimerFired)
- , m_popPopupsStateTimer(this, &PluginView::popPopupsStateTimerFired)
- , m_lifeSupportTimer(this, &PluginView::lifeSupportTimerFired)
- , m_mode(loadManually ? NP_FULL : NP_EMBED)
- , m_paramNames(0)
- , m_paramValues(0)
- , m_mimeType(mimeType)
- , m_instance(0)
- #if defined(XP_MACOSX)
- , m_isWindowed(false)
- , m_updatedCocoaTextInputRequested(false)
- , m_keyDownSent(false)
- , m_disregardKeyUpCounter(0)
- #else
- , m_isWindowed(true)
- #endif
- , m_isTransparent(false)
- , m_haveInitialized(false)
- , m_isWaitingToStart(false)
- #if defined(XP_UNIX)
- , m_needsXEmbed(false)
- #endif
- #if OS(WINDOWS) && ENABLE(NETSCAPE_PLUGIN_API)
- , m_pluginWndProc(0)
- , m_lastMessage(0)
- , m_isCallingPluginWndProc(false)
- , m_wmPrintHDC(0)
- , m_haveUpdatedPluginWidget(false)
- #endif
- #if (PLATFORM(QT) && OS(WINDOWS)) || PLATFORM(EFL)
- , m_window(0)
- #endif
- #if defined(XP_MACOSX)
- , m_contextRef(0)
- #endif
- #if defined(XP_UNIX) && ENABLE(NETSCAPE_PLUGIN_API)
- , m_hasPendingGeometryChange(true)
- , m_drawable(0)
- , m_visual(0)
- , m_colormap(0)
- , m_pluginDisplay(0)
- #endif
- #if PLATFORM(QT) && defined(MOZ_PLATFORM_MAEMO) && (MOZ_PLATFORM_MAEMO >= 5)
- , m_renderToImage(false)
- #endif
- , m_loadManually(loadManually)
- , m_manualStream(0)
- , m_isJavaScriptPaused(false)
- , m_haveCalledSetWindow(false)
- {
- if (!m_plugin) {
- m_status = PluginStatusCanNotFindPlugin;
- return;
- }
- m_instance = &m_instanceStruct;
- m_instance->ndata = this;
- m_instance->pdata = 0;
- instanceMap().add(m_instance, this);
- setParameters(paramNames, paramValues);
- memset(&m_npWindow, 0, sizeof(m_npWindow));
- #if defined(XP_MACOSX)
- memset(&m_npCgContext, 0, sizeof(m_npCgContext));
- #endif
- resize(size);
- }
- void PluginView::focusPluginElement()
- {
- if (Page* page = m_parentFrame->page())
- page->focusController()->setFocusedElement(m_element, m_parentFrame);
- else
- m_parentFrame->document()->setFocusedElement(m_element);
- }
- void PluginView::didReceiveResponse(const ResourceResponse& response)
- {
- if (m_status != PluginStatusLoadedSuccessfully)
- return;
- ASSERT(m_loadManually);
- ASSERT(!m_manualStream);
- m_manualStream = PluginStream::create(this, m_parentFrame.get(), m_parentFrame->loader()->activeDocumentLoader()->request(), false, 0, plugin()->pluginFuncs(), instance(), m_plugin->quirks());
- m_manualStream->setLoadManually(true);
- m_manualStream->didReceiveResponse(0, response);
- }
- void PluginView::didReceiveData(const char* data, int length)
- {
- if (m_status != PluginStatusLoadedSuccessfully)
- return;
- ASSERT(m_loadManually);
- ASSERT(m_manualStream);
-
- m_manualStream->didReceiveData(0, data, length);
- }
- void PluginView::didFinishLoading()
- {
- if (m_status != PluginStatusLoadedSuccessfully)
- return;
- ASSERT(m_loadManually);
- ASSERT(m_manualStream);
- m_manualStream->didFinishLoading(0);
- }
- void PluginView::didFail(const ResourceError& error)
- {
- if (m_status != PluginStatusLoadedSuccessfully)
- return;
- ASSERT(m_loadManually);
-
- if (m_manualStream)
- m_manualStream->didFail(0, error);
- }
- void PluginView::setCallingPlugin(bool b) const
- {
- if (!m_plugin->quirks().contains(PluginQuirkHasModalMessageLoop))
- return;
- if (b)
- ++s_callingPlugin;
- else
- --s_callingPlugin;
- ASSERT(s_callingPlugin >= 0);
- }
- bool PluginView::isCallingPlugin()
- {
- return s_callingPlugin > 0;
- }
- PassRefPtr<PluginView> PluginView::create(Frame* parentFrame, const IntSize& size, HTMLPlugInElement* element, const KURL& url, const Vector<String>& paramNames, const Vector<String>& paramValues, const String& mimeType, bool loadManually)
- {
- // if we fail to find a plugin for this MIME type, findPlugin will search for
- // a plugin by the file extension and update the MIME type, so pass a mutable String
- String mimeTypeCopy = mimeType;
- PluginPackage* plugin = PluginDatabase::installedPlugins()->findPlugin(url, mimeTypeCopy);
- // No plugin was found, try refreshing the database and searching again
- if (!plugin && PluginDatabase::installedPlugins()->refresh()) {
- mimeTypeCopy = mimeType;
- plugin = PluginDatabase::installedPlugins()->findPlugin(url, mimeTypeCopy);
- }
- return adoptRef(new PluginView(parentFrame, size, plugin, element, url, paramNames, paramValues, mimeTypeCopy, loadManually));
- }
- void PluginView::freeStringArray(char** stringArray, int length)
- {
- if (!stringArray)
- return;
- for (int i = 0; i < length; i++)
- fastFree(stringArray[i]);
- fastFree(stringArray);
- }
- static inline bool startsWithBlankLine(const Vector<char>& buffer)
- {
- return buffer.size() > 0 && buffer[0] == '\n';
- }
- static inline int locationAfterFirstBlankLine(const Vector<char>& buffer)
- {
- const char* bytes = buffer.data();
- unsigned length = buffer.size();
- for (unsigned i = 0; i < length - 4; i++) {
- // Support for Acrobat. It sends "\n\n".
- if (bytes[i] == '\n' && bytes[i + 1] == '\n')
- return i + 2;
-
- // Returns the position after 2 CRLF's or 1 CRLF if it is the first line.
- if (bytes[i] == '\r' && bytes[i + 1] == '\n') {
- i += 2;
- if (i == 2)
- return i;
- else if (bytes[i] == '\n')
- // Support for Director. It sends "\r\n\n" (3880387).
- return i + 1;
- else if (bytes[i] == '\r' && bytes[i + 1] == '\n')
- // Support for Flash. It sends "\r\n\r\n" (3758113).
- return i + 2;
- }
- }
- return -1;
- }
- static inline const char* findEOL(const char* bytes, unsigned length)
- {
- // According to the HTTP specification EOL is defined as
- // a CRLF pair. Unfortunately, some servers will use LF
- // instead. Worse yet, some servers will use a combination
- // of both (e.g. <header>CRLFLF<body>), so findEOL needs
- // to be more forgiving. It will now accept CRLF, LF or
- // CR.
- //
- // It returns NULL if EOLF is not found or it will return
- // a pointer to the first terminating character.
- for (unsigned i = 0; i < length; i++) {
- if (bytes[i] == '\n')
- return bytes + i;
- if (bytes[i] == '\r') {
- // Check to see if spanning buffer bounds
- // (CRLF is across reads). If so, wait for
- // next read.
- if (i + 1 == length)
- break;
- return bytes + i;
- }
- }
- return 0;
- }
- static inline String capitalizeRFC822HeaderFieldName(const String& name)
- {
- bool capitalizeCharacter = true;
- String result;
- for (unsigned i = 0; i < name.length(); i++) {
- UChar c;
- if (capitalizeCharacter && name[i] >= 'a' && name[i] <= 'z')
- c = toASCIIUpper(name[i]);
- else if (!capitalizeCharacter && name[i] >= 'A' && name[i] <= 'Z')
- c = toASCIILower(name[i]);
- else
- c = name[i];
- if (name[i] == '-')
- capitalizeCharacter = true;
- else
- capitalizeCharacter = false;
- result.append(c);
- }
- return result;
- }
- static inline HTTPHeaderMap parseRFC822HeaderFields(const Vector<char>& buffer, unsigned length)
- {
- const char* bytes = buffer.data();
- const char* eol;
- String lastKey;
- HTTPHeaderMap headerFields;
- // Loop ove rlines until we're past the header, or we can't find any more end-of-lines
- while ((eol = findEOL(bytes, length))) {
- const char* line = bytes;
- int lineLength = eol - bytes;
-
- // Move bytes to the character after the terminator as returned by findEOL.
- bytes = eol + 1;
- if ((*eol == '\r') && (*bytes == '\n'))
- bytes++; // Safe since findEOL won't return a spanning CRLF.
- length -= (bytes - line);
- if (lineLength == 0)
- // Blank line; we're at the end of the header
- break;
- else if (*line == ' ' || *line == '\t') {
- // Continuation of the previous header
- if (lastKey.isNull()) {
- // malformed header; ignore it and continue
- continue;
- } else {
- // Merge the continuation of the previous header
- String currentValue = headerFields.get(lastKey);
- String newValue(line, lineLength);
- headerFields.set(lastKey, currentValue + newValue);
- }
- } else {
- // Brand new header
- const char* colon;
- for (colon = line; *colon != ':' && colon != eol; colon++) {
- // empty loop
- }
- if (colon == eol)
- // malformed header; ignore it and continue
- continue;
- else {
- lastKey = capitalizeRFC822HeaderFieldName(String(line, colon - line));
- String value;
- for (colon++; colon != eol; colon++) {
- if (*colon != ' ' && *colon != '\t')
- break;
- }
- if (colon == eol)
- value = "";
- else
- value = String(colon, eol - colon);
- String oldValue = headerFields.get(lastKey);
- if (!oldValue.isNull())
- value = oldValue + ", " + value;
- headerFields.set(lastKey, value);
- }
- }
- }
- return headerFields;
- }
- NPError PluginView::handlePost(const char* url, const char* target, uint32_t len, const char* buf, bool file, void* notifyData, bool sendNotification, bool allowHeaders)
- {
- if (!url || !len || !buf)
- return NPERR_INVALID_PARAM;
- FrameLoadRequest frameLoadRequest(m_parentFrame->document()->securityOrigin());
- HTTPHeaderMap headerFields;
- Vector<char> buffer;
-
- if (file) {
- NPError readResult = handlePostReadFile(buffer, len, buf);
- if(readResult != NPERR_NO_ERROR)
- return readResult;
- } else {
- buffer.resize(len);
- memcpy(buffer.data(), buf, len);
- }
- const char* postData = buffer.data();
- int postDataLength = buffer.size();
- if (allowHeaders) {
- if (startsWithBlankLine(buffer)) {
- postData++;
- postDataLength--;
- } else {
- int location = locationAfterFirstBlankLine(buffer);
- if (location != -1) {
- // If the blank line is somewhere in the middle of the buffer, everything before is the header
- headerFields = parseRFC822HeaderFields(buffer, location);
- unsigned dataLength = buffer.size() - location;
- // Sometimes plugins like to set Content-Length themselves when they post,
- // but WebFoundation does not like that. So we will remove the header
- // and instead truncate the data to the requested length.
- String contentLength = headerFields.get("Content-Length");
- if (!contentLength.isNull())
- dataLength = min(contentLength.toInt(), (int)dataLength);
- headerFields.remove("Content-Length");
- postData += location;
- postDataLength = dataLength;
- }
- }
- }
- frameLoadRequest.resourceRequest().setHTTPMethod("POST");
- frameLoadRequest.resourceRequest().setURL(makeURL(m_parentFrame->document()->baseURL(), url));
- frameLoadRequest.resourceRequest().addHTTPHeaderFields(headerFields);
- frameLoadRequest.resourceRequest().setHTTPBody(FormData::create(postData, postDataLength));
- frameLoadRequest.setFrameName(target);
- return load(frameLoadRequest, sendNotification, notifyData);
- }
- void PluginView::invalidateWindowlessPluginRect(const IntRect& rect)
- {
- if (!isVisible())
- return;
-
- if (!m_element->renderer())
- return;
- RenderBox* renderer = toRenderBox(m_element->renderer());
-
- IntRect dirtyRect = rect;
- dirtyRect.move(renderer->borderLeft() + renderer->paddingLeft(), renderer->borderTop() + renderer->paddingTop());
- renderer->repaintRectangle(dirtyRect);
- }
- void PluginView::paintMissingPluginIcon(GraphicsContext* context, const IntRect& rect)
- {
- static RefPtr<Image> nullPluginImage;
- if (!nullPluginImage)
- nullPluginImage = Image::loadPlatformResource("nullPlugin");
- IntRect imageRect(frameRect().x(), frameRect().y(), nullPluginImage->width(), nullPluginImage->height());
- int xOffset = (frameRect().width() - imageRect.width()) / 2;
- int yOffset = (frameRect().height() - imageRect.height()) / 2;
- imageRect.move(xOffset, yOffset);
- if (!rect.intersects(imageRect))
- return;
- context->save();
- context->clip(windowClipRect());
- context->drawImage(nullPluginImage.get(), ColorSpaceDeviceRGB, imageRect.location());
- context->restore();
- }
- static const char* MozillaUserAgent = "Mozilla/5.0 ("
- #if defined(XP_MACOSX)
- "Macintosh; U; Intel Mac OS X;"
- #elif defined(XP_WIN)
- "Windows; U; Windows NT 5.1;"
- #elif defined(XP_UNIX)
- // The Gtk port uses X11 plugins in Mac.
- #if OS(DARWIN) && PLATFORM(GTK)
- "X11; U; Intel Mac OS X;"
- #else
- "X11; U; Linux i686;"
- #endif
- #endif
- " en-US; rv:1.8.1) Gecko/20061010 Firefox/2.0";
- static const char* const ChromeUserAgent = "Mozilla/5.0 ("
- #if defined(XP_MACOSX)
- "Macintosh; U; Intel Mac OS X;"
- #elif defined(XP_WIN)
- "Windows; U; Windows NT 5.1;"
- #elif defined(XP_UNIX)
- // The Gtk port uses X11 plugins in Mac.
- #if OS(DARWIN) && PLATFORM(GTK)
- "X11; U; Intel Mac OS X;"
- #else
- "X11; U; Linux i686;"
- #endif
- #endif
- " AppleWebKit/534.34 (KHTML, like Gecko) Chrome/19.0.1055.1 Safari/534.34";
- const char* PluginView::userAgent()
- {
- if (m_plugin->quirks().contains(PluginQuirkWantsMozillaUserAgent))
- return MozillaUserAgent;
- else if (m_plugin->quirks().contains(PluginQuirkWantsChromeUserAgent))
- return ChromeUserAgent;
- if (m_userAgent.isNull())
- m_userAgent = m_parentFrame->loader()->userAgent(m_url).utf8();
- return m_userAgent.data();
- }
- #if ENABLE(NETSCAPE_PLUGIN_API)
- const char* PluginView::userAgentStatic()
- {
- return MozillaUserAgent;
- }
- #endif
- void PluginView::lifeSupportTimerFired(Timer<PluginView>*)
- {
- deref();
- }
- void PluginView::keepAlive()
- {
- if (m_lifeSupportTimer.isActive())
- return;
- ref();
- m_lifeSupportTimer.startOneShot(0);
- }
- #if ENABLE(NETSCAPE_PLUGIN_API)
- void PluginView::keepAlive(NPP instance)
- {
- PluginView* view = instanceMap().get(instance);
- if (!view)
- return;
- view->keepAlive();
- }
- NPError PluginView::getValueStatic(NPNVariable variable, void* value)
- {
- LOG(Plugins, "PluginView::getValueStatic(%s)", prettyNameForNPNVariable(variable).data());
- NPError result;
- if (platformGetValueStatic(variable, value, &result))
- return result;
- return NPERR_GENERIC_ERROR;
- }
- NPError PluginView::getValue(NPNVariable variable, void* value)
- {
- LOG(Plugins, "PluginView::getValue(%s)", prettyNameForNPNVariable(variable).data());
- NPError result;
- if (platformGetValue(variable, value, &result))
- return result;
- if (platformGetValueStatic(variable, value, &result))
- return result;
- switch (variable) {
- case NPNVWindowNPObject: {
- if (m_isJavaScriptPaused)
- return NPERR_GENERIC_ERROR;
- NPObject* windowScriptObject = m_parentFrame->script()->windowScriptNPObject();
- // Return value is expected to be retained, as described here: <http://www.mozilla.org/projects/plugin/npruntime.html>
- if (windowScriptObject)
- _NPN_RetainObject(windowScriptObject);
- void** v = (void**)value;
- *v = windowScriptObject;
- return NPERR_NO_ERROR;
- }
- case NPNVPluginElementNPObject: {
- if (m_isJavaScriptPaused)
- return NPERR_GENERIC_ERROR;
- NPObject* pluginScriptObject = 0;
- if (m_element->hasTagName(appletTag) || m_element->hasTagName(embedTag) || m_element->hasTagName(objectTag))
- pluginScriptObject = m_element->getNPObject();
- // Return value is expected to be retained, as described here: <http://www.mozilla.org/projects/plugin/npruntime.html>
- if (pluginScriptObject)
- _NPN_RetainObject(pluginScriptObject);
- void** v = (void**)value;
- *v = pluginScriptObject;
- return NPERR_NO_ERROR;
- }
- case NPNVprivateModeBool: {
- Page* page = m_parentFrame->page();
- if (!page)
- return NPERR_GENERIC_ERROR;
- *((NPBool*)value) = !page->settings() || page->settings()->privateBrowsingEnabled();
- return NPERR_NO_ERROR;
- }
- default:
- return NPERR_GENERIC_ERROR;
- }
- }
- static Frame* getFrame(Frame* parentFrame, Element* element)
- {
- if (parentFrame)
- return parentFrame;
-
- Document* document = element->document();
- if (!document)
- document = element->ownerDocument();
- if (document)
- return document->frame();
-
- return 0;
- }
- NPError PluginView::getValueForURL(NPNURLVariable variable, const char* url, char** value, uint32_t* len)
- {
- LOG(Plugins, "PluginView::getValueForURL(%s)", prettyNameForNPNURLVariable(variable).data());
- NPError result = NPERR_NO_ERROR;
- switch (variable) {
- case NPNURLVCookie: {
- KURL u(m_parentFrame->document()->baseURL(), url);
- if (u.isValid()) {
- Frame* frame = getFrame(parentFrame(), m_element);
- if (frame) {
- const CString cookieStr = cookies(frame->document(), u).utf8();
- if (!cookieStr.isNull()) {
- const int size = cookieStr.length();
- *value = static_cast<char*>(NPN_MemAlloc(size+1));
- if (*value) {
- memset(*value, 0, size+1);
- memcpy(*value, cookieStr.data(), size+1);
- if (len)
- *len = size;
- } else
- result = NPERR_OUT_OF_MEMORY_ERROR;
- }
- }
- } else
- result = NPERR_INVALID_URL;
- break;
- }
- case NPNURLVProxy: {
- KURL u(m_parentFrame->document()->baseURL(), url);
- if (u.isValid()) {
- Frame* frame = getFrame(parentFrame(), m_element);
- const FrameLoader* frameLoader = frame ? frame->loader() : 0;
- const NetworkingContext* context = frameLoader ? frameLoader->networkingContext() : 0;
- const CString proxyStr = toString(proxyServersForURL(u, context)).utf8();
- if (!proxyStr.isNull()) {
- const int size = proxyStr.length();
- *value = static_cast<char*>(NPN_MemAlloc(size+1));
- if (*value) {
- memset(*value, 0, size+1);
- memcpy(*value, proxyStr.data(), size+1);
- if (len)
- *len = size;
- } else
- result = NPERR_OUT_OF_MEMORY_ERROR;
- }
- } else
- result = NPERR_INVALID_URL;
- break;
- }
- default:
- result = NPERR_GENERIC_ERROR;
- LOG(Plugins, "PluginView::getValueForURL: %s", prettyNameForNPNURLVariable(variable).data());
- break;
- }
- return result;
- }
- NPError PluginView::setValueForURL(NPNURLVariable variable, const char* url, const char* value, uint32_t len)
- {
- LOG(Plugins, "PluginView::setValueForURL(%s)", prettyNameForNPNURLVariable(variable).data());
- NPError result = NPERR_NO_ERROR;
- switch (variable) {
- case NPNURLVCookie: {
- KURL u(m_parentFrame->document()->baseURL(), url);
- if (u.isValid()) {
- const String cookieStr = String::fromUTF8(value, len);
- Frame* frame = getFrame(parentFrame(), m_element);
- if (frame && !cookieStr.isEmpty())
- setCookies(frame->document(), u, cookieStr);
- } else
- result = NPERR_INVALID_URL;
- break;
- }
- case NPNURLVProxy:
- LOG(Plugins, "PluginView::setValueForURL(%s): Plugins are NOT allowed to set proxy information.", prettyNameForNPNURLVariable(variable).data());
- result = NPERR_GENERIC_ERROR;
- break;
- default:
- LOG(Plugins, "PluginView::setValueForURL: %s", prettyNameForNPNURLVariable(variable).data());
- result = NPERR_GENERIC_ERROR;
- break;
- }
- return result;
- }
- NPError PluginView::getAuthenticationInfo(const char* protocol, const char* host, int32_t port, const char* /* scheme */, const char* /* realm */, char** /* username */, uint32_t* /* ulen */, char** /* password */, uint32_t* /* plen */)
- {
- #if LOG_DISABLED
- UNUSED_PARAM(protocol);
- UNUSED_PARAM(host);
- UNUSED_PARAM(port);
- #endif
- LOG(Plugins, "PluginView::getAuthenticationInfo: protocol=%s, host=%s, port=%d", protocol, host, port);
- notImplemented();
- return NPERR_GENERIC_ERROR;
- }
- #endif
- void PluginView::privateBrowsingStateChanged(bool privateBrowsingEnabled)
- {
- NPP_SetValueProcPtr setValue = m_plugin->pluginFuncs()->setvalue;
- if (!setValue)
- return;
- PluginView::setCurrentPluginView(this);
- JSC::JSLock::DropAllLocks dropAllLocks(JSDOMWindowBase::commonVM());
- setCallingPlugin(true);
- NPBool value = privateBrowsingEnabled;
- setValue(m_instance, NPNVprivateModeBool, &value);
- setCallingPlugin(false);
- PluginView::setCurrentPluginView(0);
- }
- } // namespace WebCore
- #endif // ENABLE(NETSCAPE_PLUGIN_API)
|