Unreviewed, rolling out r140983.
[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     ASSERT(context->isDocument());
113     m_notificationCenter = DOMWindowNotifications::webkitNotifications(static_cast<Document*>(context)->domWindow());
114     
115     ASSERT(m_notificationCenter->client());
116     m_taskTimer->startOneShot(0);
117 }
118 #endif
119
120 Notification::~Notification() 
121 {
122 }
123
124 #if ENABLE(LEGACY_NOTIFICATIONS)
125 PassRefPtr<Notification> Notification::create(const KURL& url, ScriptExecutionContext* context, ExceptionCode& ec, PassRefPtr<NotificationCenter> provider) 
126
127     RefPtr<Notification> notification(adoptRef(new Notification(url, context, ec, provider)));
128     notification->suspendIfNeeded();
129     return notification.release();
130 }
131
132 PassRefPtr<Notification> Notification::create(const String& title, const String& body, const String& iconURI, ScriptExecutionContext* context, ExceptionCode& ec, PassRefPtr<NotificationCenter> provider) 
133
134     RefPtr<Notification> notification(adoptRef(new Notification(title, body, iconURI, context, ec, provider)));
135     notification->suspendIfNeeded();
136     return notification.release();
137 }
138 #endif
139
140 #if ENABLE(NOTIFICATIONS)
141 PassRefPtr<Notification> Notification::create(ScriptExecutionContext* context, const String& title, const Dictionary& options)
142 {
143     RefPtr<Notification> notification(adoptRef(new Notification(context, title)));
144     String argument;
145     if (options.get("body", argument))
146         notification->setBody(argument);
147     if (options.get("tag", argument))
148         notification->setTag(argument);
149     if (options.get("lang", argument))
150         notification->setLang(argument);
151     if (options.get("dir", argument))
152         notification->setDir(argument);
153     if (options.get("icon", argument)) {
154         KURL iconURI = argument.isEmpty() ? KURL() : context->completeURL(argument);
155         if (!iconURI.isEmpty() && iconURI.isValid())
156             notification->setIconURL(iconURI);
157     }
158
159     notification->suspendIfNeeded();
160     return notification.release();
161 }
162 #endif
163
164 const AtomicString& Notification::interfaceName() const
165 {
166     return eventNames().interfaceForNotification;
167 }
168
169 void Notification::show() 
170 {
171     // prevent double-showing
172     if (m_state == Idle && m_notificationCenter->client()) {
173 #if ENABLE(NOTIFICATIONS)
174         if (!static_cast<Document*>(scriptExecutionContext())->page())
175             return;
176         if (NotificationController::from(static_cast<Document*>(scriptExecutionContext())->page())->client()->checkPermission(scriptExecutionContext()) != NotificationClient::PermissionAllowed) {
177             dispatchErrorEvent();
178             return;
179         }
180 #endif
181         if (m_notificationCenter->client()->show(this)) {
182             m_state = Showing;
183             setPendingActivity(this);
184         }
185     }
186 }
187
188 void Notification::close()
189 {
190     switch (m_state) {
191     case Idle:
192         break;
193     case Showing:
194         if (m_notificationCenter->client())
195             m_notificationCenter->client()->cancel(this);
196         break;
197     case Closed:
198         break;
199     }
200 }
201
202 EventTargetData* Notification::eventTargetData()
203 {
204     return &m_eventTargetData;
205 }
206
207 EventTargetData* Notification::ensureEventTargetData()
208 {
209     return &m_eventTargetData;
210 }
211
212 void Notification::contextDestroyed()
213 {
214     ActiveDOMObject::contextDestroyed();
215     if (m_notificationCenter->client())
216         m_notificationCenter->client()->notificationObjectDestroyed(this);
217 }
218
219 void Notification::finalize()
220 {
221     if (m_state == Closed)
222         return;
223     m_state = Closed;
224     unsetPendingActivity(this);
225 }
226
227 void Notification::dispatchShowEvent()
228 {
229     dispatchEvent(Event::create(eventNames().showEvent, false, false));
230 }
231
232 void Notification::dispatchClickEvent()
233 {
234     WindowFocusAllowedIndicator windowFocusAllowed;
235     dispatchEvent(Event::create(eventNames().clickEvent, false, false));
236 }
237
238 void Notification::dispatchCloseEvent()
239 {
240     dispatchEvent(Event::create(eventNames().closeEvent, false, false));
241     finalize();
242 }
243
244 void Notification::dispatchErrorEvent()
245 {
246     dispatchEvent(Event::create(eventNames().errorEvent, false, false));
247 }
248
249 #if ENABLE(NOTIFICATIONS)
250 void Notification::taskTimerFired(Timer<Notification>* timer)
251 {
252     ASSERT(scriptExecutionContext()->isDocument());
253     ASSERT_UNUSED(timer, timer == m_taskTimer.get());
254     show();
255 }
256 #endif
257
258
259 #if ENABLE(NOTIFICATIONS)
260 const String& Notification::permission(ScriptExecutionContext* context)
261 {
262     ASSERT(context->isDocument());
263     ASSERT(static_cast<Document*>(context)->page());
264     return permissionString(NotificationController::from(static_cast<Document*>(context)->page())->client()->checkPermission(context));
265 }
266
267 const String& Notification::permissionString(NotificationClient::Permission permission)
268 {
269     DEFINE_STATIC_LOCAL(const String, allowedPermission, (ASCIILiteral("granted")));
270     DEFINE_STATIC_LOCAL(const String, deniedPermission, (ASCIILiteral("denied")));
271     DEFINE_STATIC_LOCAL(const String, defaultPermission, (ASCIILiteral("default")));
272
273     switch (permission) {
274     case NotificationClient::PermissionAllowed:
275         return allowedPermission;
276     case NotificationClient::PermissionDenied:
277         return deniedPermission;
278     case NotificationClient::PermissionNotAllowed:
279         return defaultPermission;
280     }
281     
282     ASSERT_NOT_REACHED();
283     return deniedPermission;
284 }
285
286 void Notification::requestPermission(ScriptExecutionContext* context, PassRefPtr<NotificationPermissionCallback> callback)
287 {
288     ASSERT(context->isDocument());
289     ASSERT(static_cast<Document*>(context)->page());
290     NotificationController::from(static_cast<Document*>(context)->page())->client()->requestPermission(context, callback);
291 }
292 #endif
293
294 } // namespace WebCore
295
296 #endif // ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)