Add support for notification replaceId in Mac WebKit and WK2
[WebKit-https.git] / Source / WebKit2 / WebProcess / Notifications / WebNotificationManager.cpp
1 /*
2  * Copyright (C) 2011, 2012 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. AND ITS CONTRIBUTORS ``AS IS''
14  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
15  * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
16  * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
17  * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
19  * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
20  * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
21  * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
22  * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
23  * THE POSSIBILITY OF SUCH DAMAGE.
24  */
25
26 #include "config.h"
27 #include "WebNotificationManager.h"
28
29 #include "WebPage.h"
30 #include "WebProcess.h"
31
32 #if ENABLE(NOTIFICATIONS)
33 #include "WebNotification.h"
34 #include "WebNotificationManagerProxyMessages.h"
35 #include "WebPageProxyMessages.h"
36 #include <WebCore/Document.h>
37 #include <WebCore/Notification.h>
38 #include <WebCore/Page.h>
39 #include <WebCore/ScriptExecutionContext.h>
40 #include <WebCore/SecurityOrigin.h>
41 #include <WebCore/Settings.h>
42 #endif
43
44 using namespace WebCore;
45
46 namespace WebKit {
47
48 #if ENABLE(NOTIFICATIONS)
49 static uint64_t generateNotificationID()
50 {
51     static uint64_t uniqueNotificationID = 1;
52     return uniqueNotificationID++;
53 }
54 #endif
55
56 WebNotificationManager::WebNotificationManager(WebProcess* process)
57     : m_process(process)
58 {
59 }
60
61 WebNotificationManager::~WebNotificationManager()
62 {
63 }
64
65 void WebNotificationManager::didReceiveMessage(CoreIPC::Connection* connection, CoreIPC::MessageID messageID, CoreIPC::ArgumentDecoder* arguments)
66 {
67     didReceiveWebNotificationManagerMessage(connection, messageID, arguments);
68 }
69
70 void WebNotificationManager::initialize(const HashMap<String, bool>& permissions)
71 {
72 #if ENABLE(NOTIFICATIONS)
73     m_permissionsMap = permissions;
74 #endif
75 }
76
77 void WebNotificationManager::didUpdateNotificationDecision(const String& originString, bool allowed)
78 {
79 #if ENABLE(NOTIFICATIONS)
80     m_permissionsMap.set(originString, allowed);
81 #endif
82 }
83
84 void WebNotificationManager::didRemoveNotificationDecisions(const Vector<String>& originStrings)
85 {
86 #if ENABLE(NOTIFICATIONS)
87     size_t count = originStrings.size();
88     for (size_t i = 0; i < count; ++i)
89         m_permissionsMap.remove(originStrings[i]);
90 #endif
91 }
92
93 NotificationPresenter::Permission WebNotificationManager::policyForOrigin(WebCore::SecurityOrigin *origin) const
94 {
95 #if ENABLE(NOTIFICATIONS)
96     if (!origin)
97         return NotificationPresenter::PermissionNotAllowed;
98     
99     HashMap<String, bool>::const_iterator it = m_permissionsMap.find(origin->toString());
100     if (it != m_permissionsMap.end())
101         return it->second ? NotificationPresenter::PermissionAllowed : NotificationPresenter::PermissionDenied;
102 #endif
103     
104     return NotificationPresenter::PermissionNotAllowed;
105 }
106
107 bool WebNotificationManager::show(Notification* notification, WebPage* page)
108 {
109 #if ENABLE(NOTIFICATIONS)
110     if (!notification || !page->corePage()->settings()->notificationsEnabled())
111         return true;
112     
113     uint64_t notificationID = generateNotificationID();
114     m_notificationMap.set(notification, notificationID);
115     m_notificationIDMap.set(notificationID, notification);
116     
117     NotificationContextMap::iterator it = m_notificationContextMap.add(notification->scriptExecutionContext(), Vector<uint64_t>()).first;
118     it->second.append(notificationID);
119     
120     m_process->connection()->send(Messages::WebPageProxy::ShowNotification(notification->contents().title, notification->contents().body, notification->iconURL().string(), notification->replaceId(), notification->scriptExecutionContext()->securityOrigin()->toString(), notificationID), page->pageID());
121     return true;
122 #else
123     return false;
124 #endif
125 }
126
127 void WebNotificationManager::cancel(Notification* notification, WebPage* page)
128 {
129 #if ENABLE(NOTIFICATIONS)
130     if (!notification || !page->corePage()->settings()->notificationsEnabled())
131         return;
132     
133     uint64_t notificationID = m_notificationMap.get(notification);
134     if (!notificationID)
135         return;
136     
137     m_process->connection()->send(Messages::WebNotificationManagerProxy::Cancel(notificationID), page->pageID());
138 #endif
139 }
140
141 void WebNotificationManager::clearNotifications(WebCore::ScriptExecutionContext* context, WebPage* page)
142 {
143 #if ENABLE(NOTIFICATIONS)
144     NotificationContextMap::iterator it = m_notificationContextMap.find(context);
145     if (it == m_notificationContextMap.end())
146         return;
147     
148     Vector<uint64_t>& notificationIDs = it->second;
149     m_process->connection()->send(Messages::WebNotificationManagerProxy::ClearNotifications(notificationIDs), page->pageID());
150     size_t count = notificationIDs.size();
151     for (size_t i = 0; i < count; ++i) {
152         RefPtr<Notification> notification = m_notificationIDMap.take(notificationIDs[i]);
153         if (!notification)
154             continue;
155         m_notificationMap.remove(notification);
156     }
157     
158     m_notificationContextMap.remove(it);
159 #endif
160 }
161
162 void WebNotificationManager::didDestroyNotification(Notification* notification, WebPage* page)
163 {
164 #if ENABLE(NOTIFICATIONS)
165     uint64_t notificationID = m_notificationMap.take(notification);
166     if (!notificationID)
167         return;
168
169     m_notificationIDMap.remove(notificationID);
170     removeNotificationFromContextMap(notificationID, notification);
171     m_process->connection()->send(Messages::WebNotificationManagerProxy::DidDestroyNotification(notificationID), page->pageID());
172 #endif
173 }
174
175 void WebNotificationManager::didShowNotification(uint64_t notificationID)
176 {
177 #if ENABLE(NOTIFICATIONS)
178     if (!isNotificationIDValid(notificationID))
179         return;
180     
181     RefPtr<Notification> notification = m_notificationIDMap.get(notificationID);
182     if (!notification)
183         return;
184
185     notification->dispatchShowEvent();
186 #endif
187 }
188
189 void WebNotificationManager::didClickNotification(uint64_t notificationID)
190 {
191 #if ENABLE(NOTIFICATIONS)
192     if (!isNotificationIDValid(notificationID))
193         return;
194
195     RefPtr<Notification> notification = m_notificationIDMap.get(notificationID);
196     if (!notification)
197         return;
198
199     notification->dispatchClickEvent();
200 #endif
201 }
202
203 void WebNotificationManager::didCloseNotifications(const Vector<uint64_t>& notificationIDs)
204 {
205 #if ENABLE(NOTIFICATIONS)
206     size_t count = notificationIDs.size();
207     for (size_t i = 0; i < count; ++i) {
208         uint64_t notificationID = notificationIDs[i];
209         if (!isNotificationIDValid(notificationID))
210             continue;
211
212         RefPtr<Notification> notification = m_notificationIDMap.take(notificationID);
213         if (!notification)
214             continue;
215
216         m_notificationMap.remove(notification);
217         removeNotificationFromContextMap(notificationID, notification.get());
218
219         notification->dispatchCloseEvent();
220     }
221 #endif
222 }
223
224 #if ENABLE(NOTIFICATIONS)
225 void WebNotificationManager::removeNotificationFromContextMap(uint64_t notificationID, Notification* notification)
226 {
227     // This is a helper function for managing the hash maps.
228     NotificationContextMap::iterator it = m_notificationContextMap.find(notification->scriptExecutionContext());
229     ASSERT(it != m_notificationContextMap.end());
230     size_t index = it->second.find(notificationID);
231     ASSERT(index != notFound);
232     it->second.remove(index);
233     if (it->second.isEmpty())
234         m_notificationContextMap.remove(it);
235 }
236 #endif
237
238 } // namespace WebKit