Unreviewed attempted build fix. Adds back some includes removed in
[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 "DocumentLoader.h"
32 #include "EventNames.h"
33 #include "ExceptionCodePlaceholder.h"
34 #include "Frame.h"
35 #include "FrameLoader.h"
36 #include "FrameLoaderClient.h"
37 #include "HTMLEmbedElement.h"
38 #include "HTMLHtmlElement.h"
39 #include "HTMLNames.h"
40 #include "HTMLSourceElement.h"
41 #include "HTMLVideoElement.h"
42 #include "KeyboardEvent.h"
43 #include "MainResourceLoader.h"
44 #include "NodeList.h"
45 #include "RawDataDocumentParser.h"
46 #include "ScriptController.h"
47
48 namespace WebCore {
49
50 using namespace HTMLNames;
51
52 // FIXME: Share more code with PluginDocumentParser.
53 class MediaDocumentParser : public RawDataDocumentParser {
54 public:
55     static PassRefPtr<MediaDocumentParser> create(MediaDocument* document)
56     {
57         return adoptRef(new MediaDocumentParser(document));
58     }
59     
60 private:
61     MediaDocumentParser(Document* document)
62         : RawDataDocumentParser(document)
63         , m_mediaElement(0)
64     {
65     }
66
67     virtual void appendBytes(DocumentWriter*, const char*, size_t);
68
69     void createDocumentStructure();
70
71     HTMLMediaElement* m_mediaElement;
72 };
73     
74 void MediaDocumentParser::createDocumentStructure()
75 {
76     RefPtr<Element> rootElement = document()->createElement(htmlTag, false);
77     document()->appendChild(rootElement, IGNORE_EXCEPTION);
78     document()->setCSSTarget(rootElement.get());
79     static_cast<HTMLHtmlElement*>(rootElement.get())->insertedByParser();
80
81     if (document()->frame())
82         document()->frame()->loader()->dispatchDocumentElementAvailable();
83         
84     RefPtr<Element> body = document()->createElement(bodyTag, false);
85     rootElement->appendChild(body, IGNORE_EXCEPTION);
86
87     RefPtr<Element> mediaElement = document()->createElement(videoTag, false);
88
89     m_mediaElement = static_cast<HTMLVideoElement*>(mediaElement.get());
90     m_mediaElement->setAttribute(controlsAttr, "");
91     m_mediaElement->setAttribute(autoplayAttr, "");
92
93     m_mediaElement->setAttribute(nameAttr, "media");
94
95     RefPtr<Element> sourceElement = document()->createElement(sourceTag, false);
96     HTMLSourceElement* source = static_cast<HTMLSourceElement*>(sourceElement.get());
97     source->setSrc(document()->url());
98
99     if (DocumentLoader* loader = document()->loader())
100         source->setType(loader->responseMIMEType());
101
102     m_mediaElement->appendChild(sourceElement, IGNORE_EXCEPTION);
103     body->appendChild(mediaElement, IGNORE_EXCEPTION);
104
105     Frame* frame = document()->frame();
106     if (!frame)
107         return;
108
109     frame->loader()->activeDocumentLoader()->mainResourceLoader()->setDataBufferingPolicy(DoNotBufferData);
110 }
111
112 void MediaDocumentParser::appendBytes(DocumentWriter*, const char*, size_t)
113 {
114     if (m_mediaElement)
115         return;
116
117     createDocumentStructure();
118     finish();
119 }
120     
121 MediaDocument::MediaDocument(Frame* frame, const KURL& url)
122     : HTMLDocument(frame, url)
123     , m_replaceMediaElementTimer(this, &MediaDocument::replaceMediaElementTimerFired)
124 {
125     setCompatibilityMode(QuirksMode);
126     lockCompatibilityMode();
127 }
128
129 MediaDocument::~MediaDocument()
130 {
131     ASSERT(!m_replaceMediaElementTimer.isActive());
132 }
133
134 PassRefPtr<DocumentParser> MediaDocument::createParser()
135 {
136     return MediaDocumentParser::create(this);
137 }
138
139 static inline HTMLVideoElement* descendentVideoElement(Node* node)
140 {
141     ASSERT(node);
142
143     if (node->hasTagName(videoTag))
144         return static_cast<HTMLVideoElement*>(node);
145
146     RefPtr<NodeList> nodeList = node->getElementsByTagNameNS(videoTag.namespaceURI(), videoTag.localName());
147    
148     if (nodeList.get()->length() > 0)
149         return static_cast<HTMLVideoElement*>(nodeList.get()->item(0));
150
151     return 0;
152 }
153
154 static inline HTMLVideoElement* ancestorVideoElement(Node* node)
155 {
156     while (node && !node->hasTagName(videoTag))
157         node = node->parentOrShadowHostNode();
158
159     return static_cast<HTMLVideoElement*>(node);
160 }
161
162 void MediaDocument::defaultEventHandler(Event* event)
163 {
164     // Match the default Quicktime plugin behavior to allow 
165     // clicking and double-clicking to pause and play the media.
166     Node* targetNode = event->target()->toNode();
167     if (!targetNode)
168         return;
169
170 #if !PLATFORM(CHROMIUM)
171     if (HTMLVideoElement* video = ancestorVideoElement(targetNode)) {
172         if (event->type() == eventNames().clickEvent) {
173             if (!video->canPlay()) {
174                 video->pause();
175                 event->setDefaultHandled();
176             }
177         } else if (event->type() == eventNames().dblclickEvent) {
178             if (video->canPlay()) {
179                 video->play();
180                 event->setDefaultHandled();
181             }
182         }
183     }
184 #endif
185
186     if (event->type() == eventNames().keydownEvent && event->isKeyboardEvent()) {
187         HTMLVideoElement* video = descendentVideoElement(targetNode);
188         if (!video)
189             return;
190
191         KeyboardEvent* keyboardEvent = static_cast<KeyboardEvent*>(event);
192         if (keyboardEvent->keyIdentifier() == "U+0020") { // space
193             if (video->paused()) {
194                 if (video->canPlay())
195                     video->play();
196             } else
197                 video->pause();
198             event->setDefaultHandled();
199         }
200     }
201 }
202
203 void MediaDocument::mediaElementSawUnsupportedTracks()
204 {
205     // The HTMLMediaElement was told it has something that the underlying 
206     // MediaPlayer cannot handle so we should switch from <video> to <embed> 
207     // and let the plugin handle this. Don't do it immediately as this 
208     // function may be called directly from a media engine callback, and 
209     // replaceChild will destroy the element, media player, and media engine.
210     m_replaceMediaElementTimer.startOneShot(0);
211 }
212
213 void MediaDocument::replaceMediaElementTimerFired(Timer<MediaDocument>*)
214 {
215     HTMLElement* htmlBody = body();
216     if (!htmlBody)
217         return;
218
219     // Set body margin width and height to 0 as that is what a PluginDocument uses.
220     htmlBody->setAttribute(marginwidthAttr, "0");
221     htmlBody->setAttribute(marginheightAttr, "0");
222
223     if (HTMLVideoElement* videoElement = descendentVideoElement(htmlBody)) {
224         RefPtr<Element> element = Document::createElement(embedTag, false);
225         HTMLEmbedElement* embedElement = static_cast<HTMLEmbedElement*>(element.get());
226
227         embedElement->setAttribute(widthAttr, "100%");
228         embedElement->setAttribute(heightAttr, "100%");
229         embedElement->setAttribute(nameAttr, "plugin");
230         embedElement->setAttribute(srcAttr, url().string());
231
232         DocumentLoader* documentLoader = loader();
233         ASSERT(documentLoader);
234         if (documentLoader)
235             embedElement->setAttribute(typeAttr, documentLoader->writer()->mimeType());
236
237         videoElement->parentNode()->replaceChild(embedElement, videoElement, IGNORE_EXCEPTION);
238     }
239 }
240
241 }
242 #endif