4e838300672022a3691f0444fb4b5f21f12d817d
[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 "WorkerGlobalScope.h"
53
54 namespace WebCore {
55
56 Notification::Notification()
57     : ActiveDOMObject(0)
58 {
59 }
60
61 #if ENABLE(LEGACY_NOTIFICATIONS)
62 Notification::Notification(const String& title, const String& body, const String& iconURI, ScriptExecutionContext* context, ExceptionCode& ec, PassRefPtr<NotificationCenter> provider)
63     : ActiveDOMObject(context)
64     , m_title(title)
65     , m_body(body)
66     , m_state(Idle)
67     , m_notificationCenter(provider)
68 {
69     if (m_notificationCenter->checkPermission() != NotificationClient::PermissionAllowed) {
70         ec = SECURITY_ERR;
71         return;
72     }
73
74     m_icon = iconURI.isEmpty() ? URL() : scriptExecutionContext()->completeURL(iconURI);
75     if (!m_icon.isEmpty() && !m_icon.isValid()) {
76         ec = SYNTAX_ERR;
77         return;
78     }
79 }
80 #endif
81
82 #if ENABLE(NOTIFICATIONS)
83 Notification::Notification(ScriptExecutionContext& context, const String& title)
84     : ActiveDOMObject(&context)
85     , m_title(title)
86     , m_state(Idle)
87     , m_taskTimer(std::make_unique<Timer>(this, &Notification::taskTimerFired))
88 {
89     m_notificationCenter = DOMWindowNotifications::webkitNotifications(downcast<Document>(context).domWindow());
90     
91     ASSERT(m_notificationCenter->client());
92     m_taskTimer->startOneShot(0);
93 }
94 #endif
95
96 Notification::~Notification() 
97 {
98 }
99
100 #if ENABLE(LEGACY_NOTIFICATIONS)
101 PassRef<Notification> Notification::create(const String& title, const String& body, const String& iconURI, ScriptExecutionContext* context, ExceptionCode& ec, PassRefPtr<NotificationCenter> provider) 
102
103     auto notification = adoptRef(*new Notification(title, body, iconURI, context, ec, provider));
104     notification.get().suspendIfNeeded();
105     return notification;
106 }
107 #endif
108
109 #if ENABLE(NOTIFICATIONS)
110 PassRef<Notification> Notification::create(ScriptExecutionContext& context, const String& title, const Dictionary& options)
111 {
112     auto notification = adoptRef(*new Notification(context, title));
113     String argument;
114     if (options.get("body", argument))
115         notification.get().setBody(argument);
116     if (options.get("tag", argument))
117         notification.get().setTag(argument);
118     if (options.get("lang", argument))
119         notification.get().setLang(argument);
120     if (options.get("dir", argument))
121         notification.get().setDir(argument);
122     if (options.get("icon", argument)) {
123         URL iconURI = argument.isEmpty() ? URL() : context.completeURL(argument);
124         if (!iconURI.isEmpty() && iconURI.isValid())
125             notification.get().setIconURL(iconURI);
126     }
127
128     notification.get().suspendIfNeeded();
129     return notification;
130 }
131 #endif
132
133 void Notification::show() 
134 {
135     // prevent double-showing
136     if (m_state == Idle && m_notificationCenter->client()) {
137 #if ENABLE(NOTIFICATIONS)
138         if (!downcast<Document>(*scriptExecutionContext()).page())
139             return;
140         if (NotificationController::from(downcast<Document>(*scriptExecutionContext()).page())->client()->checkPermission(scriptExecutionContext()) != NotificationClient::PermissionAllowed) {
141             dispatchErrorEvent();
142             return;
143         }
144 #endif
145         if (m_notificationCenter->client()->show(this)) {
146             m_state = Showing;
147             setPendingActivity(this);
148         }
149     }
150 }
151
152 void Notification::close()
153 {
154     switch (m_state) {
155     case Idle:
156         break;
157     case Showing:
158         if (m_notificationCenter->client())
159             m_notificationCenter->client()->cancel(this);
160         break;
161     case Closed:
162         break;
163     }
164 }
165
166 void Notification::contextDestroyed()
167 {
168     ActiveDOMObject::contextDestroyed();
169     if (m_notificationCenter->client())
170         m_notificationCenter->client()->notificationObjectDestroyed(this);
171 }
172
173 void Notification::finalize()
174 {
175     if (m_state == Closed)
176         return;
177     m_state = Closed;
178     unsetPendingActivity(this);
179 }
180
181 void Notification::dispatchShowEvent()
182 {
183     dispatchEvent(Event::create(eventNames().showEvent, false, false));
184 }
185
186 void Notification::dispatchClickEvent()
187 {
188     WindowFocusAllowedIndicator windowFocusAllowed;
189     dispatchEvent(Event::create(eventNames().clickEvent, false, false));
190 }
191
192 void Notification::dispatchCloseEvent()
193 {
194     dispatchEvent(Event::create(eventNames().closeEvent, false, false));
195     finalize();
196 }
197
198 void Notification::dispatchErrorEvent()
199 {
200     dispatchEvent(Event::create(eventNames().errorEvent, false, false));
201 }
202
203 #if ENABLE(NOTIFICATIONS)
204 void Notification::taskTimerFired(Timer& timer)
205 {
206     ASSERT(scriptExecutionContext()->isDocument());
207     ASSERT_UNUSED(timer, &timer == m_taskTimer.get());
208     show();
209 }
210 #endif
211
212
213 #if ENABLE(NOTIFICATIONS)
214 const String Notification::permission(ScriptExecutionContext* context)
215 {
216     ASSERT(downcast<Document>(*context).page());
217     return permissionString(NotificationController::from(downcast<Document>(*context).page())->client()->checkPermission(context));
218 }
219
220 const String Notification::permissionString(NotificationClient::Permission permission)
221 {
222     switch (permission) {
223     case NotificationClient::PermissionAllowed:
224         return ASCIILiteral("granted");
225     case NotificationClient::PermissionDenied:
226         return ASCIILiteral("denied");
227     case NotificationClient::PermissionNotAllowed:
228         return ASCIILiteral("default");
229     }
230     
231     ASSERT_NOT_REACHED();
232     return String();
233 }
234
235 void Notification::requestPermission(ScriptExecutionContext* context, PassRefPtr<NotificationPermissionCallback> callback)
236 {
237     ASSERT(downcast<Document>(*context).page());
238     NotificationController::from(downcast<Document>(*context).page())->client()->requestPermission(context, callback);
239 }
240 #endif
241
242 } // namespace WebCore
243
244 #endif // ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)