blob: e6de02b091fe43db25b404f28424177b3faecba1 [file] [log] [blame]
// Copyright 2013 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "ui/message_center/message_center_impl.h"
#include "base/bind.h"
#include "base/message_loop/message_loop.h"
#include "base/run_loop.h"
#include "base/strings/utf_string_conversions.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "ui/message_center/message_center.h"
#include "ui/message_center/message_center_types.h"
#include "ui/message_center/notification_blocker.h"
#include "ui/message_center/notification_types.h"
namespace message_center {
namespace {
class MessageCenterImplTest : public testing::Test,
public MessageCenterObserver {
public:
MessageCenterImplTest() {}
virtual void SetUp() OVERRIDE {
MessageCenter::Initialize();
message_center_ = MessageCenter::Get();
loop_.reset(new base::MessageLoop(base::MessageLoop::TYPE_DEFAULT));
run_loop_.reset(new base::RunLoop());
closure_ = run_loop_->QuitClosure();
}
virtual void TearDown() OVERRIDE {
run_loop_.reset();
loop_.reset();
message_center_ = NULL;
MessageCenter::Shutdown();
}
MessageCenter* message_center() const { return message_center_; }
base::RunLoop* run_loop() const { return run_loop_.get(); }
base::Closure closure() const { return closure_; }
private:
MessageCenter* message_center_;
scoped_ptr<base::MessageLoop> loop_;
scoped_ptr<base::RunLoop> run_loop_;
base::Closure closure_;
DISALLOW_COPY_AND_ASSIGN(MessageCenterImplTest);
};
class ToggledNotificationBlocker : public NotificationBlocker {
public:
explicit ToggledNotificationBlocker(MessageCenter* message_center)
: NotificationBlocker(message_center),
notifications_enabled_(true) {}
virtual ~ToggledNotificationBlocker() {}
void SetNotificationsEnabled(bool enabled) {
if (notifications_enabled_ != enabled) {
notifications_enabled_ = enabled;
FOR_EACH_OBSERVER(
NotificationBlocker::Observer, observers(), OnBlockingStateChanged());
}
}
// NotificationBlocker overrides:
virtual bool ShouldShowNotificationAsPopup(
const message_center::NotifierId& notifier_id) const OVERRIDE {
return notifications_enabled_;
}
private:
bool notifications_enabled_;
DISALLOW_COPY_AND_ASSIGN(ToggledNotificationBlocker);
};
class PopupNotificationBlocker : public ToggledNotificationBlocker {
public:
PopupNotificationBlocker(MessageCenter* message_center,
const NotifierId& allowed_notifier)
: ToggledNotificationBlocker(message_center),
allowed_notifier_(allowed_notifier) {}
virtual ~PopupNotificationBlocker() {}
// NotificationBlocker overrides:
virtual bool ShouldShowNotificationAsPopup(
const NotifierId& notifier_id) const OVERRIDE {
return (notifier_id == allowed_notifier_) ||
ToggledNotificationBlocker::ShouldShowNotificationAsPopup(notifier_id);
}
private:
NotifierId allowed_notifier_;
DISALLOW_COPY_AND_ASSIGN(PopupNotificationBlocker);
};
bool PopupNotificationsContain(
const NotificationList::PopupNotifications& popups,
const std::string& id) {
for (NotificationList::PopupNotifications::const_iterator iter =
popups.begin(); iter != popups.end(); ++iter) {
if ((*iter)->id() == id)
return true;
}
return false;
}
} // namespace
namespace internal {
class MockPopupTimersController : public PopupTimersController {
public:
MockPopupTimersController(MessageCenter* message_center,
base::Closure quit_closure)
: PopupTimersController(message_center),
timer_finished_(false),
quit_closure_(quit_closure) {}
virtual ~MockPopupTimersController() {}
virtual void TimerFinished(const std::string& id) OVERRIDE {
LOG(INFO) << "In timer finished for id " << id;
base::MessageLoop::current()->PostTask(FROM_HERE, quit_closure_);
timer_finished_ = true;
last_id_ = id;
}
bool timer_finished() const { return timer_finished_; }
const std::string& last_id() const { return last_id_; }
private:
bool timer_finished_;
std::string last_id_;
base::Closure quit_closure_;
};
TEST_F(MessageCenterImplTest, PopupTimersEmptyController) {
scoped_ptr<PopupTimersController> popup_timers_controller =
make_scoped_ptr(new PopupTimersController(message_center()));
// Test that all functions succed without any timers created.
popup_timers_controller->PauseAll();
popup_timers_controller->StartAll();
popup_timers_controller->CancelAll();
popup_timers_controller->TimerFinished("unknown");
popup_timers_controller->PauseTimer("unknown");
popup_timers_controller->CancelTimer("unknown");
}
TEST_F(MessageCenterImplTest, PopupTimersControllerStartTimer) {
scoped_ptr<MockPopupTimersController> popup_timers_controller =
make_scoped_ptr(
new MockPopupTimersController(message_center(), closure()));
popup_timers_controller->StartTimer("test",
base::TimeDelta::FromMilliseconds(1));
run_loop()->Run();
EXPECT_TRUE(popup_timers_controller->timer_finished());
}
TEST_F(MessageCenterImplTest, PopupTimersControllerPauseTimer) {
scoped_ptr<MockPopupTimersController> popup_timers_controller =
make_scoped_ptr(
new MockPopupTimersController(message_center(), closure()));
popup_timers_controller->StartTimer("test",
base::TimeDelta::FromMilliseconds(1));
popup_timers_controller->PauseTimer("test");
run_loop()->RunUntilIdle();
EXPECT_FALSE(popup_timers_controller->timer_finished());
}
TEST_F(MessageCenterImplTest, PopupTimersControllerCancelTimer) {
scoped_ptr<MockPopupTimersController> popup_timers_controller =
make_scoped_ptr(
new MockPopupTimersController(message_center(), closure()));
popup_timers_controller->StartTimer("test",
base::TimeDelta::FromMilliseconds(1));
popup_timers_controller->CancelTimer("test");
run_loop()->RunUntilIdle();
EXPECT_FALSE(popup_timers_controller->timer_finished());
}
TEST_F(MessageCenterImplTest, PopupTimersControllerPauseAllTimers) {
scoped_ptr<MockPopupTimersController> popup_timers_controller =
make_scoped_ptr(
new MockPopupTimersController(message_center(), closure()));
popup_timers_controller->StartTimer("test",
base::TimeDelta::FromMilliseconds(1));
popup_timers_controller->PauseAll();
run_loop()->RunUntilIdle();
EXPECT_FALSE(popup_timers_controller->timer_finished());
}
TEST_F(MessageCenterImplTest, PopupTimersControllerStartAllTimers) {
scoped_ptr<MockPopupTimersController> popup_timers_controller =
make_scoped_ptr(
new MockPopupTimersController(message_center(), closure()));
popup_timers_controller->StartTimer("test",
base::TimeDelta::FromMilliseconds(1));
popup_timers_controller->PauseAll();
popup_timers_controller->StartAll();
run_loop()->Run();
EXPECT_TRUE(popup_timers_controller->timer_finished());
}
TEST_F(MessageCenterImplTest, PopupTimersControllerStartMultipleTimers) {
scoped_ptr<MockPopupTimersController> popup_timers_controller =
make_scoped_ptr(
new MockPopupTimersController(message_center(), closure()));
popup_timers_controller->StartTimer("test",
base::TimeDelta::FromMilliseconds(5));
popup_timers_controller->StartTimer("test2",
base::TimeDelta::FromMilliseconds(1));
popup_timers_controller->StartTimer("test3",
base::TimeDelta::FromMilliseconds(3));
popup_timers_controller->PauseAll();
popup_timers_controller->StartAll();
run_loop()->Run();
EXPECT_EQ(popup_timers_controller->last_id(), "test2");
EXPECT_TRUE(popup_timers_controller->timer_finished());
}
TEST_F(MessageCenterImplTest, PopupTimersControllerStartMultipleTimersPause) {
scoped_ptr<MockPopupTimersController> popup_timers_controller =
make_scoped_ptr(
new MockPopupTimersController(message_center(), closure()));
popup_timers_controller->StartTimer("test",
base::TimeDelta::FromMilliseconds(5));
popup_timers_controller->StartTimer("test2",
base::TimeDelta::FromMilliseconds(1));
popup_timers_controller->StartTimer("test3",
base::TimeDelta::FromMilliseconds(3));
popup_timers_controller->PauseTimer("test2");
run_loop()->Run();
EXPECT_EQ(popup_timers_controller->last_id(), "test3");
EXPECT_TRUE(popup_timers_controller->timer_finished());
}
TEST_F(MessageCenterImplTest, PopupTimersControllerResetTimer) {
scoped_ptr<MockPopupTimersController> popup_timers_controller =
make_scoped_ptr(
new MockPopupTimersController(message_center(), closure()));
popup_timers_controller->StartTimer("test",
base::TimeDelta::FromMilliseconds(5));
popup_timers_controller->StartTimer("test2",
base::TimeDelta::FromMilliseconds(1));
popup_timers_controller->StartTimer("test3",
base::TimeDelta::FromMilliseconds(3));
popup_timers_controller->PauseTimer("test2");
popup_timers_controller->ResetTimer("test",
base::TimeDelta::FromMilliseconds(2));
run_loop()->Run();
EXPECT_EQ(popup_timers_controller->last_id(), "test");
EXPECT_TRUE(popup_timers_controller->timer_finished());
}
TEST_F(MessageCenterImplTest, NotificationBlocker) {
NotifierId notifier_id(NotifierId::APPLICATION, "app1");
// Multiple blockers to verify the case that one blocker blocks but another
// doesn't.
ToggledNotificationBlocker blocker1(message_center());
ToggledNotificationBlocker blocker2(message_center());
message_center()->AddNotification(scoped_ptr<Notification>(new Notification(
NOTIFICATION_TYPE_SIMPLE,
"id1",
UTF8ToUTF16("title"),
UTF8ToUTF16("message"),
gfx::Image() /* icon */,
base::string16() /* display_source */,
notifier_id,
RichNotificationData(),
NULL)));
message_center()->AddNotification(scoped_ptr<Notification>(new Notification(
NOTIFICATION_TYPE_SIMPLE,
"id2",
UTF8ToUTF16("title"),
UTF8ToUTF16("message"),
gfx::Image() /* icon */,
base::string16() /* display_source */,
notifier_id,
RichNotificationData(),
NULL)));
EXPECT_EQ(2u, message_center()->GetPopupNotifications().size());
EXPECT_EQ(2u, message_center()->GetNotifications().size());
// Block all notifications. All popups are gone and message center should be
// hidden.
blocker1.SetNotificationsEnabled(false);
EXPECT_TRUE(message_center()->GetPopupNotifications().empty());
EXPECT_EQ(2u, message_center()->GetNotifications().size());
// Updates |blocker2| state, which doesn't affect the global state.
blocker2.SetNotificationsEnabled(false);
EXPECT_TRUE(message_center()->GetPopupNotifications().empty());
EXPECT_EQ(2u, message_center()->GetNotifications().size());
blocker2.SetNotificationsEnabled(true);
EXPECT_TRUE(message_center()->GetPopupNotifications().empty());
EXPECT_EQ(2u, message_center()->GetNotifications().size());
// If |blocker2| blocks, then unblocking blocker1 doesn't change the global
// state.
blocker2.SetNotificationsEnabled(false);
blocker1.SetNotificationsEnabled(true);
EXPECT_TRUE(message_center()->GetPopupNotifications().empty());
EXPECT_EQ(2u, message_center()->GetNotifications().size());
// Unblock both blockers, which recovers the global state, but the popups
// aren't shown.
blocker2.SetNotificationsEnabled(true);
EXPECT_TRUE(message_center()->GetPopupNotifications().empty());
EXPECT_EQ(2u, message_center()->GetNotifications().size());
}
TEST_F(MessageCenterImplTest, NotificationsDuringBlocked) {
NotifierId notifier_id(NotifierId::APPLICATION, "app1");
ToggledNotificationBlocker blocker(message_center());
message_center()->AddNotification(scoped_ptr<Notification>(new Notification(
NOTIFICATION_TYPE_SIMPLE,
"id1",
UTF8ToUTF16("title"),
UTF8ToUTF16("message"),
gfx::Image() /* icon */,
base::string16() /* display_source */,
notifier_id,
RichNotificationData(),
NULL)));
EXPECT_EQ(1u, message_center()->GetPopupNotifications().size());
EXPECT_EQ(1u, message_center()->GetNotifications().size());
// Create a notification during blocked. Still no popups.
blocker.SetNotificationsEnabled(false);
message_center()->AddNotification(scoped_ptr<Notification>(new Notification(
NOTIFICATION_TYPE_SIMPLE,
"id2",
UTF8ToUTF16("title"),
UTF8ToUTF16("message"),
gfx::Image() /* icon */,
base::string16() /* display_source */,
notifier_id,
RichNotificationData(),
NULL)));
EXPECT_TRUE(message_center()->GetPopupNotifications().empty());
EXPECT_EQ(2u, message_center()->GetNotifications().size());
// Unblock notifications, the id1 should appear as a popup.
blocker.SetNotificationsEnabled(true);
NotificationList::PopupNotifications popups =
message_center()->GetPopupNotifications();
EXPECT_EQ(1u, popups.size());
EXPECT_TRUE(PopupNotificationsContain(popups, "id2"));
EXPECT_EQ(2u, message_center()->GetNotifications().size());
}
// Similar to other blocker cases but this test case allows |notifier_id2| even
// in blocked.
TEST_F(MessageCenterImplTest, NotificationBlockerAllowsPopups) {
NotifierId notifier_id1(NotifierId::APPLICATION, "app1");
NotifierId notifier_id2(NotifierId::APPLICATION, "app2");
PopupNotificationBlocker blocker(message_center(), notifier_id2);
message_center()->AddNotification(scoped_ptr<Notification>(new Notification(
NOTIFICATION_TYPE_SIMPLE,
"id1",
UTF8ToUTF16("title"),
UTF8ToUTF16("message"),
gfx::Image() /* icon */,
base::string16() /* display_source */,
notifier_id1,
RichNotificationData(),
NULL)));
message_center()->AddNotification(scoped_ptr<Notification>(new Notification(
NOTIFICATION_TYPE_SIMPLE,
"id2",
UTF8ToUTF16("title"),
UTF8ToUTF16("message"),
gfx::Image() /* icon */,
base::string16() /* display_source */,
notifier_id2,
RichNotificationData(),
NULL)));
// "id1" is closed but "id2" is still visible as a popup.
blocker.SetNotificationsEnabled(false);
NotificationList::PopupNotifications popups =
message_center()->GetPopupNotifications();
EXPECT_EQ(1u, popups.size());
EXPECT_TRUE(PopupNotificationsContain(popups, "id2"));
EXPECT_EQ(2u, message_center()->GetNotifications().size());
message_center()->AddNotification(scoped_ptr<Notification>(new Notification(
NOTIFICATION_TYPE_SIMPLE,
"id3",
UTF8ToUTF16("title"),
UTF8ToUTF16("message"),
gfx::Image() /* icon */,
base::string16() /* display_source */,
notifier_id1,
RichNotificationData(),
NULL)));
message_center()->AddNotification(scoped_ptr<Notification>(new Notification(
NOTIFICATION_TYPE_SIMPLE,
"id4",
UTF8ToUTF16("title"),
UTF8ToUTF16("message"),
gfx::Image() /* icon */,
base::string16() /* display_source */,
notifier_id2,
RichNotificationData(),
NULL)));
popups = message_center()->GetPopupNotifications();
EXPECT_EQ(2u, popups.size());
EXPECT_TRUE(PopupNotificationsContain(popups, "id2"));
EXPECT_TRUE(PopupNotificationsContain(popups, "id4"));
EXPECT_EQ(4u, message_center()->GetNotifications().size());
blocker.SetNotificationsEnabled(true);
popups = message_center()->GetPopupNotifications();
EXPECT_EQ(3u, popups.size());
EXPECT_TRUE(PopupNotificationsContain(popups, "id2"));
EXPECT_TRUE(PopupNotificationsContain(popups, "id3"));
EXPECT_TRUE(PopupNotificationsContain(popups, "id4"));
EXPECT_EQ(4u, message_center()->GetNotifications().size());
}
TEST_F(MessageCenterImplTest, QueueUpdatesWithCenterVisible) {
std::string id("id1");
std::string id2("id2");
NotifierId notifier_id1(NotifierId::APPLICATION, "app1");
scoped_ptr<Notification> notification(new Notification(
NOTIFICATION_TYPE_SIMPLE,
id,
UTF8ToUTF16("title"),
UTF8ToUTF16("message"),
gfx::Image() /* icon */,
base::string16() /* display_source */,
notifier_id1,
RichNotificationData(),
NULL));
message_center()->AddNotification(notification.Pass());
notification.reset(new Notification(
NOTIFICATION_TYPE_MULTIPLE,
id2,
UTF8ToUTF16("title"),
UTF8ToUTF16("message"),
gfx::Image() /* icon */,
base::string16() /* display_source */,
notifier_id1,
RichNotificationData(),
NULL));
message_center()->UpdateNotification(id, notification.Pass());
EXPECT_TRUE(message_center()->HasNotification(id2));
EXPECT_FALSE(message_center()->HasNotification(id));
message_center()->SetVisibility(VISIBILITY_MESSAGE_CENTER);
notification.reset(new Notification(
NOTIFICATION_TYPE_MULTIPLE,
id,
UTF8ToUTF16("title"),
UTF8ToUTF16("message"),
gfx::Image() /* icon */,
base::string16() /* display_source */,
notifier_id1,
RichNotificationData(),
NULL));
message_center()->UpdateNotification(id2, notification.Pass());
EXPECT_TRUE(message_center()->HasNotification(id2));
EXPECT_FALSE(message_center()->HasNotification(id));
message_center()->SetVisibility(VISIBILITY_TRANSIENT);
EXPECT_FALSE(message_center()->HasNotification(id2));
EXPECT_TRUE(message_center()->HasNotification(id));
}
} // namespace internal
} // namespace message_center