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