Update ANGLE
[WebKit-https.git] / Source / WebCore / html / MediaDocument.cpp
1 /*
2  * Copyright (C) 2008 Apple Inc. All Rights Reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without
5  * modification, are permitted provided that the following conditions
6  * are met:
7  * 1. Redistributions of source code must retain the above copyright
8  *    notice, this list of conditions and the following disclaimer.
9  * 2. Redistributions in binary form must reproduce the above copyright
10  *    notice, this list of conditions and the following disclaimer in the
11  *    documentation and/or other materials provided with the distribution.
12  *
13  * THIS SOFTWARE IS PROVIDED BY APPLE INC. ``AS IS'' AND ANY
14  * EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
15  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED.  IN NO EVENT SHALL APPLE INC. OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
18  * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
19  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
20  * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY
21  * OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
22  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
23  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 
24  */
25
26 #include "config.h"
27 #include "MediaDocument.h"
28
29 #if ENABLE(VIDEO)
30
31 #include "Chrome.h"
32 #include "ChromeClient.h"
33 #include "CustomHeaderFields.h"
34 #include "DocumentLoader.h"
35 #include "EventNames.h"
36 #include "Frame.h"
37 #include "FrameLoader.h"
38 #include "FrameLoaderClient.h"
39 #include "HTMLBodyElement.h"
40 #include "HTMLEmbedElement.h"
41 #include "HTMLHeadElement.h"
42 #include "HTMLHtmlElement.h"
43 #include "HTMLMetaElement.h"
44 #include "HTMLNames.h"
45 #include "HTMLSourceElement.h"
46 #include "HTMLVideoElement.h"
47 #include "KeyboardEvent.h"
48 #include "NodeList.h"
49 #include "Page.h"
50 #include "RawDataDocumentParser.h"
51 #include "RuntimeEnabledFeatures.h"
52 #include "ScriptController.h"
53 #include "ShadowRoot.h"
54 #include "TypedElementDescendantIterator.h"
55 #include <wtf/IsoMallocInlines.h>
56 #include <wtf/text/StringBuilder.h>
57
58 namespace WebCore {
59
60 WTF_MAKE_ISO_ALLOCATED_IMPL(MediaDocument);
61
62 using namespace HTMLNames;
63
64 // FIXME: Share more code with PluginDocumentParser.
65 class MediaDocumentParser final : public RawDataDocumentParser {
66 public:
67     static Ref<MediaDocumentParser> create(MediaDocument& document)
68     {
69         return adoptRef(*new MediaDocumentParser(document));
70     }
71     
72 private:
73     MediaDocumentParser(MediaDocument& document)
74         : RawDataDocumentParser { document }
75         , m_outgoingReferrer { document.outgoingReferrer() }
76     {
77     }
78
79     void appendBytes(DocumentWriter&, const char*, size_t) final;
80     void createDocumentStructure();
81
82     HTMLMediaElement* m_mediaElement { nullptr };
83     String m_outgoingReferrer;
84 };
85     
86 void MediaDocumentParser::createDocumentStructure()
87 {
88     auto& document = *this->document();
89
90     auto rootElement = HTMLHtmlElement::create(document);
91     document.appendChild(rootElement);
92     document.setCSSTarget(rootElement.ptr());
93     rootElement->insertedByParser();
94
95     if (document.frame())
96         document.frame()->injectUserScripts(InjectAtDocumentStart);
97
98 #if PLATFORM(IOS_FAMILY)
99     auto headElement = HTMLHeadElement::create(document);
100     rootElement->appendChild(headElement);
101
102     auto metaElement = HTMLMetaElement::create(document);
103     metaElement->setAttributeWithoutSynchronization(nameAttr, AtomString("viewport", AtomString::ConstructFromLiteral));
104     metaElement->setAttributeWithoutSynchronization(contentAttr, AtomString("width=device-width,initial-scale=1", AtomString::ConstructFromLiteral));
105     headElement->appendChild(metaElement);
106 #endif
107
108     auto body = HTMLBodyElement::create(document);
109     rootElement->appendChild(body);
110
111     auto videoElement = HTMLVideoElement::create(document);
112     m_mediaElement = videoElement.ptr();
113     videoElement->setAttributeWithoutSynchronization(controlsAttr, emptyAtom());
114     videoElement->setAttributeWithoutSynchronization(autoplayAttr, emptyAtom());
115     videoElement->setAttributeWithoutSynchronization(srcAttr, document.url().string());
116     if (auto loader = makeRefPtr(document.loader()))
117         videoElement->setAttributeWithoutSynchronization(typeAttr, loader->responseMIMEType());
118
119     if (!RuntimeEnabledFeatures::sharedFeatures().modernMediaControlsEnabled()) {
120         StringBuilder elementStyle;
121         elementStyle.appendLiteral("max-width: 100%; max-height: 100%;");
122 #if PLATFORM(IOS_FAMILY)
123         elementStyle.appendLiteral("width: 100%; height: auto;");
124 #endif
125         videoElement->setAttribute(styleAttr, elementStyle.toString());
126     }
127
128     body->appendChild(videoElement);
129
130     RefPtr<Frame> frame = document.frame();
131     if (!frame)
132         return;
133
134     frame->loader().activeDocumentLoader()->setMainResourceDataBufferingPolicy(DataBufferingPolicy::DoNotBufferData);
135     frame->loader().setOutgoingReferrer(document.completeURL(m_outgoingReferrer));
136 }
137
138 void MediaDocumentParser::appendBytes(DocumentWriter&, const char*, size_t)
139 {
140     if (m_mediaElement)
141         return;
142
143     createDocumentStructure();
144     finish();
145 }
146     
147 MediaDocument::MediaDocument(PAL::SessionID sessionID, Frame* frame, const URL& url)
148     : HTMLDocument(sessionID, frame, url, MediaDocumentClass)
149     , m_replaceMediaElementTimer(*this, &MediaDocument::replaceMediaElementTimerFired)
150 {
151     setCompatibilityMode(DocumentCompatibilityMode::QuirksMode);
152     lockCompatibilityMode();
153     if (frame)
154         m_outgoingReferrer = frame->loader().outgoingReferrer();
155 }
156
157 MediaDocument::~MediaDocument()
158 {
159     ASSERT(!m_replaceMediaElementTimer.isActive());
160 }
161
162 Ref<DocumentParser> MediaDocument::createParser()
163 {
164     return MediaDocumentParser::create(*this);
165 }
166
167 static inline HTMLVideoElement* descendantVideoElement(ContainerNode& node)
168 {
169     if (is<HTMLVideoElement>(node))
170         return downcast<HTMLVideoElement>(&node);
171
172     return descendantsOfType<HTMLVideoElement>(node).first();
173 }
174
175 static inline HTMLVideoElement* ancestorVideoElement(Node* node)
176 {
177     while (node && !is<HTMLVideoElement>(*node))
178         node = node->parentOrShadowHostNode();
179
180     return downcast<HTMLVideoElement>(node);
181 }
182
183 void MediaDocument::defaultEventHandler(Event& event)
184 {
185     // Modern media controls have their own event handling to determine when to
186     // pause or resume playback.
187     if (RuntimeEnabledFeatures::sharedFeatures().modernMediaControlsEnabled())
188         return;
189     
190     // Match the default Quicktime plugin behavior to allow
191     // clicking and double-clicking to pause and play the media.
192     if (!is<Node>(event.target()))
193         return;
194     auto& targetNode = downcast<Node>(*event.target());
195
196     if (auto video = makeRefPtr(ancestorVideoElement(&targetNode))) {
197         if (event.type() == eventNames().clickEvent) {
198             if (!video->canPlay()) {
199                 video->pause();
200                 event.setDefaultHandled();
201             }
202         } else if (event.type() == eventNames().dblclickEvent) {
203             if (video->canPlay()) {
204                 video->play();
205                 event.setDefaultHandled();
206             }
207         }
208     }
209
210     if (!is<ContainerNode>(targetNode))
211         return;
212     auto& targetContainer = downcast<ContainerNode>(targetNode);
213
214     if (event.type() == eventNames().keydownEvent && is<KeyboardEvent>(event)) {
215         auto video = makeRefPtr(descendantVideoElement(targetContainer));
216         if (!video)
217             return;
218
219         auto& keyboardEvent = downcast<KeyboardEvent>(event);
220         if (keyboardEvent.keyIdentifier() == "U+0020") { // space
221             if (video->paused()) {
222                 if (video->canPlay())
223                     video->play();
224             } else
225                 video->pause();
226             keyboardEvent.setDefaultHandled();
227         }
228     }
229 }
230
231 void MediaDocument::mediaElementSawUnsupportedTracks()
232 {
233     // The HTMLMediaElement was told it has something that the underlying 
234     // MediaPlayer cannot handle so we should switch from <video> to <embed> 
235     // and let the plugin handle this. Don't do it immediately as this 
236     // function may be called directly from a media engine callback, and 
237     // replaceChild will destroy the element, media player, and media engine.
238     m_replaceMediaElementTimer.startOneShot(0_s);
239 }
240
241 void MediaDocument::replaceMediaElementTimerFired()
242 {
243     auto htmlBody = makeRefPtr(bodyOrFrameset());
244     if (!htmlBody)
245         return;
246
247     // Set body margin width and height to 0 as that is what a PluginDocument uses.
248     htmlBody->setAttributeWithoutSynchronization(marginwidthAttr, AtomString("0", AtomString::ConstructFromLiteral));
249     htmlBody->setAttributeWithoutSynchronization(marginheightAttr, AtomString("0", AtomString::ConstructFromLiteral));
250
251     if (auto videoElement = makeRefPtr(descendantVideoElement(*htmlBody))) {
252         auto embedElement = HTMLEmbedElement::create(*this);
253
254         embedElement->setAttributeWithoutSynchronization(widthAttr, AtomString("100%", AtomString::ConstructFromLiteral));
255         embedElement->setAttributeWithoutSynchronization(heightAttr, AtomString("100%", AtomString::ConstructFromLiteral));
256         embedElement->setAttributeWithoutSynchronization(nameAttr, AtomString("plugin", AtomString::ConstructFromLiteral));
257         embedElement->setAttributeWithoutSynchronization(srcAttr, url().string());
258
259         ASSERT(loader());
260         if (auto loader = makeRefPtr(this->loader()))
261             embedElement->setAttributeWithoutSynchronization(typeAttr, loader->writer().mimeType());
262
263         videoElement->parentNode()->replaceChild(embedElement, *videoElement);
264     }
265 }
266
267 void MediaDocument::mediaElementNaturalSizeChanged(const IntSize& newSize)
268 {
269     if (ownerElement())
270         return;
271
272     if (newSize.isZero())
273         return;
274
275     if (page())
276         page()->chrome().client().imageOrMediaDocumentSizeChanged(newSize);
277 }
278
279 }
280
281 #endif