blob: a624c20a86c31f2f39313727198a5d3e1a00207b [file] [log] [blame]
/*
* Copyright (C) 1999 Lars Knoll (knoll@kde.org)
* (C) 1999 Antti Koivisto (koivisto@kde.org)
* (C) 2001 Dirk Mueller (mueller@kde.org)
* (C) 2006 Alexey Proskuryakov (ap@webkit.org)
* Copyright (C) 2004, 2005, 2006, 2007, 2008, 2009, 2010, 2012 Apple Inc. All rights reserved.
* Copyright (C) 2008, 2009 Torch Mobile Inc. All rights reserved. (http://www.torchmobile.com/)
* Copyright (C) 2010 Nokia Corporation and/or its subsidiary(-ies)
* Copyright (C) 2013 Google 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
* License as published by the Free Software Foundation; either
* version 2 of the License, or (at your option) any later version.
*
* This library is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public License
* along with this library; see the file COPYING.LIB. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301, USA.
*
*/
#ifndef Fullscreen_h
#define Fullscreen_h
#include "core/CoreExport.h"
#include "core/dom/ContextLifecycleObserver.h"
#include "core/dom/Document.h"
#include "core/dom/Element.h"
#include "platform/Supplementable.h"
#include "platform/Timer.h"
#include "platform/geometry/LayoutRect.h"
#include "wtf/Deque.h"
#include "wtf/RefPtr.h"
#include "wtf/Vector.h"
namespace blink {
class LayoutFullScreen;
class ComputedStyle;
class CORE_EXPORT Fullscreen final
: public GarbageCollectedFinalized<Fullscreen>
, public Supplement<Document>
, public ContextLifecycleObserver {
USING_GARBAGE_COLLECTED_MIXIN(Fullscreen);
public:
virtual ~Fullscreen();
static const char* supplementName();
static Fullscreen& from(Document&);
static Fullscreen* fromIfExists(Document&);
static Element* fullscreenElementFrom(Document&);
static Element* currentFullScreenElementFrom(Document&);
static bool isFullScreen(Document&);
static bool isActiveFullScreenElement(const Element&);
enum RequestType {
// Element.requestFullscreen()
UnprefixedRequest,
// Element.webkitRequestFullscreen()/webkitRequestFullScreen() and
// HTMLVideoElement.webkitEnterFullscreen()/webkitEnterFullScreen()
PrefixedRequest,
};
// |forCrossProcessAncestor| is used in OOPIF scenarios and is set to true
// when fullscreen is requested for an out-of-process descendant element.
void requestFullscreen(Element&, RequestType, bool forCrossProcessAncestor = false);
static void fullyExitFullscreen(Document&);
void exitFullscreen();
static bool fullscreenEnabled(Document&);
Element* fullscreenElement() const { return !m_fullScreenElementStack.isEmpty() ? m_fullScreenElementStack.last().first.get() : 0; }
// |isAncestorOfFullscreenElement| is used in OOPIF scenarios and is set to
// true when these functions are called to enter/exit fullscreen for an
// out-of-process descendant element. In this case, we enter fullscreen
// for its (local) iframe container and make sure to also set the
// ContainsFullScreenElement flag on it (so that it gains the
// -webkit-full-screen-ancestor style).
void didEnterFullScreenForElement(Element*);
void didExitFullScreenForElement();
void setFullScreenLayoutObject(LayoutFullScreen*);
LayoutFullScreen* fullScreenLayoutObject() const { return m_fullScreenLayoutObject; }
void fullScreenLayoutObjectDestroyed();
void elementRemoved(Element&);
// Returns true if the current fullscreen element stack corresponds to a
// container for an actual fullscreen element in an out-of-process iframe.
bool forCrossProcessAncestor() { return m_forCrossProcessAncestor; }
// Mozilla API
Element* webkitCurrentFullScreenElement() const { return m_fullScreenElement.get(); }
// ContextLifecycleObserver:
void contextDestroyed() override;
DECLARE_VIRTUAL_TRACE();
private:
static Fullscreen* fromIfExistsSlow(Document&);
explicit Fullscreen(Document&);
Document* document();
void clearFullscreenElementStack();
void popFullscreenElementStack();
void pushFullscreenElementStack(Element&, RequestType);
void enqueueChangeEvent(Document&, RequestType);
void enqueueErrorEvent(Element&, RequestType);
void eventQueueTimerFired(Timer<Fullscreen>*);
Member<Element> m_fullScreenElement;
HeapVector<std::pair<Member<Element>, RequestType>> m_fullScreenElementStack;
LayoutFullScreen* m_fullScreenLayoutObject;
Timer<Fullscreen> m_eventQueueTimer;
HeapDeque<Member<Event>> m_eventQueue;
LayoutRect m_savedPlaceholderFrameRect;
RefPtr<ComputedStyle> m_savedPlaceholderComputedStyle;
// TODO(alexmos, dcheng): Currently, this assumes that if fullscreen was
// entered for an element in an out-of-process iframe, then it's not
// possible to re-enter fullscreen for a different element in this
// document, since that requires a user gesture, which can't be obtained
// since nothing in this document is visible, and since user gestures can't
// be forwarded across processes. However, the latter assumption could
// change if https://crbug.com/161068 is fixed so that cross-process
// postMessage can carry user gestures. If that happens, this should be
// moved to be part of |m_fullScreenElementStack|.
bool m_forCrossProcessAncestor;
};
inline bool Fullscreen::isActiveFullScreenElement(const Element& element)
{
Fullscreen* fullscreen = fromIfExists(element.document());
if (!fullscreen)
return false;
return fullscreen->webkitCurrentFullScreenElement() == &element;
}
inline Fullscreen* Fullscreen::fromIfExists(Document& document)
{
if (!document.hasFullscreenSupplement())
return 0;
return fromIfExistsSlow(document);
}
} // namespace blink
#endif // Fullscreen_h