Replace PassRef with Ref/Ref&& across the board.
[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 Ref<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 Ref<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()
205 {
206     ASSERT(scriptExecutionContext()->isDocument());
207     show();
208 }
209 #endif
210
211
212 #if ENABLE(NOTIFICATIONS)
213 const String Notification::permission(ScriptExecutionContext* context)
214 {
215     ASSERT(downcast<Document>(*context).page());
216     return permissionString(NotificationController::from(downcast<Document>(*context).page())->client()->checkPermission(context));
217 }
218
219 const String Notification::permissionString(NotificationClient::Permission permission)
220 {
221     switch (permission) {
222     case NotificationClient::PermissionAllowed:
223         return ASCIILiteral("granted");
224     case NotificationClient::PermissionDenied:
225         return ASCIILiteral("denied");
226     case NotificationClient::PermissionNotAllowed:
227         return ASCIILiteral("default");
228     }
229     
230     ASSERT_NOT_REACHED();
231     return String();
232 }
233
234 void Notification::requestPermission(ScriptExecutionContext* context, PassRefPtr<NotificationPermissionCallback> callback)
235 {
236     ASSERT(downcast<Document>(*context).page());
237     NotificationController::from(downcast<Document>(*context).page())->client()->requestPermission(context, callback);
238 }
239 #endif
240
241 } // namespace WebCore
242
243 #endif // ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)