Replace static_casts with to* functions for document types.
[WebKit-https.git] / Source / WebCore / Modules / notifications / Notification.cpp
1 /*
2  * Copyright (C) 2009 Google Inc. All rights reserved.
3  * Copyright (C) 2009, 2011, 2012 Apple Inc. All rights reserved.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are
7  * met:
8  *
9  *     * Redistributions of source code must retain the above copyright
10  * notice, this list of conditions and the following disclaimer.
11  *     * Redistributions in binary form must reproduce the above
12  * copyright notice, this list of conditions and the following disclaimer
13  * in the documentation and/or other materials provided with the
14  * distribution.
15  *     * Neither the name of Google Inc. nor the names of its
16  * contributors may be used to endorse or promote products derived from
17  * this software without specific prior written permission.
18  *
19  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
20  * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
21  * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
22  * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
23  * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
24  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
25  * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
26  * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
27  * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
28  * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
29  * OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
30  */
31
32 #include "config.h"
33
34 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
35
36 #include "Notification.h"
37
38 #include "DOMWindow.h"
39 #include "DOMWindowNotifications.h"
40 #include "Dictionary.h"
41 #include "Document.h"
42 #include "ErrorEvent.h"
43 #include "EventNames.h"
44 #include "NotificationCenter.h"
45 #include "NotificationClient.h"
46 #include "NotificationController.h"
47 #include "NotificationPermissionCallback.h"
48 #include "ResourceRequest.h"
49 #include "ResourceResponse.h"
50 #include "ThreadableLoader.h"
51 #include "WindowFocusAllowedIndicator.h"
52 #include "WorkerContext.h"
53
54 namespace WebCore {
55
56 Notification::Notification()
57     : ActiveDOMObject(0, this)
58 {
59 }
60
61 #if ENABLE(LEGACY_NOTIFICATIONS)
62 Notification::Notification(const KURL& url, ScriptExecutionContext* context, ExceptionCode& ec, PassRefPtr<NotificationCenter> provider)
63     : ActiveDOMObject(context, this)
64     , m_isHTML(true)
65     , m_state(Idle)
66     , m_notificationCenter(provider)
67 {
68     if (m_notificationCenter->checkPermission() != NotificationClient::PermissionAllowed) {
69         ec = SECURITY_ERR;
70         return;
71     }
72
73     if (url.isEmpty() || !url.isValid()) {
74         ec = SYNTAX_ERR;
75         return;
76     }
77
78     m_notificationURL = url;
79 }
80 #endif
81
82 #if ENABLE(LEGACY_NOTIFICATIONS)
83 Notification::Notification(const String& title, const String& body, const String& iconURI, ScriptExecutionContext* context, ExceptionCode& ec, PassRefPtr<NotificationCenter> provider)
84     : ActiveDOMObject(context, this)
85     , m_isHTML(false)
86     , m_title(title)
87     , m_body(body)
88     , m_state(Idle)
89     , m_notificationCenter(provider)
90 {
91     if (m_notificationCenter->checkPermission() != NotificationClient::PermissionAllowed) {
92         ec = SECURITY_ERR;
93         return;
94     }
95
96     m_icon = iconURI.isEmpty() ? KURL() : scriptExecutionContext()->completeURL(iconURI);
97     if (!m_icon.isEmpty() && !m_icon.isValid()) {
98         ec = SYNTAX_ERR;
99         return;
100     }
101 }
102 #endif
103
104 #if ENABLE(NOTIFICATIONS)
105 Notification::Notification(ScriptExecutionContext* context, const String& title)
106     : ActiveDOMObject(context, this)
107     , m_isHTML(false)
108     , m_title(title)
109     , m_state(Idle)
110     , m_taskTimer(adoptPtr(new Timer<Notification>(this, &Notification::taskTimerFired)))
111 {
112     m_notificationCenter = DOMWindowNotifications::webkitNotifications(toDocument(context)->domWindow());
113     
114     ASSERT(m_notificationCenter->client());
115     m_taskTimer->startOneShot(0);
116 }
117 #endif
118
119 Notification::~Notification() 
120 {
121 }
122
123 #if ENABLE(LEGACY_NOTIFICATIONS)
124 PassRefPtr<Notification> Notification::create(const KURL& url, ScriptExecutionContext* context, ExceptionCode& ec, PassRefPtr<NotificationCenter> provider) 
125
126     RefPtr<Notification> notification(adoptRef(new Notification(url, context, ec, provider)));
127     notification->suspendIfNeeded();
128     return notification.release();
129 }
130
131 PassRefPtr<Notification> Notification::create(const String& title, const String& body, const String& iconURI, ScriptExecutionContext* context, ExceptionCode& ec, PassRefPtr<NotificationCenter> provider) 
132
133     RefPtr<Notification> notification(adoptRef(new Notification(title, body, iconURI, context, ec, provider)));
134     notification->suspendIfNeeded();
135     return notification.release();
136 }
137 #endif
138
139 #if ENABLE(NOTIFICATIONS)
140 PassRefPtr<Notification> Notification::create(ScriptExecutionContext* context, const String& title, const Dictionary& options)
141 {
142     RefPtr<Notification> notification(adoptRef(new Notification(context, title)));
143     String argument;
144     if (options.get("body", argument))
145         notification->setBody(argument);
146     if (options.get("tag", argument))
147         notification->setTag(argument);
148     if (options.get("lang", argument))
149         notification->setLang(argument);
150     if (options.get("dir", argument))
151         notification->setDir(argument);
152     if (options.get("icon", argument)) {
153         KURL iconURI = argument.isEmpty() ? KURL() : context->completeURL(argument);
154         if (!iconURI.isEmpty() && iconURI.isValid())
155             notification->setIconURL(iconURI);
156     }
157
158     notification->suspendIfNeeded();
159     return notification.release();
160 }
161 #endif
162
163 const AtomicString& Notification::interfaceName() const
164 {
165     return eventNames().interfaceForNotification;
166 }
167
168 void Notification::show() 
169 {
170     // prevent double-showing
171     if (m_state == Idle && m_notificationCenter->client()) {
172 #if ENABLE(NOTIFICATIONS)
173         if (!toDocument(scriptExecutionContext())->page())
174             return;
175         if (NotificationController::from(toDocument(scriptExecutionContext())->page())->client()->checkPermission(scriptExecutionContext()) != NotificationClient::PermissionAllowed) {
176             dispatchErrorEvent();
177             return;
178         }
179 #endif
180         if (m_notificationCenter->client()->show(this)) {
181             m_state = Showing;
182             setPendingActivity(this);
183         }
184     }
185 }
186
187 void Notification::close()
188 {
189     switch (m_state) {
190     case Idle:
191         break;
192     case Showing:
193         if (m_notificationCenter->client())
194             m_notificationCenter->client()->cancel(this);
195         break;
196     case Closed:
197         break;
198     }
199 }
200
201 EventTargetData* Notification::eventTargetData()
202 {
203     return &m_eventTargetData;
204 }
205
206 EventTargetData* Notification::ensureEventTargetData()
207 {
208     return &m_eventTargetData;
209 }
210
211 void Notification::contextDestroyed()
212 {
213     ActiveDOMObject::contextDestroyed();
214     if (m_notificationCenter->client())
215         m_notificationCenter->client()->notificationObjectDestroyed(this);
216 }
217
218 void Notification::finalize()
219 {
220     if (m_state == Closed)
221         return;
222     m_state = Closed;
223     unsetPendingActivity(this);
224 }
225
226 void Notification::dispatchShowEvent()
227 {
228     dispatchEvent(Event::create(eventNames().showEvent, false, false));
229 }
230
231 void Notification::dispatchClickEvent()
232 {
233     WindowFocusAllowedIndicator windowFocusAllowed;
234     dispatchEvent(Event::create(eventNames().clickEvent, false, false));
235 }
236
237 void Notification::dispatchCloseEvent()
238 {
239     dispatchEvent(Event::create(eventNames().closeEvent, false, false));
240     finalize();
241 }
242
243 void Notification::dispatchErrorEvent()
244 {
245     dispatchEvent(Event::create(eventNames().errorEvent, false, false));
246 }
247
248 #if ENABLE(NOTIFICATIONS)
249 void Notification::taskTimerFired(Timer<Notification>* timer)
250 {
251     ASSERT(scriptExecutionContext()->isDocument());
252     ASSERT_UNUSED(timer, timer == m_taskTimer.get());
253     show();
254 }
255 #endif
256
257
258 #if ENABLE(NOTIFICATIONS)
259 const String& Notification::permission(ScriptExecutionContext* context)
260 {
261     ASSERT(toDocument(context)->page());
262     return permissionString(NotificationController::from(toDocument(context)->page())->client()->checkPermission(context));
263 }
264
265 const String& Notification::permissionString(NotificationClient::Permission permission)
266 {
267     DEFINE_STATIC_LOCAL(const String, allowedPermission, (ASCIILiteral("granted")));
268     DEFINE_STATIC_LOCAL(const String, deniedPermission, (ASCIILiteral("denied")));
269     DEFINE_STATIC_LOCAL(const String, defaultPermission, (ASCIILiteral("default")));
270
271     switch (permission) {
272     case NotificationClient::PermissionAllowed:
273         return allowedPermission;
274     case NotificationClient::PermissionDenied:
275         return deniedPermission;
276     case NotificationClient::PermissionNotAllowed:
277         return defaultPermission;
278     }
279     
280     ASSERT_NOT_REACHED();
281     return deniedPermission;
282 }
283
284 void Notification::requestPermission(ScriptExecutionContext* context, PassRefPtr<NotificationPermissionCallback> callback)
285 {
286     ASSERT(toDocument(context)->page());
287     NotificationController::from(toDocument(context)->page())->client()->requestPermission(context, callback);
288 }
289 #endif
290
291 } // namespace WebCore
292
293 #endif // ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)