Unreviewed, rolling out r213400.
[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 "DocumentLoader.h"
34 #include "EventNames.h"
35 #include "Frame.h"
36 #include "FrameLoader.h"
37 #include "FrameLoaderClient.h"
38 #include "HTMLBodyElement.h"
39 #include "HTMLEmbedElement.h"
40 #include "HTMLHeadElement.h"
41 #include "HTMLHtmlElement.h"
42 #include "HTMLMetaElement.h"
43 #include "HTMLNames.h"
44 #include "HTMLSourceElement.h"
45 #include "HTMLVideoElement.h"
46 #include "KeyboardEvent.h"
47 #include "NodeList.h"
48 #include "Page.h"
49 #include "RawDataDocumentParser.h"
50 #include "RuntimeEnabledFeatures.h"
51 #include "ScriptController.h"
52 #include "ShadowRoot.h"
53 #include "TypedElementDescendantIterator.h"
54 #include <wtf/text/StringBuilder.h>
55
56 namespace WebCore {
57
58 using namespace HTMLNames;
59
60 // FIXME: Share more code with PluginDocumentParser.
61 class MediaDocumentParser final : public RawDataDocumentParser {
62 public:
63     static Ref<MediaDocumentParser> create(MediaDocument& document)
64     {
65         return adoptRef(*new MediaDocumentParser(document));
66     }
67     
68 private:
69     MediaDocumentParser(MediaDocument& document)
70         : RawDataDocumentParser(document)
71         , m_mediaElement(0)
72     {
73         m_outgoingReferrer = document.outgoingReferrer();
74     }
75
76     void appendBytes(DocumentWriter&, const char*, size_t) override;
77
78     void createDocumentStructure();
79
80     HTMLMediaElement* m_mediaElement;
81     String m_outgoingReferrer;
82 };
83     
84 void MediaDocumentParser::createDocumentStructure()
85 {
86     auto& document = *this->document();
87
88     auto rootElement = HTMLHtmlElement::create(document);
89     document.appendChild(rootElement);
90     document.setCSSTarget(rootElement.ptr());
91     rootElement->insertedByParser();
92
93     if (document.frame())
94         document.frame()->injectUserScripts(InjectAtDocumentStart);
95
96 #if PLATFORM(IOS)
97     auto headElement = HTMLHeadElement::create(document);
98     rootElement->appendChild(headElement);
99
100     auto metaElement = HTMLMetaElement::create(document);
101     metaElement->setAttributeWithoutSynchronization(nameAttr, AtomicString("viewport", AtomicString::ConstructFromLiteral));
102     metaElement->setAttributeWithoutSynchronization(contentAttr, AtomicString("width=device-width,initial-scale=1", AtomicString::ConstructFromLiteral));
103     headElement->appendChild(metaElement);
104 #endif
105
106     auto body = HTMLBodyElement::create(document);
107     if (RuntimeEnabledFeatures::sharedFeatures().modernMediaControlsEnabled()) {
108         StringBuilder bodyStyle;
109         bodyStyle.appendLiteral("margin: 0; padding: 0;");
110         bodyStyle.appendLiteral("background-color: rgb(38, 38, 38);");
111         bodyStyle.appendLiteral("display: flex; justify-content: center; align-items: center;");
112         body->setAttribute(styleAttr, bodyStyle.toString());
113     }
114     rootElement->appendChild(body);
115
116     auto videoElement = HTMLVideoElement::create(document);
117     m_mediaElement = videoElement.ptr();
118     videoElement->setAttributeWithoutSynchronization(controlsAttr, emptyAtom);
119     videoElement->setAttributeWithoutSynchronization(autoplayAttr, emptyAtom);
120     videoElement->setAttributeWithoutSynchronization(nameAttr, AtomicString("media", AtomicString::ConstructFromLiteral));
121
122     StringBuilder elementStyle;
123     elementStyle.appendLiteral("max-width: 100%; max-height: 100%;");
124 #if PLATFORM(IOS)
125     elementStyle.appendLiteral("width: 100%; height: auto;");
126 #endif
127     if (RuntimeEnabledFeatures::sharedFeatures().modernMediaControlsEnabled()) {
128         elementStyle.appendLiteral("min-height: 50px;");
129     }
130     videoElement->setAttribute(styleAttr, elementStyle.toString());
131
132     auto sourceElement = HTMLSourceElement::create(document);
133     sourceElement->setSrc(document.url());
134
135     if (auto* loader = document.loader())
136         sourceElement->setType(loader->responseMIMEType());
137
138     videoElement->appendChild(sourceElement);
139     body->appendChild(videoElement);
140
141     Frame* frame = document.frame();
142     if (!frame)
143         return;
144
145     frame->loader().activeDocumentLoader()->setMainResourceDataBufferingPolicy(DoNotBufferData);
146     frame->loader().setOutgoingReferrer(document.completeURL(m_outgoingReferrer));
147 }
148
149 void MediaDocumentParser::appendBytes(DocumentWriter&, const char*, size_t)
150 {
151     if (m_mediaElement)
152         return;
153
154     createDocumentStructure();
155     finish();
156 }
157     
158 MediaDocument::MediaDocument(Frame* frame, const URL& url)
159     : HTMLDocument(frame, url, MediaDocumentClass)
160     , m_replaceMediaElementTimer(*this, &MediaDocument::replaceMediaElementTimerFired)
161 {
162     setCompatibilityMode(DocumentCompatibilityMode::QuirksMode);
163     lockCompatibilityMode();
164     if (frame)
165         m_outgoingReferrer = frame->loader().outgoingReferrer();
166 }
167
168 MediaDocument::~MediaDocument()
169 {
170     ASSERT(!m_replaceMediaElementTimer.isActive());
171 }
172
173 Ref<DocumentParser> MediaDocument::createParser()
174 {
175     return MediaDocumentParser::create(*this);
176 }
177
178 static inline HTMLVideoElement* descendantVideoElement(ContainerNode& node)
179 {
180     if (is<HTMLVideoElement>(node))
181         return downcast<HTMLVideoElement>(&node);
182
183     return descendantsOfType<HTMLVideoElement>(node).first();
184 }
185
186 static inline HTMLVideoElement* ancestorVideoElement(Node* node)
187 {
188     while (node && !is<HTMLVideoElement>(*node))
189         node = node->parentOrShadowHostNode();
190
191     return downcast<HTMLVideoElement>(node);
192 }
193
194 void MediaDocument::defaultEventHandler(Event& event)
195 {
196     // Match the default Quicktime plugin behavior to allow 
197     // clicking and double-clicking to pause and play the media.
198     Node* targetNode = event.target()->toNode();
199     if (!targetNode)
200         return;
201
202     if (HTMLVideoElement* video = ancestorVideoElement(targetNode)) {
203         if (event.type() == eventNames().clickEvent) {
204             if (!video->canPlay()) {
205                 video->pause();
206                 event.setDefaultHandled();
207             }
208         } else if (event.type() == eventNames().dblclickEvent) {
209             if (video->canPlay()) {
210                 video->play();
211                 event.setDefaultHandled();
212             }
213         }
214     }
215
216     if (!is<ContainerNode>(*targetNode))
217         return;
218     ContainerNode& targetContainer = downcast<ContainerNode>(*targetNode);
219     if (event.type() == eventNames().keydownEvent && is<KeyboardEvent>(event)) {
220         HTMLVideoElement* video = descendantVideoElement(targetContainer);
221         if (!video)
222             return;
223
224         KeyboardEvent& keyboardEvent = downcast<KeyboardEvent>(event);
225         if (keyboardEvent.keyIdentifier() == "U+0020") { // space
226             if (video->paused()) {
227                 if (video->canPlay())
228                     video->play();
229             } else
230                 video->pause();
231             keyboardEvent.setDefaultHandled();
232         }
233     }
234 }
235
236 void MediaDocument::mediaElementSawUnsupportedTracks()
237 {
238     // The HTMLMediaElement was told it has something that the underlying 
239     // MediaPlayer cannot handle so we should switch from <video> to <embed> 
240     // and let the plugin handle this. Don't do it immediately as this 
241     // function may be called directly from a media engine callback, and 
242     // replaceChild will destroy the element, media player, and media engine.
243     m_replaceMediaElementTimer.startOneShot(0);
244 }
245
246 void MediaDocument::replaceMediaElementTimerFired()
247 {
248     auto* htmlBody = bodyOrFrameset();
249     if (!htmlBody)
250         return;
251
252     // Set body margin width and height to 0 as that is what a PluginDocument uses.
253     htmlBody->setAttributeWithoutSynchronization(marginwidthAttr, AtomicString("0", AtomicString::ConstructFromLiteral));
254     htmlBody->setAttributeWithoutSynchronization(marginheightAttr, AtomicString("0", AtomicString::ConstructFromLiteral));
255
256     if (auto* videoElement = descendantVideoElement(*htmlBody)) {
257         auto embedElement = HTMLEmbedElement::create(*this);
258
259         embedElement->setAttributeWithoutSynchronization(widthAttr, AtomicString("100%", AtomicString::ConstructFromLiteral));
260         embedElement->setAttributeWithoutSynchronization(heightAttr, AtomicString("100%", AtomicString::ConstructFromLiteral));
261         embedElement->setAttributeWithoutSynchronization(nameAttr, AtomicString("plugin", AtomicString::ConstructFromLiteral));
262         embedElement->setAttributeWithoutSynchronization(srcAttr, url().string());
263
264         ASSERT(loader());
265         if (auto* loader = this->loader())
266             embedElement->setAttributeWithoutSynchronization(typeAttr, loader->writer().mimeType());
267
268         videoElement->parentNode()->replaceChild(embedElement, *videoElement);
269     }
270 }
271
272 void MediaDocument::mediaElementNaturalSizeChanged(const IntSize& newSize)
273 {
274     if (ownerElement())
275         return;
276
277     if (newSize.isZero())
278         return;
279
280     if (page())
281         page()->chrome().client().imageOrMediaDocumentSizeChanged(newSize);
282 }
283
284 }
285
286 #endif