summaryrefslogtreecommitdiff
path: root/chromium/ash/system/web_notification/web_notification_tray.cc
diff options
context:
space:
mode:
Diffstat (limited to 'chromium/ash/system/web_notification/web_notification_tray.cc')
-rw-r--r--chromium/ash/system/web_notification/web_notification_tray.cc624
1 files changed, 624 insertions, 0 deletions
diff --git a/chromium/ash/system/web_notification/web_notification_tray.cc b/chromium/ash/system/web_notification/web_notification_tray.cc
new file mode 100644
index 00000000000..b0b75d87dfc
--- /dev/null
+++ b/chromium/ash/system/web_notification/web_notification_tray.cc
@@ -0,0 +1,624 @@
+// Copyright (c) 2012 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 "ash/system/web_notification/web_notification_tray.h"
+
+#include "ash/ash_switches.h"
+#include "ash/root_window_controller.h"
+#include "ash/shelf/shelf_layout_manager.h"
+#include "ash/shelf/shelf_layout_manager_observer.h"
+#include "ash/shelf/shelf_widget.h"
+#include "ash/shell.h"
+#include "ash/shell_window_ids.h"
+#include "ash/system/status_area_widget.h"
+#include "ash/system/tray/system_tray.h"
+#include "ash/system/tray/tray_background_view.h"
+#include "ash/system/tray/tray_bubble_wrapper.h"
+#include "ash/system/tray/tray_constants.h"
+#include "ash/system/tray/tray_utils.h"
+#include "base/auto_reset.h"
+#include "base/i18n/number_formatting.h"
+#include "base/i18n/rtl.h"
+#include "base/strings/utf_string_conversions.h"
+#include "grit/ash_strings.h"
+#include "grit/ui_strings.h"
+#include "ui/aura/root_window.h"
+#include "ui/aura/window.h"
+#include "ui/base/l10n/l10n_util.h"
+#include "ui/gfx/screen.h"
+#include "ui/message_center/message_center_style.h"
+#include "ui/message_center/message_center_tray_delegate.h"
+#include "ui/message_center/message_center_util.h"
+#include "ui/message_center/views/message_bubble_base.h"
+#include "ui/message_center/views/message_center_bubble.h"
+#include "ui/message_center/views/message_popup_collection.h"
+#include "ui/views/bubble/tray_bubble_view.h"
+#include "ui/views/controls/button/custom_button.h"
+#include "ui/views/controls/image_view.h"
+#include "ui/views/controls/label.h"
+#include "ui/views/controls/menu/menu_runner.h"
+#include "ui/views/layout/fill_layout.h"
+
+#if defined(OS_CHROMEOS)
+
+namespace message_center {
+
+MessageCenterTrayDelegate* CreateMessageCenterTray() {
+ // On Windows+Ash the Tray will not be hosted in ash::Shell.
+ NOTREACHED();
+ return NULL;
+}
+
+} // namespace message_center
+
+#endif // defined(OS_CHROMEOS)
+
+namespace ash {
+namespace {
+
+// Menu commands
+const int kToggleQuietMode = 0;
+const int kEnableQuietModeDay = 2;
+
+}
+
+namespace internal {
+namespace {
+
+const SkColor kWebNotificationColorNoUnread = SkColorSetA(SK_ColorWHITE, 128);
+const SkColor kWebNotificationColorWithUnread = SK_ColorWHITE;
+
+}
+
+// Observes the change of work area (including temporary change by auto-hide)
+// and notifies MessagePopupCollection.
+class WorkAreaObserver : public ShelfLayoutManagerObserver,
+ public ShellObserver {
+ public:
+ WorkAreaObserver();
+ virtual ~WorkAreaObserver();
+
+ void SetSystemTrayHeight(int height);
+
+ // Starts observing |shelf| and shell and sends the change to |collection|.
+ void StartObserving(message_center::MessagePopupCollection* collection,
+ aura::Window* root_window);
+
+ // Stops the observing session.
+ void StopObserving();
+
+ // Overridden from ShellObserver:
+ virtual void OnDisplayWorkAreaInsetsChanged() OVERRIDE;
+
+ // Overridden from ShelfLayoutManagerObserver:
+ virtual void OnAutoHideStateChanged(ShelfAutoHideState new_state) OVERRIDE;
+
+ private:
+ // Updates |shelf_| from |root_window_|.
+ void UpdateShelf();
+
+ message_center::MessagePopupCollection* collection_;
+ aura::Window* root_window_;
+ ShelfLayoutManager* shelf_;
+ int system_tray_height_;
+
+ DISALLOW_COPY_AND_ASSIGN(WorkAreaObserver);
+};
+
+WorkAreaObserver::WorkAreaObserver()
+ : collection_(NULL),
+ root_window_(NULL),
+ shelf_(NULL),
+ system_tray_height_(0) {
+}
+
+WorkAreaObserver::~WorkAreaObserver() {
+ StopObserving();
+}
+
+void WorkAreaObserver::SetSystemTrayHeight(int height) {
+ system_tray_height_ = height;
+
+ // If the shelf is shown during auto-hide state, the distance from the edge
+ // should be reduced by the height of shelf's shown height.
+ if (shelf_ && shelf_->visibility_state() == SHELF_AUTO_HIDE &&
+ shelf_->auto_hide_state() == SHELF_AUTO_HIDE_SHOWN) {
+ system_tray_height_ -= ShelfLayoutManager::GetPreferredShelfSize() -
+ ShelfLayoutManager::kAutoHideSize;
+ }
+
+ if (system_tray_height_ > 0 && ash::switches::UseAlternateShelfLayout())
+ system_tray_height_ += message_center::kMarginBetweenItems;
+
+ if (!shelf_)
+ return;
+
+ OnAutoHideStateChanged(shelf_->auto_hide_state());
+}
+
+void WorkAreaObserver::StartObserving(
+ message_center::MessagePopupCollection* collection,
+ aura::Window* root_window) {
+ DCHECK(collection);
+ collection_ = collection;
+ root_window_ = root_window;
+ UpdateShelf();
+ Shell::GetInstance()->AddShellObserver(this);
+ if (system_tray_height_ > 0)
+ OnAutoHideStateChanged(shelf_->auto_hide_state());
+}
+
+void WorkAreaObserver::StopObserving() {
+ Shell::GetInstance()->RemoveShellObserver(this);
+ if (shelf_)
+ shelf_->RemoveObserver(this);
+ collection_ = NULL;
+ shelf_ = NULL;
+}
+
+void WorkAreaObserver::OnDisplayWorkAreaInsetsChanged() {
+ UpdateShelf();
+
+ collection_->OnDisplayBoundsChanged(
+ Shell::GetScreen()->GetDisplayNearestWindow(
+ shelf_->shelf_widget()->GetNativeView()));
+}
+
+void WorkAreaObserver::OnAutoHideStateChanged(ShelfAutoHideState new_state) {
+ gfx::Display display = Shell::GetScreen()->GetDisplayNearestWindow(
+ shelf_->shelf_widget()->GetNativeView());
+ gfx::Rect work_area = display.work_area();
+ int width = 0;
+ if ((shelf_->visibility_state() == SHELF_AUTO_HIDE) &&
+ new_state == SHELF_AUTO_HIDE_SHOWN) {
+ // Since the work_area is already reduced by kAutoHideSize, the inset width
+ // should be just the difference.
+ width = ShelfLayoutManager::GetPreferredShelfSize() -
+ ShelfLayoutManager::kAutoHideSize;
+ }
+ work_area.Inset(shelf_->SelectValueForShelfAlignment(
+ gfx::Insets(0, 0, width, 0),
+ gfx::Insets(0, width, 0, 0),
+ gfx::Insets(0, 0, 0, width),
+ gfx::Insets(width, 0, 0, 0)));
+ if (system_tray_height_ > 0) {
+ work_area.set_height(
+ std::max(0, work_area.height() - system_tray_height_));
+ if (shelf_->GetAlignment() == SHELF_ALIGNMENT_TOP)
+ work_area.set_y(work_area.y() + system_tray_height_);
+ }
+ collection_->SetDisplayInfo(work_area, display.bounds());
+}
+
+void WorkAreaObserver::UpdateShelf() {
+ if (shelf_)
+ return;
+
+ shelf_ = ShelfLayoutManager::ForLauncher(root_window_);
+ if (shelf_)
+ shelf_->AddObserver(this);
+}
+
+// Class to initialize and manage the WebNotificationBubble and
+// TrayBubbleWrapper instances for a bubble.
+class WebNotificationBubbleWrapper {
+ public:
+ // Takes ownership of |bubble| and creates |bubble_wrapper_|.
+ WebNotificationBubbleWrapper(WebNotificationTray* tray,
+ message_center::MessageBubbleBase* bubble) {
+ bubble_.reset(bubble);
+ views::TrayBubbleView::AnchorAlignment anchor_alignment =
+ tray->GetAnchorAlignment();
+ views::TrayBubbleView::InitParams init_params =
+ bubble->GetInitParams(anchor_alignment);
+ views::View* anchor = tray->tray_container();
+ if (anchor_alignment == views::TrayBubbleView::ANCHOR_ALIGNMENT_BOTTOM) {
+ gfx::Point bounds(anchor->width() / 2, 0);
+ views::View::ConvertPointToWidget(anchor, &bounds);
+ init_params.arrow_offset = bounds.x();
+ }
+ views::TrayBubbleView* bubble_view = views::TrayBubbleView::Create(
+ tray->GetBubbleWindowContainer(), anchor, tray, &init_params);
+ if (ash::switches::UseAlternateShelfLayout())
+ bubble_view->SetArrowPaintType(views::BubbleBorder::PAINT_NONE);
+ bubble_wrapper_.reset(new TrayBubbleWrapper(tray, bubble_view));
+ bubble->InitializeContents(bubble_view);
+ }
+
+ message_center::MessageBubbleBase* bubble() const { return bubble_.get(); }
+
+ // Convenience accessors.
+ views::TrayBubbleView* bubble_view() const { return bubble_->bubble_view(); }
+
+ private:
+ scoped_ptr<message_center::MessageBubbleBase> bubble_;
+ scoped_ptr<internal::TrayBubbleWrapper> bubble_wrapper_;
+
+ DISALLOW_COPY_AND_ASSIGN(WebNotificationBubbleWrapper);
+};
+
+class WebNotificationButton : public views::CustomButton {
+ public:
+ WebNotificationButton(views::ButtonListener* listener)
+ : views::CustomButton(listener),
+ is_bubble_visible_(false),
+ unread_count_(0) {
+ SetLayoutManager(new views::FillLayout);
+ unread_label_ = new views::Label();
+ SetupLabelForTray(unread_label_);
+ AddChildView(unread_label_);
+ }
+
+ void SetBubbleVisible(bool visible) {
+ if (visible == is_bubble_visible_)
+ return;
+
+ is_bubble_visible_ = visible;
+ UpdateIconVisibility();
+ }
+
+ void SetUnreadCount(int unread_count) {
+ // base::FormatNumber doesn't convert to arabic numeric characters.
+ // TODO(mukai): use ICU to support conversion for such locales.
+ unread_count_ = unread_count;
+ // TODO(mukai): move NINE_PLUS message to ui_strings, it doesn't need to be
+ // in ash_strings.
+ unread_label_->SetText((unread_count > 9) ?
+ l10n_util::GetStringUTF16(IDS_ASH_NOTIFICATION_UNREAD_COUNT_NINE_PLUS) :
+ base::FormatNumber(unread_count));
+ UpdateIconVisibility();
+ }
+
+ protected:
+ // Overridden from views::ImageButton:
+ virtual gfx::Size GetPreferredSize() OVERRIDE {
+ const int notification_item_size = GetShelfItemHeight();
+ return gfx::Size(notification_item_size, notification_item_size);
+ }
+
+ virtual int GetHeightForWidth(int width) OVERRIDE {
+ return GetPreferredSize().height();
+ }
+
+ private:
+ void UpdateIconVisibility() {
+ unread_label_->SetEnabledColor(
+ (!is_bubble_visible_ && unread_count_ > 0) ?
+ kWebNotificationColorWithUnread : kWebNotificationColorNoUnread);
+ SchedulePaint();
+ }
+
+ bool is_bubble_visible_;
+ int unread_count_;
+
+ views::Label* unread_label_;
+
+ DISALLOW_COPY_AND_ASSIGN(WebNotificationButton);
+};
+
+} // namespace internal
+
+WebNotificationTray::WebNotificationTray(
+ internal::StatusAreaWidget* status_area_widget)
+ : TrayBackgroundView(status_area_widget),
+ button_(NULL),
+ show_message_center_on_unlock_(false),
+ should_update_tray_content_(false),
+ should_block_shelf_auto_hide_(false) {
+ button_ = new internal::WebNotificationButton(this);
+ button_->set_triggerable_event_flags(
+ ui::EF_LEFT_MOUSE_BUTTON | ui::EF_RIGHT_MOUSE_BUTTON);
+ tray_container()->AddChildView(button_);
+ SetContentsBackground();
+ tray_container()->set_border(NULL);
+ SetVisible(false);
+ message_center_tray_.reset(new message_center::MessageCenterTray(
+ this,
+ message_center::MessageCenter::Get()));
+ work_area_observer_.reset(new internal::WorkAreaObserver());
+ OnMessageCenterTrayChanged();
+}
+
+WebNotificationTray::~WebNotificationTray() {
+ // Release any child views that might have back pointers before ~View().
+ message_center_bubble_.reset();
+ popup_collection_.reset();
+ work_area_observer_.reset();
+}
+
+// Public methods.
+
+bool WebNotificationTray::ShowMessageCenterInternal(bool show_settings) {
+ if (!ShouldShowMessageCenter())
+ return false;
+
+ should_block_shelf_auto_hide_ = true;
+ message_center::MessageCenterBubble* message_center_bubble =
+ new message_center::MessageCenterBubble(
+ message_center(),
+ message_center_tray_.get(),
+ ash::switches::UseAlternateShelfLayout());
+
+ int max_height = 0;
+ aura::Window* status_area_window = status_area_widget()->GetNativeView();
+ switch (GetShelfLayoutManager()->GetAlignment()) {
+ case SHELF_ALIGNMENT_BOTTOM: {
+ gfx::Rect shelf_bounds = GetShelfLayoutManager()->GetIdealBounds();
+ max_height = shelf_bounds.y();
+ break;
+ }
+ case SHELF_ALIGNMENT_TOP: {
+ aura::Window* root = status_area_window->GetRootWindow();
+ max_height =
+ root->bounds().height() - status_area_window->bounds().height();
+ break;
+ }
+ case SHELF_ALIGNMENT_LEFT:
+ case SHELF_ALIGNMENT_RIGHT: {
+ // Assume that the bottom line of the status area widget and the bubble
+ // are aligned.
+ max_height = status_area_window->GetBoundsInRootWindow().bottom();
+ break;
+ }
+ default:
+ NOTREACHED();
+ }
+
+ message_center_bubble->SetMaxHeight(std::max(0,
+ max_height - GetTraySpacing()));
+ if (show_settings)
+ message_center_bubble->SetSettingsVisible();
+ message_center_bubble_.reset(
+ new internal::WebNotificationBubbleWrapper(this, message_center_bubble));
+
+ status_area_widget()->SetHideSystemNotifications(true);
+ GetShelfLayoutManager()->UpdateAutoHideState();
+ button_->SetBubbleVisible(true);
+ SetDrawBackgroundAsActive(true);
+ return true;
+}
+
+bool WebNotificationTray::ShowMessageCenter() {
+ return ShowMessageCenterInternal(false /* show_settings */);
+}
+
+void WebNotificationTray::HideMessageCenter() {
+ if (!message_center_bubble())
+ return;
+ SetDrawBackgroundAsActive(false);
+ message_center_bubble_.reset();
+ should_block_shelf_auto_hide_ = false;
+ show_message_center_on_unlock_ = false;
+ status_area_widget()->SetHideSystemNotifications(false);
+ GetShelfLayoutManager()->UpdateAutoHideState();
+ button_->SetBubbleVisible(false);
+}
+
+void WebNotificationTray::SetSystemTrayHeight(int height) {
+ work_area_observer_->SetSystemTrayHeight(height);
+}
+
+bool WebNotificationTray::ShowPopups() {
+ if (message_center_bubble())
+ return false;
+
+ popup_collection_.reset(new message_center::MessagePopupCollection(
+ ash::Shell::GetContainer(
+ GetWidget()->GetNativeView()->GetRootWindow(),
+ internal::kShellWindowId_StatusContainer),
+ message_center(),
+ message_center_tray_.get(),
+ ash::switches::UseAlternateShelfLayout()));
+ work_area_observer_->StartObserving(
+ popup_collection_.get(), GetWidget()->GetNativeView()->GetRootWindow());
+ return true;
+}
+
+void WebNotificationTray::HidePopups() {
+ DCHECK(popup_collection_.get());
+
+ popup_collection_->MarkAllPopupsShown();
+ popup_collection_.reset();
+ work_area_observer_->StopObserving();
+}
+
+// Private methods.
+
+bool WebNotificationTray::ShouldShowMessageCenter() {
+ return status_area_widget()->login_status() != user::LOGGED_IN_LOCKED &&
+ !(status_area_widget()->system_tray() &&
+ status_area_widget()->system_tray()->HasNotificationBubble());
+}
+
+bool WebNotificationTray::ShouldBlockLauncherAutoHide() const {
+ return should_block_shelf_auto_hide_;
+}
+
+bool WebNotificationTray::IsMessageCenterBubbleVisible() const {
+ return (message_center_bubble() &&
+ message_center_bubble()->bubble()->IsVisible());
+}
+
+bool WebNotificationTray::IsMouseInNotificationBubble() const {
+ return false;
+}
+
+void WebNotificationTray::ShowMessageCenterBubble() {
+ if (!IsMessageCenterBubbleVisible())
+ message_center_tray_->ShowMessageCenterBubble();
+}
+
+void WebNotificationTray::UpdateAfterLoginStatusChange(
+ user::LoginStatus login_status) {
+ OnMessageCenterTrayChanged();
+}
+
+void WebNotificationTray::SetShelfAlignment(ShelfAlignment alignment) {
+ if (alignment == shelf_alignment())
+ return;
+ internal::TrayBackgroundView::SetShelfAlignment(alignment);
+ tray_container()->set_border(NULL);
+ // Destroy any existing bubble so that it will be rebuilt correctly.
+ message_center_tray_->HideMessageCenterBubble();
+ message_center_tray_->HidePopupBubble();
+}
+
+void WebNotificationTray::AnchorUpdated() {
+ if (message_center_bubble()) {
+ message_center_bubble()->bubble_view()->UpdateBubble();
+ UpdateBubbleViewArrow(message_center_bubble()->bubble_view());
+ }
+}
+
+base::string16 WebNotificationTray::GetAccessibleNameForTray() {
+ return l10n_util::GetStringUTF16(
+ IDS_MESSAGE_CENTER_ACCESSIBLE_NAME);
+}
+
+void WebNotificationTray::HideBubbleWithView(
+ const views::TrayBubbleView* bubble_view) {
+ if (message_center_bubble() &&
+ bubble_view == message_center_bubble()->bubble_view()) {
+ message_center_tray_->HideMessageCenterBubble();
+ } else if (popup_collection_.get()) {
+ message_center_tray_->HidePopupBubble();
+ }
+}
+
+bool WebNotificationTray::PerformAction(const ui::Event& event) {
+ if (message_center_bubble())
+ message_center_tray_->HideMessageCenterBubble();
+ else
+ message_center_tray_->ShowMessageCenterBubble();
+ return true;
+}
+
+void WebNotificationTray::BubbleViewDestroyed() {
+ if (message_center_bubble())
+ message_center_bubble()->bubble()->BubbleViewDestroyed();
+}
+
+void WebNotificationTray::OnMouseEnteredView() {}
+
+void WebNotificationTray::OnMouseExitedView() {}
+
+base::string16 WebNotificationTray::GetAccessibleNameForBubble() {
+ return GetAccessibleNameForTray();
+}
+
+gfx::Rect WebNotificationTray::GetAnchorRect(
+ views::Widget* anchor_widget,
+ views::TrayBubbleView::AnchorType anchor_type,
+ views::TrayBubbleView::AnchorAlignment anchor_alignment) {
+ return GetBubbleAnchorRect(anchor_widget, anchor_type, anchor_alignment);
+}
+
+void WebNotificationTray::HideBubble(const views::TrayBubbleView* bubble_view) {
+ HideBubbleWithView(bubble_view);
+}
+
+bool WebNotificationTray::ShowNotifierSettings() {
+ if (message_center_bubble()) {
+ static_cast<message_center::MessageCenterBubble*>(
+ message_center_bubble()->bubble())->SetSettingsVisible();
+ return true;
+ }
+ return ShowMessageCenterInternal(true /* show_settings */);
+}
+
+message_center::MessageCenterTray* WebNotificationTray::GetMessageCenterTray() {
+ return message_center_tray_.get();
+}
+
+bool WebNotificationTray::IsCommandIdChecked(int command_id) const {
+ if (command_id != kToggleQuietMode)
+ return false;
+ return message_center()->IsQuietMode();
+}
+
+bool WebNotificationTray::IsCommandIdEnabled(int command_id) const {
+ return true;
+}
+
+bool WebNotificationTray::GetAcceleratorForCommandId(
+ int command_id,
+ ui::Accelerator* accelerator) {
+ return false;
+}
+
+void WebNotificationTray::ExecuteCommand(int command_id, int event_flags) {
+ if (command_id == kToggleQuietMode) {
+ bool in_quiet_mode = message_center()->IsQuietMode();
+ message_center()->SetQuietMode(!in_quiet_mode);
+ return;
+ }
+ base::TimeDelta expires_in = command_id == kEnableQuietModeDay ?
+ base::TimeDelta::FromDays(1):
+ base::TimeDelta::FromHours(1);
+ message_center()->EnterQuietModeWithExpire(expires_in);
+}
+
+void WebNotificationTray::ButtonPressed(views::Button* sender,
+ const ui::Event& event) {
+ DCHECK_EQ(button_, sender);
+ PerformAction(event);
+}
+
+void WebNotificationTray::OnMessageCenterTrayChanged() {
+ // Do not update the tray contents directly. Multiple change events can happen
+ // consecutively, and calling Update in the middle of those events will show
+ // intermediate unread counts for a moment.
+ should_update_tray_content_ = true;
+ base::MessageLoop::current()->PostTask(
+ FROM_HERE,
+ base::Bind(&WebNotificationTray::UpdateTrayContent, AsWeakPtr()));
+}
+
+void WebNotificationTray::UpdateTrayContent() {
+ if (!should_update_tray_content_)
+ return;
+ should_update_tray_content_ = false;
+
+ message_center::MessageCenter* message_center =
+ message_center_tray_->message_center();
+ button_->SetUnreadCount(message_center->UnreadNotificationCount());
+ if (IsMessageCenterBubbleVisible())
+ button_->SetState(views::CustomButton::STATE_PRESSED);
+ else
+ button_->SetState(views::CustomButton::STATE_NORMAL);
+ SetVisible((status_area_widget()->login_status() != user::LOGGED_IN_NONE) &&
+ (status_area_widget()->login_status() != user::LOGGED_IN_LOCKED) &&
+ (message_center->NotificationCount() > 0));
+ Layout();
+ SchedulePaint();
+}
+
+bool WebNotificationTray::ClickedOutsideBubble() {
+ // Only hide the message center
+ if (!message_center_bubble())
+ return false;
+
+ message_center_tray_->HideMessageCenterBubble();
+ return true;
+}
+
+message_center::MessageCenter* WebNotificationTray::message_center() const {
+ return message_center_tray_->message_center();
+}
+
+// Methods for testing
+
+bool WebNotificationTray::IsPopupVisible() const {
+ return message_center_tray_->popups_visible();
+}
+
+message_center::MessageCenterBubble*
+WebNotificationTray::GetMessageCenterBubbleForTest() {
+ if (!message_center_bubble())
+ return NULL;
+ return static_cast<message_center::MessageCenterBubble*>(
+ message_center_bubble()->bubble());
+}
+
+} // namespace ash