blob: 7c2e6635b532f91eb57d62f1a6afbad1f2de28cb [file] [log] [blame]
// Copyright 2014 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "core/frame/RemoteFrame.h"
#include "bindings/core/v8/WindowProxy.h"
#include "bindings/core/v8/WindowProxyManager.h"
#include "core/dom/RemoteSecurityContext.h"
#include "core/frame/LocalFrame.h"
#include "core/frame/RemoteDOMWindow.h"
#include "core/frame/RemoteFrameClient.h"
#include "core/frame/RemoteFrameView.h"
#include "core/html/HTMLFrameOwnerElement.h"
#include "core/layout/LayoutPart.h"
#include "core/loader/FrameLoadRequest.h"
#include "core/paint/PaintLayer.h"
#include "platform/PluginScriptForbiddenScope.h"
#include "platform/UserGestureIndicator.h"
#include "platform/graphics/GraphicsLayer.h"
#include "platform/weborigin/SecurityPolicy.h"
#include "public/platform/WebLayer.h"
namespace blink {
inline RemoteFrame::RemoteFrame(RemoteFrameClient* client, FrameHost* host, FrameOwner* owner)
: Frame(client, host, owner)
, m_view(nullptr)
, m_securityContext(RemoteSecurityContext::create())
, m_domWindow(RemoteDOMWindow::create(*this))
, m_windowProxyManager(WindowProxyManager::create(*this))
, m_remotePlatformLayer(nullptr)
{
}
PassRefPtrWillBeRawPtr<RemoteFrame> RemoteFrame::create(RemoteFrameClient* client, FrameHost* host, FrameOwner* owner)
{
return adoptRefWillBeNoop(new RemoteFrame(client, host, owner));
}
RemoteFrame::~RemoteFrame()
{
ASSERT(!m_view);
}
DEFINE_TRACE(RemoteFrame)
{
visitor->trace(m_view);
visitor->trace(m_securityContext);
visitor->trace(m_domWindow);
visitor->trace(m_windowProxyManager);
Frame::trace(visitor);
}
DOMWindow* RemoteFrame::domWindow() const
{
return m_domWindow.get();
}
WindowProxy* RemoteFrame::windowProxy(DOMWrapperWorld& world)
{
WindowProxy* windowProxy = m_windowProxyManager->windowProxy(world);
ASSERT(windowProxy);
windowProxy->initializeIfNeeded();
return windowProxy;
}
void RemoteFrame::navigate(Document& originDocument, const KURL& url, bool replaceCurrentItem, UserGestureStatus userGestureStatus)
{
// The process where this frame actually lives won't have sufficient information to determine
// correct referrer, since it won't have access to the originDocument. Set it now.
ResourceRequest request(url);
request.setHTTPReferrer(SecurityPolicy::generateReferrer(originDocument.referrerPolicy(), url, originDocument.outgoingReferrer()));
request.setHasUserGesture(userGestureStatus == UserGestureStatus::Active);
remoteFrameClient()->navigate(request, replaceCurrentItem);
}
void RemoteFrame::navigate(const FrameLoadRequest& passedRequest)
{
UserGestureStatus gesture = UserGestureIndicator::processingUserGesture() ? UserGestureStatus::Active : UserGestureStatus::None;
navigate(*passedRequest.originDocument(), passedRequest.resourceRequest().url(), passedRequest.replacesCurrentItem(), gesture);
}
void RemoteFrame::reload(FrameLoadType frameLoadType, ClientRedirectPolicy clientRedirectPolicy)
{
remoteFrameClient()->reload(frameLoadType, clientRedirectPolicy);
}
void RemoteFrame::detach(FrameDetachType type)
{
PluginScriptForbiddenScope forbidPluginDestructorScripting;
// Frame::detach() requires the caller to keep a reference to this, since
// otherwise it may clear the last reference to this, causing it to be
// deleted, which can cause a use-after-free.
RefPtrWillBeRawPtr<RemoteFrame> protect(this);
detachChildren();
if (!client())
return;
// Clean up the frame's view if needed. A remote frame only has a view if
// the parent is a local frame.
if (m_view)
m_view->dispose();
client()->willBeDetached();
m_windowProxyManager->clearForClose();
setView(nullptr);
Frame::detach(type);
}
bool RemoteFrame::prepareForCommit()
{
detachChildren();
return !!host();
}
RemoteSecurityContext* RemoteFrame::securityContext() const
{
return m_securityContext.get();
}
void RemoteFrame::disconnectOwnerElement()
{
// The RemotePlatformLayer needs to be cleared in disconnectOwnerElement()
// because it must happen on WebFrame::swap() and Frame::detach().
if (m_remotePlatformLayer)
setRemotePlatformLayer(nullptr);
Frame::disconnectOwnerElement();
}
bool RemoteFrame::shouldClose()
{
// TODO(nasko): Implement running the beforeunload handler in the actual
// LocalFrame running in a different process and getting back a real result.
return true;
}
void RemoteFrame::forwardInputEvent(Event* event)
{
remoteFrameClient()->forwardInputEvent(event);
}
void RemoteFrame::frameRectsChanged(const IntRect& frameRect)
{
remoteFrameClient()->frameRectsChanged(frameRect);
}
void RemoteFrame::visibilityChanged(bool visible)
{
if (remoteFrameClient())
remoteFrameClient()->visibilityChanged(visible);
}
void RemoteFrame::setView(PassRefPtrWillBeRawPtr<RemoteFrameView> view)
{
// Oilpan: as RemoteFrameView performs no finalization actions,
// no explicit dispose() of it needed here. (cf. FrameView::dispose().)
m_view = view;
// ... the RemoteDOMWindow will need to be informed of detachment,
// as otherwise it will keep a strong reference back to this RemoteFrame.
// That combined with wrappers (owned and kept alive by RemoteFrame) keeping
// persistent strong references to RemoteDOMWindow will prevent the GCing
// of all these objects. Break the cycle by notifying of detachment.
if (!m_view)
m_domWindow->frameDetached();
}
void RemoteFrame::createView()
{
// If the RemoteFrame does not have a LocalFrame parent, there's no need to
// create a widget for it.
if (!deprecatedLocalOwner())
return;
ASSERT(!deprecatedLocalOwner()->ownedWidget());
setView(RemoteFrameView::create(this));
if (ownerLayoutObject())
deprecatedLocalOwner()->setWidget(m_view);
}
RemoteFrameClient* RemoteFrame::remoteFrameClient() const
{
return static_cast<RemoteFrameClient*>(client());
}
void RemoteFrame::setRemotePlatformLayer(WebLayer* layer)
{
if (m_remotePlatformLayer)
GraphicsLayer::unregisterContentsLayer(m_remotePlatformLayer);
m_remotePlatformLayer = layer;
if (m_remotePlatformLayer)
GraphicsLayer::registerContentsLayer(layer);
ASSERT(owner());
toHTMLFrameOwnerElement(owner())->setNeedsCompositingUpdate();
}
void RemoteFrame::advanceFocus(WebFocusType type, LocalFrame* source)
{
remoteFrameClient()->advanceFocus(type, source);
}
} // namespace blink