diff options
author | Lorry Tar Creator <lorry-tar-importer@lorry> | 2016-04-10 09:28:39 +0000 |
---|---|---|
committer | Lorry Tar Creator <lorry-tar-importer@lorry> | 2016-04-10 09:28:39 +0000 |
commit | 32761a6cee1d0dee366b885b7b9c777e67885688 (patch) | |
tree | d6bec92bebfb216f4126356e55518842c2f476a1 /Source/WebCore/html/HTMLPlugInImageElement.cpp | |
parent | a4e969f4965059196ca948db781e52f7cfebf19e (diff) | |
download | WebKitGtk-tarball-32761a6cee1d0dee366b885b7b9c777e67885688.tar.gz |
webkitgtk-2.4.11webkitgtk-2.4.11
Diffstat (limited to 'Source/WebCore/html/HTMLPlugInImageElement.cpp')
-rw-r--r-- | Source/WebCore/html/HTMLPlugInImageElement.cpp | 340 |
1 files changed, 171 insertions, 169 deletions
diff --git a/Source/WebCore/html/HTMLPlugInImageElement.cpp b/Source/WebCore/html/HTMLPlugInImageElement.cpp index 2a40da174..18ef20628 100644 --- a/Source/WebCore/html/HTMLPlugInImageElement.cpp +++ b/Source/WebCore/html/HTMLPlugInImageElement.cpp @@ -1,5 +1,5 @@ /* - * Copyright (C) 2008, 2011, 2012, 2014 Apple Inc. All rights reserved. + * Copyright (C) 2008, 2011, 2012 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 @@ -49,19 +49,23 @@ #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> #include <wtf/text/StringHash.h> +#if PLATFORM(IOS) +#include "HTMLIFrameElement.h" +#include "RenderBlockFlow.h" +#include "YouTubeEmbedShadowElement.h" +#endif + namespace WebCore { using namespace HTMLNames; -typedef Vector<Ref<HTMLPlugInImageElement>> HTMLPlugInImageElementList; +typedef Vector<RefPtr<HTMLPlugInImageElement>> HTMLPlugInImageElementList; typedef HashMap<String, String> MimeTypeToLocalizedStringMap; static const int sizingTinyDimensionThreshold = 40; @@ -69,57 +73,54 @@ 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 auto simulatedMouseClickTimerDelay = std::chrono::milliseconds { 750 }; - -#if PLATFORM(COCOA) -static const auto removeSnapshotTimerDelay = std::chrono::milliseconds { 1500 }; -#endif +static const double simulatedMouseClickTimerDelay = .75; +static const double removeSnapshotTimerDelay = 1.5; static const String titleText(Page* page, String mimeType) { - static NeverDestroyed<MimeTypeToLocalizedStringMap> mimeTypeToLabelTitleMap; - String titleText = mimeTypeToLabelTitleMap.get().get(mimeType); + DEFINE_STATIC_LOCAL(MimeTypeToLocalizedStringMap, mimeTypeToLabelTitleMap, ()); + String titleText = mimeTypeToLabelTitleMap.get(mimeType); if (!titleText.isEmpty()) return titleText; titleText = page->chrome().client().plugInStartLabelTitle(mimeType); if (titleText.isEmpty()) titleText = snapshottedPlugInLabelTitle(); - mimeTypeToLabelTitleMap.get().set(mimeType, titleText); + mimeTypeToLabelTitleMap.set(mimeType, titleText); return titleText; }; static const String subtitleText(Page* page, String mimeType) { - static NeverDestroyed<MimeTypeToLocalizedStringMap> mimeTypeToLabelSubtitleMap; - String subtitleText = mimeTypeToLabelSubtitleMap.get().get(mimeType); + DEFINE_STATIC_LOCAL(MimeTypeToLocalizedStringMap, mimeTypeToLabelSubtitleMap, ()); + String subtitleText = mimeTypeToLabelSubtitleMap.get(mimeType); if (!subtitleText.isEmpty()) return subtitleText; subtitleText = page->chrome().client().plugInStartLabelSubtitle(mimeType); if (subtitleText.isEmpty()) subtitleText = snapshottedPlugInLabelSubtitle(); - mimeTypeToLabelSubtitleMap.get().set(mimeType, subtitleText); + mimeTypeToLabelSubtitleMap.set(mimeType, subtitleText); return subtitleText; }; -HTMLPlugInImageElement::HTMLPlugInImageElement(const QualifiedName& tagName, Document& document, bool createdByParser) +HTMLPlugInImageElement::HTMLPlugInImageElement(const QualifiedName& tagName, Document& document, bool createdByParser, PreferPlugInsForImagesOption preferPlugInsForImagesOption) : 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_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) { setHasCustomStyleResolveCallbacks(); } @@ -127,12 +128,12 @@ HTMLPlugInImageElement::HTMLPlugInImageElement(const QualifiedName& tagName, Doc HTMLPlugInImageElement::~HTMLPlugInImageElement() { if (m_needsDocumentActivationCallbacks) - document().unregisterForDocumentSuspensionCallbacks(this); + document().unregisterForPageCacheSuspensionCallbacks(this); } void HTMLPlugInImageElement::setDisplayState(DisplayState state) { -#if PLATFORM(COCOA) +#if PLATFORM(MAC) if (state == RestartingWithPendingMouseClick || state == Restarting) { m_isRestartedPlugin = true; m_snapshotDecision = NeverSnapshot; @@ -149,7 +150,9 @@ RenderEmbeddedObject* HTMLPlugInImageElement::renderEmbeddedObject() const { // HTMLObjectElement and HTMLEmbedElement may return arbitrary renderers // when using fallback content. - return is<RenderEmbeddedObject>(renderer()) ? downcast<RenderEmbeddedObject>(renderer()) : nullptr; + if (!renderer() || !renderer()->isEmbeddedObject()) + return 0; + return toRenderEmbeddedObject(renderer()); } bool HTMLPlugInImageElement::isImageType() @@ -159,7 +162,7 @@ bool HTMLPlugInImageElement::isImageType() if (Frame* frame = document().frame()) { URL completedURL = document().completeURL(m_url); - return frame->loader().client().objectContentType(completedURL, m_serviceType) == ObjectContentImage; + return frame->loader().client().objectContentType(completedURL, m_serviceType, shouldPreferPlugInsForImages()) == ObjectContentImage; } return Image::supportsType(m_serviceType); @@ -188,57 +191,52 @@ bool HTMLPlugInImageElement::wouldLoadAsNetscapePlugin(const String& url, const completedURL = document().completeURL(url); FrameLoader& frameLoader = document().frame()->loader(); - if (frameLoader.client().objectContentType(completedURL, serviceType) == ObjectContentNetscapePlugin) + if (frameLoader.client().objectContentType(completedURL, serviceType, shouldPreferPlugInsForImages()) == ObjectContentNetscapePlugin) return true; return false; } -RenderPtr<RenderElement> HTMLPlugInImageElement::createElementRenderer(Ref<RenderStyle>&& style, const RenderTreePosition& insertionPosition) +RenderPtr<RenderElement> HTMLPlugInImageElement::createElementRenderer(PassRef<RenderStyle> style) { ASSERT(!document().inPageCache()); if (displayState() >= PreparingPluginReplacement) - return HTMLPlugInElement::createElementRenderer(WTFMove(style), insertionPosition); + return HTMLPlugInElement::createElementRenderer(std::move(style)); // 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().registerForDocumentSuspensionCallbacks(this); + document().registerForPageCacheSuspensionCallbacks(this); } if (displayState() == DisplayingSnapshot) { - auto renderSnapshottedPlugIn = createRenderer<RenderSnapshottedPlugIn>(*this, WTFMove(style)); + auto renderSnapshottedPlugIn = createRenderer<RenderSnapshottedPlugIn>(*this, std::move(style)); renderSnapshottedPlugIn->updateSnapshot(m_snapshotImage); - return WTFMove(renderSnapshottedPlugIn); + return std::move(renderSnapshottedPlugIn); } // Fallback content breaks the DOM->Renderer class relationship of this // class and all superclasses because createObject won't necessarily // return a RenderEmbeddedObject or RenderWidget. if (useFallbackContent()) - return RenderElement::createFor(*this, WTFMove(style)); + return RenderElement::createFor(*this, std::move(style)); if (isImageType()) - return createRenderer<RenderImage>(*this, WTFMove(style)); - - return HTMLPlugInElement::createElementRenderer(WTFMove(style), insertionPosition); -} + return createRenderer<RenderImage>(*this, std::move(style)); -bool HTMLPlugInImageElement::childShouldCreateRenderer(const Node& child) const -{ - if (is<RenderSnapshottedPlugIn>(renderer()) && !hasShadowRootParent(child)) - return false; - - return HTMLPlugInElement::childShouldCreateRenderer(child); +#if PLATFORM(IOS) + if (ShadowRoot* shadowRoot = this->shadowRoot()) { + Element* shadowElement = toElement(shadowRoot->firstChild()); + if (shadowElement && shadowElement->shadowPseudoId() == "-apple-youtube-shadow-iframe") + return createRenderer<RenderBlockFlow>(*this, std::move(style)); + } +#endif + return HTMLPlugInElement::createElementRenderer(std::move(style)); } -bool HTMLPlugInImageElement::willRecalcStyle(Style::Change change) +bool HTMLPlugInImageElement::willRecalcStyle(Style::Change) { - // 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; - // 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)) @@ -249,36 +247,25 @@ bool HTMLPlugInImageElement::willRecalcStyle(Style::Change change) void HTMLPlugInImageElement::didAttachRenderers() { if (!isImageType()) { - RefPtr<HTMLPlugInImageElement> element = this; - Style::queuePostResolutionCallback([element]{ - element->updateWidgetIfNecessary(); - }); + queuePostAttachCallback(&HTMLPlugInImageElement::updateWidgetCallback, *this); 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(); - }); + // Image load might complete synchronously and cause us to re-enter attach. + queuePostAttachCallback(&HTMLPlugInImageElement::startLoadingImageCallback, *this); } void HTMLPlugInImageElement::willDetachRenderers() { // 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. + // 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); } - - Widget* widget = pluginWidget(PluginLoadingPolicy::DoNotLoad); - if (is<PluginViewBase>(widget)) - downcast<PluginViewBase>(*widget).willDetatchRenderer(); - HTMLPlugInElement::willDetachRenderers(); } @@ -309,38 +296,48 @@ void HTMLPlugInImageElement::finishParsingChildren() void HTMLPlugInImageElement::didMoveToNewDocument(Document* oldDocument) { if (m_needsDocumentActivationCallbacks) { - oldDocument->unregisterForDocumentSuspensionCallbacks(this); - document().registerForDocumentSuspensionCallbacks(this); + if (oldDocument) + oldDocument->unregisterForPageCacheSuspensionCallbacks(this); + document().registerForPageCacheSuspensionCallbacks(this); } if (m_imageLoader) m_imageLoader->elementDidMoveToNewDocument(); - HTMLPlugInElement::didMoveToNewDocument(oldDocument); } -void HTMLPlugInImageElement::prepareForDocumentSuspension() +void HTMLPlugInImageElement::documentWillSuspendForPageCache() { if (renderer()) Style::detachRenderTree(*this); - HTMLPlugInElement::prepareForDocumentSuspension(); + HTMLPlugInElement::documentWillSuspendForPageCache(); } -void HTMLPlugInImageElement::resumeFromDocumentSuspension() +void HTMLPlugInImageElement::documentDidResumeFromPageCache() { setNeedsStyleRecalc(ReconstructRenderTree); - HTMLPlugInElement::resumeFromDocumentSuspension(); + HTMLPlugInElement::documentDidResumeFromPageCache(); +} + +void HTMLPlugInImageElement::updateWidgetCallback(Node& node, unsigned) +{ + toHTMLPlugInImageElement(node).updateWidgetIfNecessary(); } void HTMLPlugInImageElement::startLoadingImage() { if (!m_imageLoader) - m_imageLoader = std::make_unique<HTMLImageLoader>(*this); + m_imageLoader = adoptPtr(new HTMLImageLoader(this)); m_imageLoader->updateFromElement(); } +void HTMLPlugInImageElement::startLoadingImageCallback(Node& node, unsigned) +{ + toHTMLPlugInImageElement(node).startLoadingImage(); +} + void HTMLPlugInImageElement::updateSnapshot(PassRefPtr<Image> image) { if (displayState() > DisplayingSnapshot) @@ -348,22 +345,30 @@ void HTMLPlugInImageElement::updateSnapshot(PassRefPtr<Image> image) m_snapshotImage = image; - if (!renderer()) + if (renderer()->isSnapshottedPlugIn()) { + toRenderSnapshottedPlugIn(renderer())->updateSnapshot(image); return; - auto& renderer = *this->renderer(); + } + + if (renderer()->isEmbeddedObject()) + renderer()->repaint(); +} - if (is<RenderSnapshottedPlugIn>(renderer)) { - downcast<RenderSnapshottedPlugIn>(renderer).updateSnapshot(image); +void HTMLPlugInImageElement::checkSnapshotStatus() +{ + if (!renderer()->isSnapshottedPlugIn()) { + if (displayState() == Playing) + checkSizeChangeForSnapshotting(); return; } - if (is<RenderEmbeddedObject>(renderer)) - renderer.repaint(); + // 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& plugInImageElementIsolatedWorld() { - static DOMWrapperWorld& isolatedWorld = DOMWrapperWorld::create(JSDOMWindow::commonVM()).leakRef(); + static DOMWrapperWorld& isolatedWorld = *DOMWrapperWorld::create(JSDOMWindow::commonVM()).leakRef(); return isolatedWorld; } @@ -387,7 +392,7 @@ void HTMLPlugInImageElement::didAddUserAgentShadowRoot(ShadowRoot* root) DOMWrapperWorld& isolatedWorld = plugInImageElementIsolatedWorld(); document().ensurePlugInsInjectedScript(isolatedWorld); - ScriptController& scriptController = document().frame()->script(); + ScriptController& scriptController = page->mainFrame().script(); JSDOMGlobalObject* globalObject = JSC::jsCast<JSDOMGlobalObject*>(scriptController.globalObject(isolatedWorld)); JSC::ExecState* exec = globalObject->globalExec(); @@ -403,27 +408,52 @@ 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::fromString(exec, "createOverlay")).toObject(exec); + JSC::JSObject* overlay = globalObject->get(exec, JSC::Identifier(exec, "createOverlay")).toObject(exec); JSC::CallData callData; JSC::CallType callType = overlay->methodTable()->getCallData(overlay, callData); if (callType == JSC::CallTypeNone) return; JSC::call(exec, overlay, callType, callData, globalObject, argList); - exec->clearException(); } -bool HTMLPlugInImageElement::partOfSnapshotOverlay(const Node* node) const +bool HTMLPlugInImageElement::partOfSnapshotOverlay(Node* node) { - 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); + DEFINE_STATIC_LOCAL(AtomicString, selector, (".snapshot-overlay", AtomicString::ConstructFromLiteral)); + RefPtr<Element> snapshotLabel = ensureUserAgentShadowRoot().querySelector(selector, ASSERT_NO_EXCEPTION); return node && snapshotLabel && (node == snapshotLabel.get() || node->isDescendantOf(snapshotLabel.get())); } -void HTMLPlugInImageElement::removeSnapshotTimerFired() +#if PLATFORM(IOS) +void HTMLPlugInImageElement::createShadowIFrameSubtree(const String& src) +{ + if (this->shadowRoot()) + return; + + if (src.isEmpty()) + return; + + RefPtr<YouTubeEmbedShadowElement> shadowElement = YouTubeEmbedShadowElement::create(document()); + ShadowRoot& root = this->ensureUserAgentShadowRoot(); + root.appendChild(shadowElement, ASSERT_NO_EXCEPTION); + + RefPtr<HTMLIFrameElement> iframeElement = HTMLIFrameElement::create(HTMLNames::iframeTag, document()); + if (hasAttribute(HTMLNames::widthAttr)) + iframeElement->setAttribute(HTMLNames::widthAttr, AtomicString("100%", AtomicString::ConstructFromLiteral)); + if (hasAttribute(HTMLNames::heightAttr)) { + iframeElement->setAttribute(HTMLNames::styleAttr, AtomicString("max-height: 100%", AtomicString::ConstructFromLiteral)); + iframeElement->setAttribute(HTMLNames::heightAttr, getAttribute(HTMLNames::heightAttr)); + } + iframeElement->setAttribute(HTMLNames::srcAttr, src); + iframeElement->setAttribute(HTMLNames::frameborderAttr, AtomicString("0", AtomicString::ConstructFromLiteral)); + + // Disable frame flattening for this iframe. + iframeElement->setAttribute(HTMLNames::scrollingAttr, AtomicString("no", AtomicString::ConstructFromLiteral)); + shadowElement->appendChild(iframeElement, ASSERT_NO_EXCEPTION); +} +#endif + +void HTMLPlugInImageElement::removeSnapshotTimerFired(Timer<HTMLPlugInImageElement>&) { m_snapshotImage = nullptr; m_isRestartedPlugin = false; @@ -432,6 +462,23 @@ void HTMLPlugInImageElement::removeSnapshotTimerFired() 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 URL& 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 @@ -451,15 +498,19 @@ void HTMLPlugInImageElement::restartSimilarPlugIns() if (!frame->document()) continue; - for (auto& element : descendantsOfType<HTMLPlugInImageElement>(*frame->document())) { - if (plugInOrigin == element.loadedUrl().host() && mimeType == element.loadedMimeType()) - similarPlugins.append(element); - } + 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& plugInToRestart : similarPlugins) { + for (size_t i = 0, length = similarPlugins.size(); i < length; ++i) { + HTMLPlugInImageElement* plugInToRestart = similarPlugins[i].get(); if (plugInToRestart->displayState() <= HTMLPlugInElement::DisplayingSnapshot) { - LOG(Plugins, "%p Plug-in looks similar to a restarted plug-in. Restart.", plugInToRestart.ptr()); + LOG(Plugins, "%p Plug-in looks similar to a restarted plug-in. Restart.", plugInToRestart); plugInToRestart->restartSnapshottedPlugIn(); } plugInToRestart->m_snapshotDecision = NeverSnapshot; @@ -473,7 +524,7 @@ void HTMLPlugInImageElement::userDidClickSnapshot(PassRefPtr<MouseEvent> 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(), document().page()->sessionID()); + document().page()->plugInClient()->didStartFromOrigin(document().page()->mainFrame().document()->baseURL().host(), plugInOrigin, loadedMimeType()); LOG(Plugins, "%p User clicked on snapshotted plug-in. Restart.", this); restartSnapshottedPlugIn(); @@ -514,7 +565,7 @@ void HTMLPlugInImageElement::dispatchPendingMouseClick() m_simulatedMouseClickTimer.restart(); } -void HTMLPlugInImageElement::simulatedMouseClickTimerFired() +void HTMLPlugInImageElement::simulatedMouseClickTimerFired(DeferrableOneShotTimer<HTMLPlugInImageElement>&) { ASSERT(displayState() == RestartingWithPendingMouseClick); ASSERT(m_pendingClickEventFromSnapshot); @@ -544,7 +595,7 @@ void HTMLPlugInImageElement::checkSizeChangeForSnapshotting() return; m_needsCheckForSizeChange = false; - LayoutRect contentBoxRect = downcast<RenderBox>(*renderer()).contentBoxRect(); + LayoutRect contentBoxRect = toRenderBox(renderer())->contentBoxRect(); int contentWidth = contentBoxRect.width(); int contentHeight = contentBoxRect.height(); @@ -555,64 +606,10 @@ void HTMLPlugInImageElement::checkSizeChangeForSnapshotting() setDisplayState(WaitingForSnapshot); m_snapshotDecision = Snapshotted; Widget* widget = pluginWidget(); - if (is<PluginViewBase>(widget)) - downcast<PluginViewBase>(*widget).beginSnapshottingRunningPlugin(); + if (widget && widget->isPluginViewBase()) + toPluginViewBase(widget)->beginSnapshottingRunningPlugin(); } -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()); @@ -696,23 +693,28 @@ void HTMLPlugInImageElement::subframeLoaderWillCreatePlugIn(const URL& url) m_snapshotDecision = NeverSnapshot; return; } - - 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)) { + + 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)) { LOG(Plugins, "%p Plug-in is top level full page, set to play", this); m_snapshotDecision = NeverSnapshot; return; } - if (isSmallerThanTinySizingThreshold(renderer)) { + if (contentWidth <= sizingTinyDimensionThreshold || contentHeight <= sizingTinyDimensionThreshold) { LOG(Plugins, "%p Plug-in is very small %dx%d, set to play", this, contentWidth, contentHeight); - m_sizeWhenSnapshotted = IntSize(contentWidth, contentHeight); + m_sizeWhenSnapshotted = IntSize(contentBoxRect.width().toInt(), contentBoxRect.height().toInt()); m_snapshotDecision = MaySnapshotWhenResized; return; } @@ -724,16 +726,16 @@ void HTMLPlugInImageElement::subframeLoaderWillCreatePlugIn(const URL& url) return; } - 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); + 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); m_snapshotDecision = Snapshotted; setDisplayState(WaitingForSnapshot); } -void HTMLPlugInImageElement::subframeLoaderDidCreatePlugIn(const Widget& widget) +void HTMLPlugInImageElement::subframeLoaderDidCreatePlugIn(const Widget* widget) { m_plugInWasCreated = true; - if (is<PluginViewBase>(widget) && downcast<PluginViewBase>(widget).shouldAlwaysAutoStart()) { + if (widget->isPluginViewBase() && toPluginViewBase(widget)->shouldAlwaysAutoStart()) { LOG(Plugins, "%p Plug-in should auto-start, set to play", this); m_snapshotDecision = NeverSnapshot; setDisplayState(Playing); @@ -751,11 +753,11 @@ void HTMLPlugInImageElement::defaultEventHandler(Event* event) { RenderElement* r = renderer(); if (r && r->isEmbeddedObject()) { - if (displayState() == WaitingForSnapshot && is<MouseEvent>(*event) && event->type() == eventNames().clickEvent) { - MouseEvent& mouseEvent = downcast<MouseEvent>(*event); - if (mouseEvent.button() == LeftButton) { - userDidClickSnapshot(&mouseEvent, true); - mouseEvent.setDefaultHandled(); + if (isPlugInImageElement() && displayState() == WaitingForSnapshot && event->isMouseEvent() && event->type() == eventNames().clickEvent) { + MouseEvent* mouseEvent = toMouseEvent(event); + if (mouseEvent->button() == LeftButton) { + userDidClickSnapshot(mouseEvent, true); + event->setDefaultHandled(); return; } } |