Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(695)

Unified Diff: Source/core/html/HTMLFrameOwnerElement.cpp

Issue 23618022: BrowserPlugin/WebView - Move plugin lifetime to DOM (Closed) Base URL: svn://svn.chromium.org/blink/trunk
Patch Set: Add tests, make plugin creation synchronous. Created 7 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: Source/core/html/HTMLFrameOwnerElement.cpp
diff --git a/Source/core/html/HTMLFrameOwnerElement.cpp b/Source/core/html/HTMLFrameOwnerElement.cpp
index 20774db3cc50b436864598b568014e034aa8c0d2..3339e2bd0f168687448ce4752485cf0bc6fb8abc 100644
--- a/Source/core/html/HTMLFrameOwnerElement.cpp
+++ b/Source/core/html/HTMLFrameOwnerElement.cpp
@@ -23,6 +23,7 @@
#include "bindings/v8/ExceptionMessages.h"
#include "bindings/v8/ExceptionState.h"
+#include "core/accessibility/AXObjectCache.h"
#include "core/dom/ExceptionCode.h"
#include "core/loader/FrameLoader.h"
#include "core/loader/FrameLoaderClient.h"
@@ -35,6 +36,54 @@
namespace WebCore {
+typedef HashMap<RefPtr<Widget>, FrameView*> WidgetToParentMap;
+static WidgetToParentMap& widgetNewParentMap()
+{
+ DEFINE_STATIC_LOCAL(WidgetToParentMap, map, ());
+ return map;
+}
+
+static unsigned s_updateSuspendCount = 0;
+
+HTMLFrameOwnerElement::UpdateSuspendScope::UpdateSuspendScope()
+{
+ ++s_updateSuspendCount;
+}
+
+HTMLFrameOwnerElement::UpdateSuspendScope::~UpdateSuspendScope()
+{
+ ASSERT(s_updateSuspendCount > 0);
+ if (s_updateSuspendCount == 1) {
+ WidgetToParentMap map;
+ widgetNewParentMap().swap(map);
+ WidgetToParentMap::iterator end = map.end();
+ for (WidgetToParentMap::iterator it = map.begin(); it != end; ++it) {
+ Widget* child = it->key.get();
+ ScrollView* currentParent = toScrollView(child->parent());
+ FrameView* newParent = it->value;
+ if (newParent != currentParent) {
+ if (currentParent)
+ currentParent->removeChild(child);
+ if (newParent)
+ newParent->addChild(child);
+ }
+ }
+ }
+ --s_updateSuspendCount;
+}
+
+static void moveWidgetToParentSoon(Widget* child, FrameView* parent)
+{
+ if (!s_updateSuspendCount) {
+ if (parent)
+ parent->addChild(child);
+ else if (toScrollView(child->parent()))
+ toScrollView(child->parent())->removeChild(child);
+ return;
+ }
+ widgetNewParentMap().set(child, parent);
+}
+
HTMLFrameOwnerElement::HTMLFrameOwnerElement(const QualifiedName& tagName, Document& document)
: HTMLElement(tagName, document)
, m_contentFrame(0)
@@ -121,6 +170,52 @@ SVGDocument* HTMLFrameOwnerElement::getSVGDocument(ExceptionState& exceptionStat
return 0;
}
+void HTMLFrameOwnerElement::setWidget(PassRefPtr<Widget> widget)
+{
+ if (widget == m_widget)
+ return;
+
+ if (m_widget) {
+ if (m_widget->parent())
+ moveWidgetToParentSoon(m_widget.get(), 0);
+ m_widget = 0;
+ }
+
+ m_widget = widget;
+
+ RenderWidget* renderWidget = toRenderWidget(renderer());
+ if (!renderWidget || !renderWidget->isWidget())
eseidel 2013/12/12 20:40:13 What does isWidget() do here?
wjmaclean 2013/12/12 22:31:15 I'm not entirely sure, but even RenderWidget() was
+ return;
+
+ if (m_widget) {
+ if (renderWidget->style()) {
eseidel 2013/12/12 20:40:13 It feels liek this wants to call some method on th
wjmaclean 2013/12/12 22:31:15 OK.
+ if (!renderWidget->needsLayout())
+ renderWidget->updateWidgetGeometry();
+
+ if (renderWidget->style()->visibility() != VISIBLE) {
eseidel 2013/12/12 20:40:13 This all seems suspect. The DOM shouldn't be deal
wjmaclean 2013/12/12 22:31:15 Is it ok if dealt with as described above? Or do w
+ m_widget->hide();
+ } else {
+ m_widget->show();
+ renderWidget->repaint();
+ }
+ }
+ ASSERT(document().view() == renderWidget->frameView());
+ ASSERT(renderWidget->frameView());
+ moveWidgetToParentSoon(m_widget.get(), renderWidget->frameView());
+ }
+
+ if (AXObjectCache* cache = document().existingAXObjectCache())
+ cache->childrenChanged(renderWidget);
+
+ if (RenderPart* renderPart = toRenderPart(renderWidget))
+ renderPart->viewCleared();
+}
+
+Widget* HTMLFrameOwnerElement::widget() const
+{
+ return m_widget.get();
+}
+
bool HTMLFrameOwnerElement::loadOrRedirectSubframe(const KURL& url, const AtomicString& frameName, bool lockBackForwardList)
{
RefPtr<Frame> parentFrame = document().frame();
@@ -153,10 +248,12 @@ bool HTMLFrameOwnerElement::loadOrRedirectSubframe(const KURL& url, const Atomic
// FIXME: Can we remove this entirely? m_isComplete normally gets set to false when a load is committed.
childFrame->loader().started();
- RenderObject* renderObject = renderer();
FrameView* view = childFrame->view();
+ RenderObject* renderObject = renderer();
+ // We need to test the existence of renderObject and its widget-ness, as
+ // failing to do so causes problems.
if (renderObject && renderObject->isWidget() && view)
- toRenderWidget(renderObject)->setWidget(view);
+ setWidget(view);
// Some loads are performed synchronously (e.g., about:blank and loads
// cancelled by returning a null ResourceRequest from requestFromDelegate).

Powered by Google App Engine
This is Rietveld 408576698