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