#include "content/browser/notification_service_impl.h"
#include "base/logging.h"
#include "base/observer_list.h"
#include "content/public/browser/notification_observer.h"
#include "content/public/browser/notification_types.h"
#include "third_party/abseil-cpp/absl/base/attributes.h"
namespace content {
namespace {
ABSL_CONST_INIT thread_local NotificationServiceImpl* notification_service =
nullptr;
}
NotificationServiceImpl* NotificationServiceImpl::current() {
return notification_service;
}
NotificationService* NotificationService::current() {
return NotificationServiceImpl::current();
}
NotificationService* NotificationService::Create() {
return new NotificationServiceImpl;
}
bool NotificationServiceImpl::HasKey(const NotificationSourceMap& map,
const NotificationSource& source) {
return map.find(source.map_key()) != map.end();
}
NotificationServiceImpl::NotificationServiceImpl()
: resetter_(¬ification_service, this, nullptr) {}
void NotificationServiceImpl::AddObserver(NotificationObserver* observer,
int type,
const NotificationSource& source) {
CHECK(observer);
NotificationObserverList* observer_list;
if (HasKey(observers_[type], source)) {
observer_list = observers_[type][source.map_key()];
} else {
observer_list = new NotificationObserverList;
observers_[type][source.map_key()] = observer_list;
}
observer_list->AddObserver(observer);
#ifndef NDEBUG
++observer_counts_[type];
#endif
}
void NotificationServiceImpl::RemoveObserver(NotificationObserver* observer,
int type,
const NotificationSource& source) {
CHECK(HasKey(observers_[type], source));
NotificationObserverList* observer_list =
observers_[type][source.map_key()];
if (observer_list) {
observer_list->RemoveObserver(observer);
if (observer_list->empty()) {
observers_[type].erase(source.map_key());
delete observer_list;
}
#ifndef NDEBUG
--observer_counts_[type];
#endif
}
}
void NotificationServiceImpl::Notify(int type,
const NotificationSource& source,
const NotificationDetails& details) {
if (HasKey(observers_[type], AllSources()) &&
source != AllSources()) {
for (auto& observer : *observers_[type][AllSources().map_key()])
observer.Observe(type, source, details);
}
if (HasKey(observers_[type], source)) {
for (auto& observer : *observers_[type][source.map_key()])
observer.Observe(type, source, details);
}
}
NotificationServiceImpl::~NotificationServiceImpl() {
#ifndef NDEBUG
for (int i = 0; i < static_cast<int>(observer_counts_.size()); i++) {
if (observer_counts_[i] > 0) {
VLOG(1) << observer_counts_[i] << " notification observer(s) leaked "
"of notification type " << i;
}
}
#endif
for (int i = 0; i < static_cast<int>(observers_.size()); i++) {
NotificationSourceMap omap = observers_[i];
for (auto it = omap.begin(); it != omap.end(); ++it)
delete it->second;
}
}
}