Unreviewed, rolling out r126836.
[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) || ENABLE(LEGACY_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) || ENABLE(LEGACY_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) || ENABLE(LEGACY_NOTIFICATIONS)
73     m_permissionsMap = permissions;
74 #endif
75 }
76
77 void WebNotificationManager::didUpdateNotificationDecision(const String& originString, bool allowed)
78 {
79 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
80     m_permissionsMap.set(originString, allowed);
81 #endif
82 }
83
84 void WebNotificationManager::didRemoveNotificationDecisions(const Vector<String>& originStrings)
85 {
86 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_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 NotificationClient::Permission WebNotificationManager::policyForOrigin(WebCore::SecurityOrigin *origin) const
94 {
95 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
96     if (!origin)
97         return NotificationClient::PermissionNotAllowed;
98
99     ASSERT(!origin->isUnique());
100     HashMap<String, bool>::const_iterator it = m_permissionsMap.find(origin->toRawString());
101     if (it != m_permissionsMap.end())
102         return it->second ? NotificationClient::PermissionAllowed : NotificationClient::PermissionDenied;
103 #endif
104     
105     return NotificationClient::PermissionNotAllowed;
106 }
107
108 bool WebNotificationManager::show(Notification* notification, WebPage* page)
109 {
110 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
111     if (!notification || !page->corePage()->settings()->notificationsEnabled())
112         return false;
113     
114     uint64_t notificationID = generateNotificationID();
115     m_notificationMap.set(notification, notificationID);
116     m_notificationIDMap.set(notificationID, notification);
117     
118     NotificationContextMap::iterator it = m_notificationContextMap.add(notification->scriptExecutionContext(), Vector<uint64_t>()).iterator;
119     it->second.append(notificationID);
120
121 #if ENABLE(NOTIFICATIONS)
122     m_process->connection()->send(Messages::WebPageProxy::ShowNotification(notification->title(), notification->body(), notification->iconURL().string(), notification->tag(), notification->scriptExecutionContext()->securityOrigin()->toString(), notificationID), page->pageID());
123 #else
124     m_process->connection()->send(Messages::WebPageProxy::ShowNotification(notification->title(), notification->body(), notification->iconURL().string(), notification->replaceId(), notification->scriptExecutionContext()->securityOrigin()->toString(), notificationID), page->pageID());
125 #endif
126     return true;
127 #else
128     return false;
129 #endif
130 }
131
132 void WebNotificationManager::cancel(Notification* notification, WebPage* page)
133 {
134 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
135     if (!notification || !page->corePage()->settings()->notificationsEnabled())
136         return;
137     
138     uint64_t notificationID = m_notificationMap.get(notification);
139     if (!notificationID)
140         return;
141     
142     m_process->connection()->send(Messages::WebNotificationManagerProxy::Cancel(notificationID), page->pageID());
143 #endif
144 }
145
146 void WebNotificationManager::clearNotifications(WebCore::ScriptExecutionContext* context, WebPage* page)
147 {
148 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
149     NotificationContextMap::iterator it = m_notificationContextMap.find(context);
150     if (it == m_notificationContextMap.end())
151         return;
152     
153     Vector<uint64_t>& notificationIDs = it->second;
154     m_process->connection()->send(Messages::WebNotificationManagerProxy::ClearNotifications(notificationIDs), page->pageID());
155     size_t count = notificationIDs.size();
156     for (size_t i = 0; i < count; ++i) {
157         RefPtr<Notification> notification = m_notificationIDMap.take(notificationIDs[i]);
158         if (!notification)
159             continue;
160         notification->finalize();
161         m_notificationMap.remove(notification);
162     }
163     
164     m_notificationContextMap.remove(it);
165 #endif
166 }
167
168 void WebNotificationManager::didDestroyNotification(Notification* notification, WebPage* page)
169 {
170 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
171     uint64_t notificationID = m_notificationMap.take(notification);
172     if (!notificationID)
173         return;
174
175     m_notificationIDMap.remove(notificationID);
176     removeNotificationFromContextMap(notificationID, notification);
177     m_process->connection()->send(Messages::WebNotificationManagerProxy::DidDestroyNotification(notificationID), page->pageID());
178 #endif
179 }
180
181 void WebNotificationManager::didShowNotification(uint64_t notificationID)
182 {
183 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
184     if (!isNotificationIDValid(notificationID))
185         return;
186     
187     RefPtr<Notification> notification = m_notificationIDMap.get(notificationID);
188     if (!notification)
189         return;
190
191     notification->dispatchShowEvent();
192 #endif
193 }
194
195 void WebNotificationManager::didClickNotification(uint64_t notificationID)
196 {
197 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
198     if (!isNotificationIDValid(notificationID))
199         return;
200
201     RefPtr<Notification> notification = m_notificationIDMap.get(notificationID);
202     if (!notification)
203         return;
204
205     notification->dispatchClickEvent();
206 #endif
207 }
208
209 void WebNotificationManager::didCloseNotifications(const Vector<uint64_t>& notificationIDs)
210 {
211 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
212     size_t count = notificationIDs.size();
213     for (size_t i = 0; i < count; ++i) {
214         uint64_t notificationID = notificationIDs[i];
215         if (!isNotificationIDValid(notificationID))
216             continue;
217
218         RefPtr<Notification> notification = m_notificationIDMap.take(notificationID);
219         if (!notification)
220             continue;
221
222         m_notificationMap.remove(notification);
223         removeNotificationFromContextMap(notificationID, notification.get());
224
225         notification->dispatchCloseEvent();
226     }
227 #endif
228 }
229
230 #if ENABLE(NOTIFICATIONS) || ENABLE(LEGACY_NOTIFICATIONS)
231 void WebNotificationManager::removeNotificationFromContextMap(uint64_t notificationID, Notification* notification)
232 {
233     // This is a helper function for managing the hash maps.
234     NotificationContextMap::iterator it = m_notificationContextMap.find(notification->scriptExecutionContext());
235     ASSERT(it != m_notificationContextMap.end());
236     size_t index = it->second.find(notificationID);
237     ASSERT(index != notFound);
238     it->second.remove(index);
239     if (it->second.isEmpty())
240         m_notificationContextMap.remove(it);
241 }
242 #endif
243
244 } // namespace WebKit