diff options
author | Oswald Buddenhagen <oswald.buddenhagen@qt.io> | 2017-05-30 12:48:17 +0200 |
---|---|---|
committer | Oswald Buddenhagen <oswald.buddenhagen@qt.io> | 2017-05-30 12:48:17 +0200 |
commit | 881da28418d380042aa95a97f0cbd42560a64f7c (patch) | |
tree | a794dff3274695e99c651902dde93d934ea7a5af /Source/WebCore/html/HTMLPlugInImageElement.cpp | |
parent | 7e104c57a70fdf551bb3d22a5d637cdcbc69dbea (diff) | |
parent | 0fcedcd17cc00d3dd44c718b3cb36c1033319671 (diff) | |
download | qtwebkit-881da28418d380042aa95a97f0cbd42560a64f7c.tar.gz |
Merge 'wip/next' into dev
Change-Id: Iff9ee5e23bb326c4371ec8ed81d56f2f05d680e9
Diffstat (limited to 'Source/WebCore/html/HTMLPlugInImageElement.cpp')
-rw-r--r-- | Source/WebCore/html/HTMLPlugInImageElement.cpp | 470 |
1 files changed, 255 insertions, 215 deletions
diff --git a/Source/WebCore/html/HTMLPlugInImageElement.cpp b/Source/WebCore/html/HTMLPlugInImageElement.cpp index f85cbdc20..2a40da174 100644 --- a/Source/WebCore/html/HTMLPlugInImageElement.cpp +++ b/Source/WebCore/html/HTMLPlugInImageElement.cpp @@ -1,5 +1,5 @@ /* - * Copyright (C) 2008, 2011, 2012 Apple Inc. All rights reserved. + * Copyright (C) 2008, 2011, 2012, 2014 Apple Inc. All rights reserved. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public @@ -30,13 +30,14 @@ #include "FrameLoaderClient.h" #include "FrameView.h" #include "HTMLImageLoader.h" -#include "Image.h" #include "JSDocumentFragment.h" #include "LocalizedStrings.h" #include "Logging.h" +#include "MainFrame.h" #include "MouseEvent.h" #include "NodeList.h" #include "NodeRenderStyle.h" +#include "Page.h" #include "PlugInClient.h" #include "PluginViewBase.h" #include "RenderEmbeddedObject.h" @@ -48,6 +49,9 @@ #include "Settings.h" #include "ShadowRoot.h" #include "StyleResolver.h" +#include "StyleTreeResolver.h" +#include "SubframeLoader.h" +#include "TypedElementDescendantIterator.h" #include <JavaScriptCore/APICast.h> #include <JavaScriptCore/JSBase.h> #include <wtf/HashMap.h> @@ -57,7 +61,7 @@ namespace WebCore { using namespace HTMLNames; -typedef Vector<RefPtr<HTMLPlugInImageElement> > HTMLPlugInImageElementList; +typedef Vector<Ref<HTMLPlugInImageElement>> HTMLPlugInImageElementList; typedef HashMap<String, String> MimeTypeToLocalizedStringMap; static const int sizingTinyDimensionThreshold = 40; @@ -65,68 +69,70 @@ static const float sizingFullPageAreaRatioThreshold = 0.96; static const float autostartSoonAfterUserGestureThreshold = 5.0; // This delay should not exceed the snapshot delay in PluginView.cpp -static const double simulatedMouseClickTimerDelay = .75; -static const double removeSnapshotTimerDelay = 1.5; +static const auto simulatedMouseClickTimerDelay = std::chrono::milliseconds { 750 }; + +#if PLATFORM(COCOA) +static const auto removeSnapshotTimerDelay = std::chrono::milliseconds { 1500 }; +#endif static const String titleText(Page* page, String mimeType) { - DEFINE_STATIC_LOCAL(MimeTypeToLocalizedStringMap, mimeTypeToLabelTitleMap, ()); - String titleText = mimeTypeToLabelTitleMap.get(mimeType); + static NeverDestroyed<MimeTypeToLocalizedStringMap> mimeTypeToLabelTitleMap; + String titleText = mimeTypeToLabelTitleMap.get().get(mimeType); if (!titleText.isEmpty()) return titleText; - titleText = page->chrome().client()->plugInStartLabelTitle(mimeType); + titleText = page->chrome().client().plugInStartLabelTitle(mimeType); if (titleText.isEmpty()) titleText = snapshottedPlugInLabelTitle(); - mimeTypeToLabelTitleMap.set(mimeType, titleText); + mimeTypeToLabelTitleMap.get().set(mimeType, titleText); return titleText; }; static const String subtitleText(Page* page, String mimeType) { - DEFINE_STATIC_LOCAL(MimeTypeToLocalizedStringMap, mimeTypeToLabelSubtitleMap, ()); - String subtitleText = mimeTypeToLabelSubtitleMap.get(mimeType); + static NeverDestroyed<MimeTypeToLocalizedStringMap> mimeTypeToLabelSubtitleMap; + String subtitleText = mimeTypeToLabelSubtitleMap.get().get(mimeType); if (!subtitleText.isEmpty()) return subtitleText; - subtitleText = page->chrome().client()->plugInStartLabelSubtitle(mimeType); + subtitleText = page->chrome().client().plugInStartLabelSubtitle(mimeType); if (subtitleText.isEmpty()) subtitleText = snapshottedPlugInLabelSubtitle(); - mimeTypeToLabelSubtitleMap.set(mimeType, subtitleText); + mimeTypeToLabelSubtitleMap.get().set(mimeType, subtitleText); return subtitleText; }; -HTMLPlugInImageElement::HTMLPlugInImageElement(const QualifiedName& tagName, Document* document, bool createdByParser, PreferPlugInsForImagesOption preferPlugInsForImagesOption) +HTMLPlugInImageElement::HTMLPlugInImageElement(const QualifiedName& tagName, Document& document, bool createdByParser) : HTMLPlugInElement(tagName, document) // m_needsWidgetUpdate(!createdByParser) allows HTMLObjectElement to delay // widget updates until after all children are parsed. For HTMLEmbedElement // this delay is unnecessary, but it is simpler to make both classes share // the same codepath in this class. , m_needsWidgetUpdate(!createdByParser) - , m_shouldPreferPlugInsForImages(preferPlugInsForImagesOption == ShouldPreferPlugInsForImages) , m_needsDocumentActivationCallbacks(false) - , m_simulatedMouseClickTimer(this, &HTMLPlugInImageElement::simulatedMouseClickTimerFired, simulatedMouseClickTimerDelay) - , m_swapRendererTimer(this, &HTMLPlugInImageElement::swapRendererTimerFired) - , m_removeSnapshotTimer(this, &HTMLPlugInImageElement::removeSnapshotTimerFired) + , m_simulatedMouseClickTimer(*this, &HTMLPlugInImageElement::simulatedMouseClickTimerFired, simulatedMouseClickTimerDelay) + , m_removeSnapshotTimer(*this, &HTMLPlugInImageElement::removeSnapshotTimerFired) , m_createdDuringUserGesture(ScriptController::processingUserGesture()) , m_isRestartedPlugin(false) , m_needsCheckForSizeChange(false) , m_plugInWasCreated(false) , m_deferredPromotionToPrimaryPlugIn(false) , m_snapshotDecision(SnapshotNotYetDecided) + , m_plugInDimensionsSpecified(false) { - setHasCustomStyleCallbacks(); + setHasCustomStyleResolveCallbacks(); } HTMLPlugInImageElement::~HTMLPlugInImageElement() { if (m_needsDocumentActivationCallbacks) - document()->unregisterForPageCacheSuspensionCallbacks(this); + document().unregisterForDocumentSuspensionCallbacks(this); } void HTMLPlugInImageElement::setDisplayState(DisplayState state) { -#if PLATFORM(MAC) +#if PLATFORM(COCOA) if (state == RestartingWithPendingMouseClick || state == Restarting) { m_isRestartedPlugin = true; m_snapshotDecision = NeverSnapshot; @@ -137,18 +143,13 @@ void HTMLPlugInImageElement::setDisplayState(DisplayState state) #endif HTMLPlugInElement::setDisplayState(state); - - if (state == DisplayingSnapshot) - m_swapRendererTimer.startOneShot(0); } RenderEmbeddedObject* HTMLPlugInImageElement::renderEmbeddedObject() const { // HTMLObjectElement and HTMLEmbedElement may return arbitrary renderers // when using fallback content. - if (!renderer() || !renderer()->isEmbeddedObject()) - return 0; - return toRenderEmbeddedObject(renderer()); + return is<RenderEmbeddedObject>(renderer()) ? downcast<RenderEmbeddedObject>(renderer()) : nullptr; } bool HTMLPlugInImageElement::isImageType() @@ -156,9 +157,9 @@ bool HTMLPlugInImageElement::isImageType() if (m_serviceType.isEmpty() && protocolIs(m_url, "data")) m_serviceType = mimeTypeFromDataURL(m_url); - if (Frame* frame = document()->frame()) { - KURL completedURL = document()->completeURL(m_url); - return frame->loader()->client()->objectContentType(completedURL, m_serviceType, shouldPreferPlugInsForImages()) == ObjectContentImage; + if (Frame* frame = document().frame()) { + URL completedURL = document().completeURL(m_url); + return frame->loader().client().objectContentType(completedURL, m_serviceType) == ObjectContentImage; } return Image::supportsType(m_serviceType); @@ -168,102 +169,122 @@ bool HTMLPlugInImageElement::isImageType() // depending on <param> values. bool HTMLPlugInImageElement::allowedToLoadFrameURL(const String& url) { - KURL completeURL = document()->completeURL(url); + URL completeURL = document().completeURL(url); if (contentFrame() && protocolIsJavaScript(completeURL) - && !document()->securityOrigin()->canAccess(contentDocument()->securityOrigin())) + && !document().securityOrigin()->canAccess(contentDocument()->securityOrigin())) return false; - return document()->frame()->isURLAllowed(completeURL); + return document().frame()->isURLAllowed(completeURL); } // We don't use m_url, or m_serviceType as they may not be the final values // that <object> uses depending on <param> values. bool HTMLPlugInImageElement::wouldLoadAsNetscapePlugin(const String& url, const String& serviceType) { - ASSERT(document()); - ASSERT(document()->frame()); - KURL completedURL; + ASSERT(document().frame()); + URL completedURL; if (!url.isEmpty()) - completedURL = document()->completeURL(url); + completedURL = document().completeURL(url); - FrameLoader* frameLoader = document()->frame()->loader(); - ASSERT(frameLoader); - if (frameLoader->client()->objectContentType(completedURL, serviceType, shouldPreferPlugInsForImages()) == ObjectContentNetscapePlugin) + FrameLoader& frameLoader = document().frame()->loader(); + if (frameLoader.client().objectContentType(completedURL, serviceType) == ObjectContentNetscapePlugin) return true; return false; } -RenderObject* HTMLPlugInImageElement::createRenderer(RenderArena* arena, RenderStyle* style) +RenderPtr<RenderElement> HTMLPlugInImageElement::createElementRenderer(Ref<RenderStyle>&& style, const RenderTreePosition& insertionPosition) { + ASSERT(!document().inPageCache()); + + if (displayState() >= PreparingPluginReplacement) + return HTMLPlugInElement::createElementRenderer(WTFMove(style), insertionPosition); + // Once a PlugIn Element creates its renderer, it needs to be told when the Document goes // inactive or reactivates so it can clear the renderer before going into the page cache. if (!m_needsDocumentActivationCallbacks) { m_needsDocumentActivationCallbacks = true; - document()->registerForPageCacheSuspensionCallbacks(this); + document().registerForDocumentSuspensionCallbacks(this); } if (displayState() == DisplayingSnapshot) { - RenderSnapshottedPlugIn* renderSnapshottedPlugIn = new (arena) RenderSnapshottedPlugIn(this); + auto renderSnapshottedPlugIn = createRenderer<RenderSnapshottedPlugIn>(*this, WTFMove(style)); renderSnapshottedPlugIn->updateSnapshot(m_snapshotImage); - return renderSnapshottedPlugIn; + return WTFMove(renderSnapshottedPlugIn); } // Fallback content breaks the DOM->Renderer class relationship of this // class and all superclasses because createObject won't necessarily - // return a RenderEmbeddedObject, RenderPart or even RenderWidget. + // return a RenderEmbeddedObject or RenderWidget. if (useFallbackContent()) - return RenderObject::createObject(this, style); + return RenderElement::createFor(*this, WTFMove(style)); - if (isImageType()) { - RenderImage* image = new (arena) RenderImage(this); - image->setImageResource(RenderImageResource::create()); - return image; - } + if (isImageType()) + return createRenderer<RenderImage>(*this, WTFMove(style)); - return new (arena) RenderEmbeddedObject(this); + return HTMLPlugInElement::createElementRenderer(WTFMove(style), insertionPosition); } -bool HTMLPlugInImageElement::willRecalcStyle(StyleChange) +bool HTMLPlugInImageElement::childShouldCreateRenderer(const Node& child) const { - // FIXME: Why is this necessary? Manual re-attach is almost always wrong. - if (!useFallbackContent() && needsWidgetUpdate() && renderer() && !isImageType() && (displayState() != DisplayingSnapshot)) - reattach(); - return true; + if (is<RenderSnapshottedPlugIn>(renderer()) && !hasShadowRootParent(child)) + return false; + + return HTMLPlugInElement::childShouldCreateRenderer(child); } -void HTMLPlugInImageElement::attach(const AttachContext& context) +bool HTMLPlugInImageElement::willRecalcStyle(Style::Change change) { - PostAttachCallbackDisabler disabler(this); - - bool isImage = isImageType(); - - if (!isImage) - queuePostAttachCallback(&HTMLPlugInImageElement::updateWidgetCallback, this); + // Make sure style recalcs scheduled by a child shadow tree don't trigger reconstruction and cause flicker. + if (change == Style::NoChange && styleChangeType() == NoStyleChange) + return true; - HTMLPlugInElement::attach(context); + // FIXME: There shoudn't be need to force render tree reconstruction here. + // It is only done because loading and load event dispatching is tied to render tree construction. + if (!useFallbackContent() && needsWidgetUpdate() && renderer() && !isImageType() && (displayState() != DisplayingSnapshot)) + setNeedsStyleRecalc(ReconstructRenderTree); + return true; +} - if (isImage && renderer() && !useFallbackContent()) { - if (!m_imageLoader) - m_imageLoader = adoptPtr(new HTMLImageLoader(this)); - m_imageLoader->updateFromElement(); +void HTMLPlugInImageElement::didAttachRenderers() +{ + if (!isImageType()) { + RefPtr<HTMLPlugInImageElement> element = this; + Style::queuePostResolutionCallback([element]{ + element->updateWidgetIfNecessary(); + }); + return; } + if (!renderer() || useFallbackContent()) + return; + + // Image load might complete synchronously and cause us to re-enter. + RefPtr<HTMLPlugInImageElement> element = this; + Style::queuePostResolutionCallback([element]{ + element->startLoadingImage(); + }); } -void HTMLPlugInImageElement::detach(const AttachContext& context) +void HTMLPlugInImageElement::willDetachRenderers() { - // FIXME: Because of the insanity that is HTMLPlugInImageElement::recalcStyle, + // FIXME: Because of the insanity that is HTMLPlugInImageElement::willRecalcStyle, // we can end up detaching during an attach() call, before we even have a - // renderer. In that case, don't mark the widget for update. - if (attached() && renderer() && !useFallbackContent()) + // renderer. In that case, don't mark the widget for update. + if (renderer() && !useFallbackContent()) { // Update the widget the next time we attach (detaching destroys the plugin). setNeedsWidgetUpdate(true); - HTMLPlugInElement::detach(context); + } + + Widget* widget = pluginWidget(PluginLoadingPolicy::DoNotLoad); + if (is<PluginViewBase>(widget)) + downcast<PluginViewBase>(*widget).willDetatchRenderer(); + + HTMLPlugInElement::willDetachRenderers(); } void HTMLPlugInImageElement::updateWidgetIfNecessary() { - document()->updateStyleIfNeeded(); + document().updateStyleIfNeeded(); if (!needsWidgetUpdate() || useFallbackContent() || isImageType()) return; @@ -288,47 +309,36 @@ void HTMLPlugInImageElement::finishParsingChildren() void HTMLPlugInImageElement::didMoveToNewDocument(Document* oldDocument) { if (m_needsDocumentActivationCallbacks) { - if (oldDocument) - oldDocument->unregisterForPageCacheSuspensionCallbacks(this); - document()->registerForPageCacheSuspensionCallbacks(this); + oldDocument->unregisterForDocumentSuspensionCallbacks(this); + document().registerForDocumentSuspensionCallbacks(this); } if (m_imageLoader) m_imageLoader->elementDidMoveToNewDocument(); + HTMLPlugInElement::didMoveToNewDocument(oldDocument); } -void HTMLPlugInImageElement::documentWillSuspendForPageCache() +void HTMLPlugInImageElement::prepareForDocumentSuspension() { - if (RenderStyle* renderStyle = this->renderStyle()) { - m_customStyleForPageCache = RenderStyle::clone(renderStyle); - m_customStyleForPageCache->setDisplay(NONE); - recalcStyle(Force); - } + if (renderer()) + Style::detachRenderTree(*this); - HTMLPlugInElement::documentWillSuspendForPageCache(); + HTMLPlugInElement::prepareForDocumentSuspension(); } -void HTMLPlugInImageElement::documentDidResumeFromPageCache() +void HTMLPlugInImageElement::resumeFromDocumentSuspension() { - if (m_customStyleForPageCache) { - m_customStyleForPageCache = 0; - recalcStyle(Force); - } - - HTMLPlugInElement::documentDidResumeFromPageCache(); -} + setNeedsStyleRecalc(ReconstructRenderTree); -PassRefPtr<RenderStyle> HTMLPlugInImageElement::customStyleForRenderer() -{ - if (!m_customStyleForPageCache) - return document()->ensureStyleResolver()->styleForElement(this); - return m_customStyleForPageCache; + HTMLPlugInElement::resumeFromDocumentSuspension(); } -void HTMLPlugInImageElement::updateWidgetCallback(Node* n, unsigned) +void HTMLPlugInImageElement::startLoadingImage() { - static_cast<HTMLPlugInImageElement*>(n)->updateWidgetIfNecessary(); + if (!m_imageLoader) + m_imageLoader = std::make_unique<HTMLImageLoader>(*this); + m_imageLoader->updateFromElement(); } void HTMLPlugInImageElement::updateSnapshot(PassRefPtr<Image> image) @@ -338,30 +348,32 @@ void HTMLPlugInImageElement::updateSnapshot(PassRefPtr<Image> image) m_snapshotImage = image; - if (renderer()->isSnapshottedPlugIn()) { - toRenderSnapshottedPlugIn(renderer())->updateSnapshot(image); + if (!renderer()) + return; + auto& renderer = *this->renderer(); + + if (is<RenderSnapshottedPlugIn>(renderer)) { + downcast<RenderSnapshottedPlugIn>(renderer).updateSnapshot(image); return; } - if (renderer()->isEmbeddedObject()) - renderer()->repaint(); + if (is<RenderEmbeddedObject>(renderer)) + renderer.repaint(); } -void HTMLPlugInImageElement::checkSnapshotStatus() +static DOMWrapperWorld& plugInImageElementIsolatedWorld() { - if (!renderer()->isSnapshottedPlugIn()) { - if (displayState() == Playing) - checkSizeChangeForSnapshotting(); - return; - } - - // Notify the shadow root that the size changed so that we may update the overlay layout. - ensureUserAgentShadowRoot()->dispatchEvent(Event::create(eventNames().resizeEvent, true, false)); + static DOMWrapperWorld& isolatedWorld = DOMWrapperWorld::create(JSDOMWindow::commonVM()).leakRef(); + return isolatedWorld; } void HTMLPlugInImageElement::didAddUserAgentShadowRoot(ShadowRoot* root) { - Page* page = document()->page(); + HTMLPlugInElement::didAddUserAgentShadowRoot(root); + if (displayState() >= PreparingPluginReplacement) + return; + + Page* page = document().page(); if (!page) return; @@ -372,11 +384,11 @@ void HTMLPlugInImageElement::didAddUserAgentShadowRoot(ShadowRoot* root) String mimeType = loadedMimeType(); - DEFINE_STATIC_LOCAL(RefPtr<DOMWrapperWorld>, isolatedWorld, (DOMWrapperWorld::create(JSDOMWindow::commonVM()))); - document()->ensurePlugInsInjectedScript(isolatedWorld.get()); + DOMWrapperWorld& isolatedWorld = plugInImageElementIsolatedWorld(); + document().ensurePlugInsInjectedScript(isolatedWorld); - ScriptController* scriptController = page->mainFrame()->script(); - JSDOMGlobalObject* globalObject = JSC::jsCast<JSDOMGlobalObject*>(scriptController->globalObject(isolatedWorld.get())); + ScriptController& scriptController = document().frame()->script(); + JSDOMGlobalObject* globalObject = JSC::jsCast<JSDOMGlobalObject*>(scriptController.globalObject(isolatedWorld)); JSC::ExecState* exec = globalObject->globalExec(); JSC::JSLockHolder lock(exec); @@ -391,36 +403,27 @@ void HTMLPlugInImageElement::didAddUserAgentShadowRoot(ShadowRoot* root) argList.append(JSC::jsBoolean(!m_snapshotImage)); // It is expected the JS file provides a createOverlay(shadowRoot, title, subtitle) function. - JSC::JSObject* overlay = globalObject->get(exec, JSC::Identifier(exec, "createOverlay")).toObject(exec); + JSC::JSObject* overlay = globalObject->get(exec, JSC::Identifier::fromString(exec, "createOverlay")).toObject(exec); JSC::CallData callData; JSC::CallType callType = overlay->methodTable()->getCallData(overlay, callData); if (callType == JSC::CallTypeNone) return; - JSC::JSObject* thisObj = globalObject->methodTable()->toThisObject(globalObject, exec); - - JSC::call(exec, overlay, callType, callData, thisObj, argList); + JSC::call(exec, overlay, callType, callData, globalObject, argList); + exec->clearException(); } -bool HTMLPlugInImageElement::partOfSnapshotOverlay(Node* node) +bool HTMLPlugInImageElement::partOfSnapshotOverlay(const Node* node) const { - DEFINE_STATIC_LOCAL(AtomicString, selector, (".snapshot-overlay", AtomicString::ConstructFromLiteral)); - RefPtr<Element> snapshotLabel = ensureUserAgentShadowRoot()->querySelector(selector, ASSERT_NO_EXCEPTION); + static NeverDestroyed<AtomicString> selector(".snapshot-overlay", AtomicString::ConstructFromLiteral); + ShadowRoot* shadow = userAgentShadowRoot(); + if (!shadow) + return false; + RefPtr<Element> snapshotLabel = shadow->querySelector(selector.get(), ASSERT_NO_EXCEPTION); return node && snapshotLabel && (node == snapshotLabel.get() || node->isDescendantOf(snapshotLabel.get())); } -void HTMLPlugInImageElement::swapRendererTimerFired(Timer<HTMLPlugInImageElement>*) -{ - ASSERT(displayState() == DisplayingSnapshot); - if (userAgentShadowRoot()) - return; - - // Create a shadow root, which will trigger the code to add a snapshot container - // and reattach, thus making a new Renderer. - ensureUserAgentShadowRoot(); -} - -void HTMLPlugInImageElement::removeSnapshotTimerFired(Timer<HTMLPlugInImageElement>*) +void HTMLPlugInImageElement::removeSnapshotTimerFired() { m_snapshotImage = nullptr; m_isRestartedPlugin = false; @@ -429,23 +432,6 @@ void HTMLPlugInImageElement::removeSnapshotTimerFired(Timer<HTMLPlugInImageEleme renderer()->repaint(); } -static void addPlugInsFromNodeListMatchingPlugInOrigin(HTMLPlugInImageElementList& plugInList, PassRefPtr<NodeList> collection, const String& plugInOrigin, const String& mimeType) -{ - for (unsigned i = 0, length = collection->length(); i < length; i++) { - Node* node = collection->item(i); - if (node->isPluginElement()) { - HTMLPlugInElement* plugInElement = toHTMLPlugInElement(node); - if (plugInElement->isPlugInImageElement()) { - HTMLPlugInImageElement* plugInImageElement = toHTMLPlugInImageElement(node); - const KURL& loadedURL = plugInImageElement->loadedUrl(); - String otherMimeType = plugInImageElement->loadedMimeType(); - if (plugInOrigin == loadedURL.host() && mimeType == otherMimeType) - plugInList.append(plugInImageElement); - } - } - } -} - void HTMLPlugInImageElement::restartSimilarPlugIns() { // Restart any other snapshotted plugins in the page with the same origin. Note that they @@ -455,29 +441,25 @@ void HTMLPlugInImageElement::restartSimilarPlugIns() String mimeType = loadedMimeType(); HTMLPlugInImageElementList similarPlugins; - if (!document()->page()) + if (!document().page()) return; - for (Frame* frame = document()->page()->mainFrame(); frame; frame = frame->tree()->traverseNext()) { - if (!frame->loader()->subframeLoader()->containsPlugins()) + for (Frame* frame = &document().page()->mainFrame(); frame; frame = frame->tree().traverseNext()) { + if (!frame->loader().subframeLoader().containsPlugins()) continue; if (!frame->document()) continue; - RefPtr<NodeList> plugIns = frame->document()->getElementsByTagName(embedTag.localName()); - if (plugIns) - addPlugInsFromNodeListMatchingPlugInOrigin(similarPlugins, plugIns, plugInOrigin, mimeType); - - plugIns = frame->document()->getElementsByTagName(objectTag.localName()); - if (plugIns) - addPlugInsFromNodeListMatchingPlugInOrigin(similarPlugins, plugIns, plugInOrigin, mimeType); + for (auto& element : descendantsOfType<HTMLPlugInImageElement>(*frame->document())) { + if (plugInOrigin == element.loadedUrl().host() && mimeType == element.loadedMimeType()) + similarPlugins.append(element); + } } - for (size_t i = 0, length = similarPlugins.size(); i < length; ++i) { - HTMLPlugInImageElement* plugInToRestart = similarPlugins[i].get(); + for (auto& plugInToRestart : similarPlugins) { if (plugInToRestart->displayState() <= HTMLPlugInElement::DisplayingSnapshot) { - LOG(Plugins, "%p Plug-in looks similar to a restarted plug-in. Restart.", plugInToRestart); + LOG(Plugins, "%p Plug-in looks similar to a restarted plug-in. Restart.", plugInToRestart.ptr()); plugInToRestart->restartSnapshottedPlugIn(); } plugInToRestart->m_snapshotDecision = NeverSnapshot; @@ -490,19 +472,19 @@ void HTMLPlugInImageElement::userDidClickSnapshot(PassRefPtr<MouseEvent> event, m_pendingClickEventFromSnapshot = event; String plugInOrigin = m_loadedUrl.host(); - if (document()->page() && !SchemeRegistry::shouldTreatURLSchemeAsLocal(document()->page()->mainFrame()->document()->baseURL().protocol()) && document()->page()->settings()->autostartOriginPlugInSnapshottingEnabled()) - document()->page()->plugInClient()->didStartFromOrigin(document()->page()->mainFrame()->document()->baseURL().host(), plugInOrigin, loadedMimeType()); + if (document().page() && !SchemeRegistry::shouldTreatURLSchemeAsLocal(document().page()->mainFrame().document()->baseURL().protocol()) && document().page()->settings().autostartOriginPlugInSnapshottingEnabled()) + document().page()->plugInClient()->didStartFromOrigin(document().page()->mainFrame().document()->baseURL().host(), plugInOrigin, loadedMimeType(), document().page()->sessionID()); LOG(Plugins, "%p User clicked on snapshotted plug-in. Restart.", this); restartSnapshottedPlugIn(); if (forwardEvent) - setDisplayState(HTMLPlugInElement::RestartingWithPendingMouseClick); + setDisplayState(RestartingWithPendingMouseClick); restartSimilarPlugIns(); } void HTMLPlugInImageElement::setIsPrimarySnapshottedPlugIn(bool isPrimarySnapshottedPlugIn) { - if (!document()->page() || !document()->page()->settings()->primaryPlugInSnapshotDetectionEnabled() || document()->page()->settings()->snapshotAllPlugIns()) + if (!document().page() || !document().page()->settings().primaryPlugInSnapshotDetectionEnabled() || document().page()->settings().snapshotAllPlugIns()) return; if (isPrimarySnapshottedPlugIn) { @@ -523,7 +505,7 @@ void HTMLPlugInImageElement::restartSnapshottedPlugIn() return; setDisplayState(Restarting); - reattach(); + setNeedsStyleRecalc(ReconstructRenderTree); } void HTMLPlugInImageElement::dispatchPendingMouseClick() @@ -532,7 +514,7 @@ void HTMLPlugInImageElement::dispatchPendingMouseClick() m_simulatedMouseClickTimer.restart(); } -void HTMLPlugInImageElement::simulatedMouseClickTimerFired(DeferrableOneShotTimer<HTMLPlugInImageElement>*) +void HTMLPlugInImageElement::simulatedMouseClickTimerFired() { ASSERT(displayState() == RestartingWithPendingMouseClick); ASSERT(m_pendingClickEventFromSnapshot); @@ -543,14 +525,14 @@ void HTMLPlugInImageElement::simulatedMouseClickTimerFired(DeferrableOneShotTime m_pendingClickEventFromSnapshot = nullptr; } -static bool documentHadRecentUserGesture(Document* document) +static bool documentHadRecentUserGesture(Document& document) { - double lastKnownUserGestureTimestamp = document->lastHandledUserGestureTimestamp(); + double lastKnownUserGestureTimestamp = document.lastHandledUserGestureTimestamp(); - if (document->frame() != document->page()->mainFrame() && document->page()->mainFrame() && document->page()->mainFrame()->document()) - lastKnownUserGestureTimestamp = std::max(lastKnownUserGestureTimestamp, document->page()->mainFrame()->document()->lastHandledUserGestureTimestamp()); + if (document.frame() != &document.page()->mainFrame() && document.page()->mainFrame().document()) + lastKnownUserGestureTimestamp = std::max(lastKnownUserGestureTimestamp, document.page()->mainFrame().document()->lastHandledUserGestureTimestamp()); - if (currentTime() - lastKnownUserGestureTimestamp < autostartSoonAfterUserGestureThreshold) + if (monotonicallyIncreasingTime() - lastKnownUserGestureTimestamp < autostartSoonAfterUserGestureThreshold) return true; return false; @@ -562,7 +544,7 @@ void HTMLPlugInImageElement::checkSizeChangeForSnapshotting() return; m_needsCheckForSizeChange = false; - LayoutRect contentBoxRect = toRenderBox(renderer())->contentBoxRect(); + LayoutRect contentBoxRect = downcast<RenderBox>(*renderer()).contentBoxRect(); int contentWidth = contentBoxRect.width(); int contentHeight = contentBoxRect.height(); @@ -573,11 +555,65 @@ void HTMLPlugInImageElement::checkSizeChangeForSnapshotting() setDisplayState(WaitingForSnapshot); m_snapshotDecision = Snapshotted; Widget* widget = pluginWidget(); - if (widget && widget->isPluginViewBase()) - toPluginViewBase(widget)->beginSnapshottingRunningPlugin(); + if (is<PluginViewBase>(widget)) + downcast<PluginViewBase>(*widget).beginSnapshottingRunningPlugin(); } -void HTMLPlugInImageElement::subframeLoaderWillCreatePlugIn(const KURL& url) +static inline bool is100Percent(Length length) +{ + return length.isPercent() && length.percent() == 100; +} + +static inline bool isSmallerThanTinySizingThreshold(const RenderEmbeddedObject& renderer) +{ + LayoutRect contentRect = renderer.contentBoxRect(); + return contentRect.width() <= sizingTinyDimensionThreshold || contentRect.height() <= sizingTinyDimensionThreshold; +} + +bool HTMLPlugInImageElement::isTopLevelFullPagePlugin(const RenderEmbeddedObject& renderer) const +{ + Frame& frame = *document().frame(); + if (!frame.isMainFrame()) + return false; + + auto& style = renderer.style(); + IntSize visibleSize = frame.view()->visibleSize(); + LayoutRect contentRect = renderer.contentBoxRect(); + int contentWidth = contentRect.width(); + int contentHeight = contentRect.height(); + return is100Percent(style.width()) && is100Percent(style.height()) && contentWidth * contentHeight > visibleSize.area() * sizingFullPageAreaRatioThreshold; +} + +void HTMLPlugInImageElement::checkSnapshotStatus() +{ + if (!is<RenderSnapshottedPlugIn>(*renderer())) { + if (displayState() == Playing) + checkSizeChangeForSnapshotting(); + return; + } + + // If width and height styles were previously not set and we've snapshotted the plugin we may need to restart the plugin so that its state can be updated appropriately. + if (!document().page()->settings().snapshotAllPlugIns() && displayState() <= DisplayingSnapshot && !m_plugInDimensionsSpecified) { + RenderSnapshottedPlugIn& renderer = downcast<RenderSnapshottedPlugIn>(*this->renderer()); + if (!renderer.style().logicalWidth().isSpecified() && !renderer.style().logicalHeight().isSpecified()) + return; + + m_plugInDimensionsSpecified = true; + if (isTopLevelFullPagePlugin(renderer)) { + m_snapshotDecision = NeverSnapshot; + restartSnapshottedPlugIn(); + } else if (isSmallerThanTinySizingThreshold(renderer)) { + m_snapshotDecision = MaySnapshotWhenResized; + restartSnapshottedPlugIn(); + } + return; + } + + // Notify the shadow root that the size changed so that we may update the overlay layout. + ensureUserAgentShadowRoot().dispatchEvent(Event::create(eventNames().resizeEvent, true, false)); +} + +void HTMLPlugInImageElement::subframeLoaderWillCreatePlugIn(const URL& url) { LOG(Plugins, "%p Plug-in URL: %s", this, m_url.utf8().data()); LOG(Plugins, " Actual URL: %s", url.string().utf8().data()); @@ -587,7 +623,7 @@ void HTMLPlugInImageElement::subframeLoaderWillCreatePlugIn(const KURL& url) m_plugInWasCreated = false; m_deferredPromotionToPrimaryPlugIn = false; - if (!document()->page() || !document()->page()->settings()->plugInSnapshottingEnabled()) { + if (!document().page() || !document().page()->settings().plugInSnapshottingEnabled()) { m_snapshotDecision = NeverSnapshot; return; } @@ -610,9 +646,9 @@ void HTMLPlugInImageElement::subframeLoaderWillCreatePlugIn(const KURL& url) return; } - bool inMainFrame = document()->frame() == document()->page()->mainFrame(); + bool inMainFrame = document().frame()->isMainFrame(); - if (document()->isPluginDocument() && inMainFrame) { + if (document().isPluginDocument() && inMainFrame) { LOG(Plugins, "%p Plug-in document in main frame", this); m_snapshotDecision = NeverSnapshot; return; @@ -636,15 +672,15 @@ void HTMLPlugInImageElement::subframeLoaderWillCreatePlugIn(const KURL& url) return; } - if (document()->page()->settings()->snapshotAllPlugIns()) { + if (document().page()->settings().snapshotAllPlugIns()) { LOG(Plugins, "%p Plug-in forced to snapshot by user preference", this); m_snapshotDecision = Snapshotted; setDisplayState(WaitingForSnapshot); return; } - if (document()->page()->settings()->autostartOriginPlugInSnapshottingEnabled() && document()->page()->plugInClient() && document()->page()->plugInClient()->shouldAutoStartFromOrigin(document()->page()->mainFrame()->document()->baseURL().host(), url.host(), loadedMimeType())) { - LOG(Plugins, "%p Plug-in from (%s, %s) is marked to auto-start, set to play", this, document()->page()->mainFrame()->document()->baseURL().host().utf8().data(), url.host().utf8().data()); + if (document().page()->settings().autostartOriginPlugInSnapshottingEnabled() && document().page()->plugInClient() && document().page()->plugInClient()->shouldAutoStartFromOrigin(document().page()->mainFrame().document()->baseURL().host(), url.host(), loadedMimeType())) { + LOG(Plugins, "%p Plug-in from (%s, %s) is marked to auto-start, set to play", this, document().page()->mainFrame().document()->baseURL().host().utf8().data(), url.host().utf8().data()); m_snapshotDecision = NeverSnapshot; return; } @@ -655,54 +691,49 @@ void HTMLPlugInImageElement::subframeLoaderWillCreatePlugIn(const KURL& url) return; } - if (!SchemeRegistry::shouldTreatURLSchemeAsLocal(m_loadedUrl.protocol()) && !m_loadedUrl.host().isEmpty() && m_loadedUrl.host() == document()->page()->mainFrame()->document()->baseURL().host()) { + if (!SchemeRegistry::shouldTreatURLSchemeAsLocal(m_loadedUrl.protocol()) && !m_loadedUrl.host().isEmpty() && m_loadedUrl.host() == document().page()->mainFrame().document()->baseURL().host()) { LOG(Plugins, "%p Plug-in is served from page's domain, set to play", this); m_snapshotDecision = NeverSnapshot; return; } - - RenderBox* renderEmbeddedObject = toRenderBox(renderer()); - Length styleWidth = renderEmbeddedObject->style()->width(); - Length styleHeight = renderEmbeddedObject->style()->height(); - LayoutRect contentBoxRect = renderEmbeddedObject->contentBoxRect(); - int contentWidth = contentBoxRect.width(); - int contentHeight = contentBoxRect.height(); - int contentArea = contentWidth * contentHeight; - IntSize visibleViewSize = document()->frame()->view()->visibleSize(); - int visibleArea = visibleViewSize.width() * visibleViewSize.height(); - - if (inMainFrame && styleWidth.isPercent() && (styleWidth.percent() == 100) - && styleHeight.isPercent() && (styleHeight.percent() == 100) - && (static_cast<float>(contentArea) / visibleArea > sizingFullPageAreaRatioThreshold)) { + + auto& renderer = downcast<RenderEmbeddedObject>(*this->renderer()); + LayoutRect contentRect = renderer.contentBoxRect(); + int contentWidth = contentRect.width(); + int contentHeight = contentRect.height(); + + m_plugInDimensionsSpecified = renderer.style().logicalWidth().isSpecified() || renderer.style().logicalHeight().isSpecified(); + + if (isTopLevelFullPagePlugin(renderer)) { LOG(Plugins, "%p Plug-in is top level full page, set to play", this); m_snapshotDecision = NeverSnapshot; return; } - if (contentWidth <= sizingTinyDimensionThreshold || contentHeight <= sizingTinyDimensionThreshold) { + if (isSmallerThanTinySizingThreshold(renderer)) { LOG(Plugins, "%p Plug-in is very small %dx%d, set to play", this, contentWidth, contentHeight); - m_sizeWhenSnapshotted = IntSize(contentBoxRect.width().toInt(), contentBoxRect.height().toInt()); + m_sizeWhenSnapshotted = IntSize(contentWidth, contentHeight); m_snapshotDecision = MaySnapshotWhenResized; return; } - if (!document()->page()->plugInClient()) { + if (!document().page()->plugInClient()) { LOG(Plugins, "%p There is no plug-in client. Set to wait for snapshot", this); m_snapshotDecision = NeverSnapshot; setDisplayState(WaitingForSnapshot); return; } - LOG(Plugins, "%p Plug-in from (%s, %s) is not auto-start, sized at %dx%d, set to wait for snapshot", this, document()->page()->mainFrame()->document()->baseURL().host().utf8().data(), url.host().utf8().data(), contentWidth, contentHeight); + LOG(Plugins, "%p Plug-in from (%s, %s) is not auto-start, sized at %dx%d, set to wait for snapshot", this, document().topDocument().baseURL().host().utf8().data(), url.host().utf8().data(), contentWidth, contentHeight); m_snapshotDecision = Snapshotted; setDisplayState(WaitingForSnapshot); } -void HTMLPlugInImageElement::subframeLoaderDidCreatePlugIn(const Widget* widget) +void HTMLPlugInImageElement::subframeLoaderDidCreatePlugIn(const Widget& widget) { m_plugInWasCreated = true; - if (widget->isPluginViewBase() && toPluginViewBase(widget)->shouldAlwaysAutoStart()) { + if (is<PluginViewBase>(widget) && downcast<PluginViewBase>(widget).shouldAlwaysAutoStart()) { LOG(Plugins, "%p Plug-in should auto-start, set to play", this); m_snapshotDecision = NeverSnapshot; setDisplayState(Playing); @@ -718,13 +749,13 @@ void HTMLPlugInImageElement::subframeLoaderDidCreatePlugIn(const Widget* widget) void HTMLPlugInImageElement::defaultEventHandler(Event* event) { - RenderObject* r = renderer(); + RenderElement* r = renderer(); if (r && r->isEmbeddedObject()) { - if (isPlugInImageElement() && displayState() == WaitingForSnapshot && event->isMouseEvent() && event->type() == eventNames().clickEvent) { - MouseEvent* mouseEvent = toMouseEvent(event); - if (mouseEvent->button() == LeftButton) { - userDidClickSnapshot(mouseEvent, true); - event->setDefaultHandled(); + if (displayState() == WaitingForSnapshot && is<MouseEvent>(*event) && event->type() == eventNames().clickEvent) { + MouseEvent& mouseEvent = downcast<MouseEvent>(*event); + if (mouseEvent.button() == LeftButton) { + userDidClickSnapshot(&mouseEvent, true); + mouseEvent.setDefaultHandled(); return; } } @@ -732,4 +763,13 @@ void HTMLPlugInImageElement::defaultEventHandler(Event* event) HTMLPlugInElement::defaultEventHandler(event); } +bool HTMLPlugInImageElement::requestObject(const String& url, const String& mimeType, const Vector<String>& paramNames, const Vector<String>& paramValues) +{ + if (HTMLPlugInElement::requestObject(url, mimeType, paramNames, paramValues)) + return true; + + SubframeLoader& loader = document().frame()->loader().subframeLoader(); + return loader.requestObject(*this, url, getNameAttribute(), mimeType, paramNames, paramValues); +} + } // namespace WebCore |