Chromium Code Reviews| OLD | NEW |
|---|---|
| (Empty) | |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | |
| 2 // Use of this source code is governed by a BSD-style license that can be | |
| 3 // found in the LICENSE file. | |
| 4 | |
| 5 #include "ash/system/web_notification/web_notification_tray.h" | |
| 6 | |
| 7 #include "ash/shell.h" | |
| 8 #include "ash/system/status_area_widget.h" | |
| 9 #include "ash/system/status_area_widget.h" | |
| 10 #include "ash/system/tray/tray_bubble_view.h" | |
| 11 #include "ash/system/tray/tray_constants.h" | |
| 12 #include "ash/system/tray/tray_views.h" | |
| 13 #include "grit/ash_strings.h" | |
| 14 #include "grit/ui_resources.h" | |
| 15 #include "grit/ui_resources_standard.h" | |
| 16 #include "ui/aura/event.h" | |
| 17 #include "ui/aura/window.h" | |
| 18 #include "ui/base/l10n/l10n_util.h" | |
| 19 #include "ui/base/models/simple_menu_model.h" | |
| 20 #include "ui/base/resource/resource_bundle.h" | |
| 21 #include "ui/views/controls/button/button.h" | |
| 22 #include "ui/views/controls/button/menu_button.h" | |
| 23 #include "ui/views/controls/button/menu_button_listener.h" | |
| 24 #include "ui/views/controls/label.h" | |
| 25 #include "ui/views/controls/menu/menu_model_adapter.h" | |
| 26 #include "ui/views/controls/menu/menu_runner.h" | |
| 27 #include "ui/views/layout/box_layout.h" | |
| 28 #include "ui/views/layout/fill_layout.h" | |
| 29 #include "ui/views/layout/grid_layout.h" | |
| 30 #include "ui/views/painter.h" | |
| 31 | |
| 32 namespace { | |
| 33 | |
| 34 const int kTrayBorder = 4; | |
| 35 const int kNotificationIconWidth = 40; | |
| 36 const int kNotificationIconHeight = 25; | |
| 37 const int kWebNotificationBubbleMinHeight = 80; | |
| 38 const int kWebNotificationBubbleMaxHeight = 400; | |
| 39 const int kWebNotificationWidth = 400; | |
| 40 const int kWebNotificationButtonWidth = 32; | |
| 41 | |
| 42 const int kTogglePermissionCommand = 0; | |
| 43 const int kToggleExtensionCommand = 1; | |
| 44 const int kShowSettingsCommand = 2; | |
| 45 | |
| 46 // The image has three icons: 1 notifiaction, 2 notifications, and 3+. | |
| 47 SkBitmap GetNotificationImage(int notification_count) { | |
| 48 SkBitmap image; | |
| 49 gfx::Image all = ui::ResourceBundle::GetSharedInstance().GetImageNamed( | |
| 50 IDR_AURA_UBER_TRAY_WEB_NOTIFICATON); | |
| 51 int image_index = notification_count - 1; | |
| 52 image_index = std::max(0, std::min(image_index, 2)); | |
| 53 SkIRect region = SkIRect::MakeXYWH( | |
| 54 0, image_index * kNotificationIconHeight, | |
| 55 kNotificationIconWidth, kNotificationIconHeight); | |
| 56 all.ToSkBitmap()->extractSubset(&image, region); | |
| 57 return image; | |
| 58 } | |
| 59 | |
| 60 } // namespace | |
| 61 | |
| 62 namespace ash { | |
| 63 | |
| 64 namespace internal { | |
| 65 | |
| 66 struct WebNotification { | |
| 67 WebNotification(const std::string& i, | |
| 68 const string16& t, | |
| 69 const string16& m, | |
| 70 const string16& s, | |
| 71 const std::string& e) | |
| 72 : id(i), | |
| 73 title(t), | |
| 74 message(m), | |
| 75 display_source(s), | |
| 76 extension_id(e) { | |
| 77 } | |
| 78 | |
| 79 std::string id; | |
| 80 string16 title; | |
| 81 string16 message; | |
| 82 string16 display_source; | |
| 83 std::string extension_id; | |
| 84 gfx::ImageSkia image; | |
| 85 }; | |
| 86 | |
| 87 // A helper class to manage the list of notifications. | |
| 88 class WebNotificationList { | |
| 89 public: | |
| 90 typedef std::list<WebNotification> Notifications; | |
| 91 | |
| 92 WebNotificationList() { | |
| 93 } | |
| 94 | |
| 95 void AddNotification(const std::string& id, | |
| 96 const string16& title, | |
| 97 const string16& message, | |
| 98 const string16& display_source, | |
| 99 const std::string& extension_id) { | |
| 100 Notifications::iterator iter = GetNotification(id); | |
| 101 if (iter != notifications_.end()) { | |
| 102 // Update existing notification. | |
| 103 iter->title = title; | |
| 104 iter->message = message; | |
| 105 iter->display_source = display_source; | |
| 106 iter->extension_id = extension_id; | |
| 107 } else { | |
| 108 notifications_.push_back( | |
| 109 WebNotification(id, title, message, display_source, extension_id)); | |
| 110 } | |
| 111 } | |
| 112 | |
| 113 void UpdateNotificationMessage(const std::string& id, | |
| 114 const string16& title, | |
| 115 const string16& message) { | |
| 116 Notifications::iterator iter = GetNotification(id); | |
| 117 if (iter == notifications_.end()) | |
| 118 return; | |
| 119 iter->title = title; | |
| 120 iter->message = message; | |
| 121 } | |
| 122 | |
| 123 bool RemoveNotification(const std::string& id) { | |
| 124 Notifications::iterator iter = GetNotification(id); | |
| 125 if (iter == notifications_.end()) | |
| 126 return false; | |
| 127 notifications_.erase(iter); | |
| 128 return true; | |
| 129 } | |
| 130 | |
| 131 void RemoveAllNotifications() { | |
| 132 notifications_.clear(); | |
| 133 } | |
| 134 | |
| 135 void RemoveNotificationsBySource(const std::string& id) { | |
| 136 Notifications::iterator source_iter = GetNotification(id); | |
| 137 if (source_iter == notifications_.end()) | |
| 138 return; | |
| 139 string16 display_source = source_iter->display_source; | |
| 140 for (Notifications::iterator loopiter = notifications_.begin(); | |
| 141 loopiter != notifications_.end(); ) { | |
| 142 Notifications::iterator curiter = loopiter++; | |
| 143 if (curiter->display_source == display_source) | |
| 144 notifications_.erase(curiter); | |
| 145 } | |
| 146 } | |
| 147 | |
| 148 void RemoveNotificationsByExtension(const std::string& id) { | |
| 149 Notifications::iterator source_iter = GetNotification(id); | |
| 150 if (source_iter == notifications_.end()) | |
| 151 return; | |
| 152 std::string extension_id = source_iter->extension_id; | |
| 153 for (Notifications::iterator loopiter = notifications_.begin(); | |
| 154 loopiter != notifications_.end(); ) { | |
| 155 Notifications::iterator curiter = loopiter++; | |
| 156 if (curiter->extension_id == extension_id) | |
| 157 notifications_.erase(curiter); | |
| 158 } | |
| 159 } | |
| 160 | |
| 161 bool SetNotificationImage(const std::string& id, | |
| 162 const gfx::ImageSkia& image) { | |
| 163 Notifications::iterator iter = GetNotification(id); | |
| 164 if (iter == notifications_.end()) | |
| 165 return false; | |
| 166 iter->image = image; | |
| 167 return true; | |
| 168 } | |
| 169 | |
| 170 const Notifications& notifications() const { return notifications_; } | |
| 171 | |
| 172 private: | |
| 173 Notifications::iterator GetNotification(const std::string& id) { | |
| 174 for (Notifications::iterator iter = notifications_.begin(); | |
| 175 iter != notifications_.end(); ++iter) { | |
| 176 if (iter->id == id) | |
| 177 return iter; | |
| 178 } | |
| 179 return notifications_.end(); | |
| 180 } | |
| 181 | |
| 182 Notifications notifications_; | |
| 183 | |
| 184 DISALLOW_COPY_AND_ASSIGN(WebNotificationList); | |
| 185 }; | |
| 186 | |
| 187 // A simple view for the text (title and message) of a notification. | |
| 188 class WebNotificationMessageView : public views::View { | |
| 189 public: | |
| 190 explicit WebNotificationMessageView(const WebNotification& notification) { | |
| 191 views::Label* title = new views::Label(notification.title); | |
| 192 title->SetHorizontalAlignment(views::Label::ALIGN_LEFT); | |
| 193 title->SetFont(title->font().DeriveFont(0, gfx::Font::BOLD)); | |
| 194 views::Label* message = new views::Label(notification.message); | |
| 195 message->SetHorizontalAlignment(views::Label::ALIGN_LEFT); | |
| 196 message->SetMultiLine(true); | |
| 197 | |
| 198 SetLayoutManager( | |
| 199 new views::BoxLayout(views::BoxLayout::kVertical, 0, 0, 1)); | |
| 200 AddChildView(title); | |
| 201 AddChildView(message); | |
| 202 } | |
| 203 | |
| 204 virtual ~WebNotificationMessageView() { | |
| 205 } | |
| 206 | |
| 207 private: | |
| 208 DISALLOW_COPY_AND_ASSIGN(WebNotificationMessageView); | |
| 209 }; | |
| 210 | |
| 211 // A dropdown menu for notifications. | |
| 212 class WebNotificationMenuModel : public ui::SimpleMenuModel, | |
| 213 public ui::SimpleMenuModel::Delegate { | |
| 214 public: | |
| 215 explicit WebNotificationMenuModel(WebNotificationTray* tray, | |
| 216 const WebNotification& notification) | |
| 217 : ALLOW_THIS_IN_INITIALIZER_LIST(ui::SimpleMenuModel(this)), | |
| 218 tray_(tray), | |
| 219 notification_(notification) { | |
| 220 // Add 'disable notifications' menu item. | |
| 221 if (!notification.extension_id.empty()) { | |
| 222 AddItem(kToggleExtensionCommand, | |
| 223 GetLabelForCommandId(kToggleExtensionCommand)); | |
| 224 } else if (!notification.display_source.empty()) { | |
| 225 AddItem(kTogglePermissionCommand, | |
| 226 GetLabelForCommandId(kTogglePermissionCommand)); | |
| 227 } | |
| 228 // Add settings menu item. | |
| 229 if (!notification.display_source.empty()) { | |
| 230 AddItem(kShowSettingsCommand, | |
| 231 GetLabelForCommandId(kShowSettingsCommand)); | |
| 232 } | |
| 233 } | |
| 234 | |
| 235 virtual ~WebNotificationMenuModel() { | |
| 236 } | |
| 237 | |
| 238 // Overridden from ui::SimpleMenuModel: | |
| 239 virtual string16 GetLabelForCommandId(int command_id) const OVERRIDE { | |
| 240 switch (command_id) { | |
| 241 case kToggleExtensionCommand: | |
| 242 return l10n_util::GetStringUTF16( | |
| 243 IDS_ASH_WEB_NOTFICATION_TRAY_EXTENSIONS_DISABLE); | |
| 244 case kTogglePermissionCommand: | |
| 245 return l10n_util::GetStringFUTF16( | |
| 246 IDS_ASH_WEB_NOTFICATION_TRAY_SITE_DISABLE, | |
| 247 notification_.display_source); | |
| 248 case kShowSettingsCommand: | |
| 249 return l10n_util::GetStringUTF16( | |
| 250 IDS_ASH_WEB_NOTFICATION_TRAY_SETTINGS); | |
| 251 default: | |
| 252 NOTREACHED(); | |
| 253 } | |
| 254 return string16(); | |
| 255 } | |
| 256 | |
| 257 // Overridden from ui::SimpleMenuModel::Delegate: | |
| 258 virtual bool IsCommandIdChecked(int command_id) const OVERRIDE { | |
| 259 return false; | |
| 260 } | |
| 261 | |
| 262 virtual bool IsCommandIdEnabled(int command_id) const OVERRIDE { | |
| 263 return true; | |
| 264 } | |
| 265 | |
| 266 virtual bool GetAcceleratorForCommandId( | |
| 267 int command_id, | |
| 268 ui::Accelerator* accelerator) OVERRIDE { | |
| 269 return false; | |
| 270 } | |
| 271 | |
| 272 virtual void ExecuteCommand(int command_id) OVERRIDE { | |
| 273 switch (command_id) { | |
| 274 case kToggleExtensionCommand: | |
| 275 tray_->DisableByExtension(notification_.id); | |
| 276 break; | |
| 277 case kTogglePermissionCommand: | |
| 278 tray_->DisableByUrl(notification_.id); | |
| 279 break; | |
| 280 case kShowSettingsCommand: | |
| 281 tray_->ShowSettings(notification_.id); | |
| 282 break; | |
| 283 default: | |
| 284 NOTREACHED(); | |
| 285 } | |
| 286 } | |
| 287 | |
| 288 private: | |
| 289 WebNotificationTray* tray_; | |
| 290 WebNotification notification_; | |
| 291 | |
| 292 DISALLOW_COPY_AND_ASSIGN(WebNotificationMenuModel); | |
| 293 }; | |
| 294 | |
| 295 // The view for a notification entry (icon + message + buttons). | |
| 296 class WebNotificationView : public views::View, | |
| 297 public views::ButtonListener, | |
| 298 public views::MenuButtonListener { | |
| 299 public: | |
| 300 WebNotificationView(WebNotificationTray* tray, | |
| 301 const WebNotification& notification) | |
| 302 : tray_(tray), | |
| 303 notification_(notification), | |
| 304 icon_(NULL), | |
| 305 menu_button_(NULL), | |
| 306 close_button_(NULL) { | |
| 307 InitView(tray, notification); | |
| 308 } | |
| 309 | |
| 310 virtual ~WebNotificationView() { | |
| 311 } | |
| 312 | |
| 313 void InitView(WebNotificationTray* tray, | |
| 314 const WebNotification& notification) { | |
| 315 set_border(views::Border::CreateSolidSidedBorder( | |
| 316 1, 0, 0, 0, kBorderLightColor)); | |
| 317 set_background(views::Background::CreateSolidBackground(kBackgroundColor)); | |
| 318 | |
| 319 icon_ = new views::ImageView; | |
| 320 icon_->SetImage(notification.image); | |
| 321 | |
| 322 WebNotificationMessageView* message_view | |
| 323 = new WebNotificationMessageView(notification); | |
| 324 | |
| 325 close_button_ = new views::ImageButton(this); | |
| 326 close_button_->SetImage( | |
| 327 views::CustomButton::BS_NORMAL, | |
| 328 ResourceBundle::GetSharedInstance().GetImageSkiaNamed( | |
| 329 IDR_AURA_WINDOW_CLOSE)); | |
| 330 | |
| 331 if (!notification.extension_id.empty() || | |
| 332 !notification.display_source.empty()) { | |
| 333 menu_button_ = new views::MenuButton(NULL, string16(), this, true); | |
| 334 menu_button_->set_border(NULL); | |
| 335 } | |
| 336 | |
| 337 views::GridLayout* layout = new views::GridLayout(this); | |
| 338 SetLayoutManager(layout); | |
| 339 | |
| 340 views::ColumnSet* columns = layout->AddColumnSet(0); | |
| 341 | |
| 342 columns->AddPaddingColumn(0, kTrayPopupPaddingHorizontal/2); | |
| 343 | |
| 344 // Notification Icon. | |
| 345 columns->AddColumn(views::GridLayout::CENTER, views::GridLayout::CENTER, | |
| 346 0, /* resize percent */ | |
| 347 views::GridLayout::FIXED, | |
| 348 kNotificationIconWidth, kNotificationIconWidth); | |
| 349 | |
| 350 columns->AddPaddingColumn(0, kTrayPopupPaddingHorizontal/2); | |
| 351 | |
| 352 // Notification message text. | |
| 353 columns->AddColumn(views::GridLayout::FILL, views::GridLayout::FILL, | |
| 354 100, /* resize percent */ | |
| 355 views::GridLayout::USE_PREF, 0, 0); | |
| 356 | |
| 357 columns->AddPaddingColumn(0, kTrayPopupPaddingHorizontal/2); | |
| 358 | |
| 359 // Close and menu buttons. | |
| 360 columns->AddColumn(views::GridLayout::CENTER, views::GridLayout::CENTER, | |
| 361 0, /* resize percent */ | |
| 362 views::GridLayout::FIXED, | |
| 363 kWebNotificationButtonWidth, | |
| 364 kWebNotificationButtonWidth); | |
| 365 | |
| 366 columns->AddPaddingColumn(0, kTrayPopupPaddingHorizontal/2); | |
| 367 | |
| 368 // Layout rows | |
| 369 layout->AddPaddingRow(0, kTrayPopupPaddingBetweenItems); | |
| 370 | |
| 371 layout->StartRow(0, 0); | |
| 372 layout->AddView(icon_, 1, 2); | |
| 373 layout->AddView(message_view, 1, 2); | |
| 374 layout->AddView(close_button_); | |
| 375 | |
| 376 layout->StartRow(0, 0); | |
| 377 if (menu_button_) { | |
| 378 layout->SkipColumns(4); | |
| 379 layout->AddView(menu_button_); | |
| 380 } | |
| 381 layout->AddPaddingRow(0, kTrayPopupPaddingBetweenItems); | |
| 382 } | |
| 383 | |
| 384 // view::Views overrodes. | |
| 385 virtual bool OnMousePressed(const views::MouseEvent& event) OVERRIDE { | |
| 386 tray_->OnClicked(notification_.id); | |
| 387 return true; | |
| 388 } | |
| 389 | |
| 390 // Overridden from ButtonListener. | |
| 391 virtual void ButtonPressed(views::Button* sender, | |
| 392 const views::Event& event) OVERRIDE { | |
| 393 if (sender == close_button_) | |
| 394 tray_->RemoveNotification(notification_.id); | |
| 395 } | |
| 396 | |
| 397 // Overridden from MenuButtonListener. | |
| 398 virtual void OnMenuButtonClicked( | |
| 399 View* source, const gfx::Point& point) OVERRIDE { | |
| 400 if (source != menu_button_) | |
| 401 return; | |
| 402 WebNotificationMenuModel menu_model(tray_, notification_); | |
| 403 views::MenuModelAdapter menu_model_adapter(&menu_model); | |
| 404 views::MenuRunner menu_runner(menu_model_adapter.CreateMenu()); | |
| 405 | |
| 406 gfx::Point screen_location; | |
| 407 views::View::ConvertPointToScreen(menu_button_, &screen_location); | |
| 408 ignore_result(menu_runner.RunMenuAt( | |
| 409 source->GetWidget()->GetTopLevelWidget(), | |
| 410 menu_button_, | |
| 411 gfx::Rect(screen_location, menu_button_->size()), | |
| 412 views::MenuItemView::TOPRIGHT, | |
| 413 views::MenuRunner::HAS_MNEMONICS)); | |
| 414 } | |
| 415 | |
| 416 private: | |
| 417 WebNotificationTray* tray_; | |
| 418 WebNotification notification_; | |
| 419 views::ImageView* icon_; | |
| 420 views::MenuButton* menu_button_; | |
| 421 views::ImageButton* close_button_; | |
| 422 | |
| 423 DISALLOW_COPY_AND_ASSIGN(WebNotificationView); | |
| 424 }; | |
| 425 | |
| 426 // The view for the buttons at the bottom of the web notification tray. | |
| 427 class WebNotificationButtonView : public TrayPopupTextButtonContainer, | |
| 428 public views::ButtonListener { | |
| 429 public: | |
| 430 explicit WebNotificationButtonView(WebNotificationTray* tray) | |
| 431 : tray_(tray), | |
| 432 settings_button_(NULL), | |
| 433 close_all_button_(NULL) { | |
| 434 set_background(views::Background::CreateBackgroundPainter( | |
| 435 true, | |
| 436 views::Painter::CreateVerticalGradient( | |
| 437 kHeaderBackgroundColorLight, | |
| 438 kHeaderBackgroundColorDark))); | |
| 439 set_border(views::Border::CreateSolidSidedBorder( | |
| 440 2, 0, 0, 0, ash::kBorderDarkColor)); | |
| 441 | |
| 442 ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); | |
| 443 settings_button_ = new TrayPopupTextButton( | |
| 444 this, rb.GetLocalizedString(IDS_ASH_WEB_NOTFICATION_TRAY_SETTINGS)); | |
| 445 AddTextButton(settings_button_); | |
| 446 | |
| 447 close_all_button_ = new TrayPopupTextButton( | |
| 448 this, rb.GetLocalizedString(IDS_ASH_WEB_NOTFICATION_TRAY_CLOSE_ALL)); | |
| 449 AddTextButton(close_all_button_); | |
| 450 } | |
| 451 | |
| 452 virtual ~WebNotificationButtonView() { | |
| 453 } | |
| 454 | |
| 455 // Overridden from ButtonListener. | |
| 456 virtual void ButtonPressed(views::Button* sender, | |
| 457 const views::Event& event) OVERRIDE { | |
| 458 if (sender == settings_button_) | |
| 459 tray_->ShowSettings(""); | |
| 460 else if (sender == close_all_button_) | |
| 461 tray_->RemoveAllNotifications(); | |
| 462 } | |
| 463 | |
| 464 private: | |
| 465 WebNotificationTray* tray_; | |
| 466 TrayPopupTextButton* settings_button_; | |
| 467 TrayPopupTextButton* close_all_button_; | |
| 468 | |
| 469 DISALLOW_COPY_AND_ASSIGN(WebNotificationButtonView); | |
| 470 }; | |
| 471 | |
| 472 } // namespace internal | |
| 473 | |
| 474 using internal::WebNotificationList; | |
| 475 using internal::WebNotificationView; | |
| 476 | |
| 477 class WebNotificationTray::BubbleContentsView : public views::View { | |
| 478 public: | |
| 479 explicit BubbleContentsView(WebNotificationTray* tray) | |
| 480 : tray_(tray) { | |
| 481 SetLayoutManager( | |
| 482 new views::BoxLayout(views::BoxLayout::kVertical, 0, 0, 1)); | |
| 483 set_background(views::Background::CreateSolidBackground(kBackgroundColor)); | |
| 484 | |
| 485 scroll_content_ = new views::View; | |
| 486 scroll_content_->SetLayoutManager( | |
| 487 new views::BoxLayout(views::BoxLayout::kVertical, 0, 0, 1)); | |
| 488 scroller_ = new internal::FixedSizedScrollView; | |
| 489 scroller_->SetContentsView(scroll_content_); | |
| 490 AddChildView(scroller_); | |
| 491 | |
| 492 button_view_ = new internal::WebNotificationButtonView(tray); | |
| 493 AddChildView(button_view_); | |
| 494 } | |
| 495 | |
| 496 void Update(const WebNotificationList::Notifications& notifications) { | |
| 497 scroll_content_->RemoveAllChildViews(true); | |
| 498 for (WebNotificationList::Notifications::const_iterator iter = | |
| 499 notifications.begin(); iter != notifications.end(); ++iter) { | |
| 500 WebNotificationView* view = new WebNotificationView(tray_, *iter); | |
| 501 scroll_content_->AddChildView(view); | |
| 502 } | |
| 503 SizeScrollContent(); | |
| 504 scroller_->Layout(); | |
| 505 Layout(); | |
| 506 PreferredSizeChanged(); | |
| 507 SchedulePaint(); | |
| 508 } | |
| 509 | |
| 510 private: | |
| 511 void SizeScrollContent() { | |
| 512 gfx::Size scroll_size = scroll_content_->GetPreferredSize(); | |
| 513 int button_height = button_view_->GetPreferredSize().height(); | |
| 514 int scroll_height = std::min( | |
| 515 std::max(scroll_size.height(), | |
| 516 kWebNotificationBubbleMinHeight - button_height), | |
| 517 kWebNotificationBubbleMaxHeight - button_height); | |
| 518 scroll_size.set_height(scroll_height); | |
| 519 scroller_->set_fixed_size(scroll_size); | |
| 520 } | |
| 521 | |
| 522 WebNotificationTray* tray_; | |
| 523 internal::FixedSizedScrollView* scroller_; | |
| 524 views::View* scroll_content_; | |
| 525 internal::WebNotificationButtonView* button_view_; | |
| 526 | |
| 527 DISALLOW_COPY_AND_ASSIGN(BubbleContentsView); | |
| 528 }; | |
| 529 | |
| 530 class WebNotificationTray::Bubble : public internal::TrayBubbleView::Host, | |
| 531 public views::Widget::Observer { | |
| 532 public: | |
| 533 explicit Bubble(WebNotificationTray* tray) | |
| 534 : tray_(tray), | |
| 535 bubble_view_(NULL), | |
| 536 bubble_widget_(NULL), | |
| 537 contents_view_(NULL) { | |
| 538 views::View* anchor = tray->tray_container(); | |
| 539 views::BubbleBorder::ArrowLocation arrow_location; | |
| 540 int arrow_offset = 0; | |
| 541 if (tray_->shelf_alignment() == SHELF_ALIGNMENT_BOTTOM) { | |
| 542 arrow_location = views::BubbleBorder::BOTTOM_RIGHT; | |
| 543 arrow_offset = anchor->GetContentsBounds().width() / 2; | |
| 544 } else if (tray_->shelf_alignment() == SHELF_ALIGNMENT_LEFT) { | |
| 545 arrow_location = views::BubbleBorder::LEFT_BOTTOM; | |
| 546 } else { | |
| 547 arrow_location = views::BubbleBorder::RIGHT_BOTTOM; | |
| 548 } | |
| 549 bubble_view_ = new internal::TrayBubbleView( | |
| 550 anchor, arrow_location, this, false, kWebNotificationWidth); | |
| 551 bubble_view_->SetMaxHeight(kWebNotificationBubbleMaxHeight); | |
| 552 | |
| 553 bubble_widget_ = views::BubbleDelegateView::CreateBubble(bubble_view_); | |
| 554 | |
| 555 bubble_view_->SetAlignment(views::BubbleBorder::ALIGN_EDGE_TO_ANCHOR_EDGE); | |
| 556 bubble_widget_->non_client_view()->frame_view()->set_background(NULL); | |
| 557 bubble_view_->SetBubbleBorder(arrow_offset); | |
| 558 | |
| 559 bubble_widget_->AddObserver(this); | |
| 560 | |
| 561 contents_view_ = new BubbleContentsView(tray); | |
| 562 bubble_view_->AddChildView(contents_view_); | |
| 563 | |
| 564 Update(); | |
| 565 bubble_view_->Show(); | |
| 566 } | |
| 567 | |
| 568 virtual ~Bubble() { | |
| 569 if (bubble_view_) | |
| 570 bubble_view_->reset_host(); | |
| 571 if (bubble_widget_) { | |
| 572 bubble_widget_->RemoveObserver(this); | |
| 573 bubble_widget_->Close(); | |
| 574 } | |
| 575 } | |
| 576 | |
| 577 void Update() { | |
| 578 contents_view_->Update(tray_->notification_list()->notifications()); | |
| 579 bubble_view_->Layout(); | |
| 580 bubble_view_->SchedulePaint(); | |
| 581 } | |
| 582 | |
| 583 views::Widget* bubble_widget() const { return bubble_widget_; } | |
| 584 | |
| 585 // Overridden from TrayBubbleView::Host. | |
| 586 virtual void BubbleViewDestroyed() OVERRIDE { | |
| 587 bubble_view_ = NULL; | |
| 588 contents_view_ = NULL; | |
| 589 } | |
| 590 | |
| 591 virtual gfx::Rect GetAnchorRect() const OVERRIDE { | |
| 592 gfx::Rect anchor_rect = tray_->tray_container()->GetScreenBounds(); | |
| 593 return anchor_rect; | |
| 594 } | |
| 595 | |
| 596 virtual void OnMouseEnteredView() OVERRIDE { | |
| 597 } | |
| 598 | |
| 599 virtual void OnMouseExitedView() OVERRIDE { | |
| 600 } | |
| 601 | |
| 602 // Overridden from views::Widget::Observer. | |
| 603 virtual void OnWidgetClosing(views::Widget* widget) OVERRIDE { | |
|
stevenjb
2012/06/13 19:05:02
Added this.
| |
| 604 CHECK_EQ(bubble_widget_, widget); | |
| 605 bubble_widget_ = NULL; | |
| 606 tray_->HideBubble(); // Will destroy |this|. | |
| 607 } | |
| 608 | |
| 609 private: | |
| 610 WebNotificationTray* tray_; | |
| 611 internal::TrayBubbleView* bubble_view_; | |
| 612 views::Widget* bubble_widget_; | |
| 613 BubbleContentsView* contents_view_; | |
| 614 | |
| 615 DISALLOW_COPY_AND_ASSIGN(Bubble); | |
| 616 }; | |
| 617 | |
| 618 WebNotificationTray::WebNotificationTray( | |
| 619 internal::StatusAreaWidget* status_area_widget) | |
| 620 : status_area_widget_(status_area_widget), | |
| 621 notification_list_(new WebNotificationList()), | |
| 622 tray_container_(NULL), | |
| 623 icon_(NULL), | |
| 624 delegate_(NULL) { | |
| 625 tray_container_ = new views::View; | |
| 626 tray_container_->set_border(views::Border::CreateEmptyBorder( | |
| 627 kTrayBorder, kTrayBorder, kTrayBorder, kTrayBorder)); | |
| 628 SetShelfAlignment(shelf_alignment()); | |
| 629 | |
| 630 icon_ = new views::ImageView; | |
| 631 tray_container_->AddChildView(icon_); | |
| 632 UpdateIcon(); // Hides the tray initially. | |
| 633 | |
| 634 SetContents(tray_container_); | |
| 635 | |
| 636 Shell::GetInstance()->AddEnvEventFilter(this); | |
| 637 } | |
| 638 | |
| 639 WebNotificationTray::~WebNotificationTray() { | |
| 640 Shell::GetInstance()->RemoveEnvEventFilter(this); | |
| 641 } | |
| 642 | |
| 643 void WebNotificationTray::SetDelegate(Delegate* delegate) { | |
| 644 DCHECK(!delegate_); | |
| 645 delegate_ = delegate; | |
| 646 } | |
| 647 | |
| 648 void WebNotificationTray::AddNotification(const std::string& id, | |
| 649 const string16& title, | |
| 650 const string16& message, | |
| 651 const string16& display_source, | |
| 652 const std::string& extension_id) { | |
| 653 notification_list_->AddNotification( | |
| 654 id, title, message, display_source, extension_id); | |
| 655 UpdateIcon(); | |
| 656 if (bubble()) { | |
| 657 bubble_->Update(); | |
| 658 } else { | |
| 659 status_area_widget_->ShowWebNotificationBubble( | |
| 660 internal::StatusAreaWidget::NON_USER_ACTION); | |
| 661 } | |
| 662 } | |
| 663 | |
| 664 void WebNotificationTray::UpdateNotification(const std::string& id, | |
| 665 const string16& title, | |
| 666 const string16& message) { | |
| 667 notification_list_->UpdateNotificationMessage(id, title, message); | |
| 668 if (bubble()) | |
| 669 bubble_->Update(); | |
| 670 } | |
| 671 | |
| 672 void WebNotificationTray::RemoveNotification(const std::string& id) { | |
| 673 if (!notification_list_->RemoveNotification(id)) | |
| 674 return; | |
| 675 if (delegate_) | |
| 676 delegate_->NotificationRemoved(id); | |
| 677 UpdateBubbleAndIcon(); | |
| 678 } | |
| 679 | |
| 680 void WebNotificationTray::RemoveAllNotifications() { | |
| 681 const WebNotificationList::Notifications& notifications = | |
| 682 notification_list_->notifications(); | |
| 683 if (delegate_) { | |
| 684 for (WebNotificationList::Notifications::const_iterator loopiter = | |
| 685 notifications.begin(); | |
| 686 loopiter != notifications.end(); ) { | |
| 687 WebNotificationList::Notifications::const_iterator curiter = loopiter++; | |
| 688 std::string notification_id = curiter->id; | |
| 689 // May call RemoveNotification and erase curiter. | |
| 690 delegate_->NotificationRemoved(notification_id); | |
| 691 } | |
| 692 } | |
| 693 notification_list_->RemoveAllNotifications(); | |
| 694 UpdateBubbleAndIcon(); | |
| 695 } | |
| 696 | |
| 697 void WebNotificationTray::SetNotificationImage(const std::string& id, | |
| 698 const gfx::ImageSkia& image) { | |
| 699 if (!notification_list_->SetNotificationImage(id, image)) | |
| 700 return; | |
| 701 if (bubble()) | |
| 702 bubble_->Update(); | |
| 703 } | |
| 704 | |
| 705 void WebNotificationTray::DisableByExtension(const std::string& id) { | |
| 706 // When we disable notifications, we remove any existing matching | |
| 707 // notifications to avoid adding complicated UI to re-enable the source. | |
| 708 notification_list_->RemoveNotificationsByExtension(id); | |
| 709 UpdateBubbleAndIcon(); | |
| 710 if (delegate_) | |
| 711 delegate_->DisableExtension(id); | |
| 712 } | |
| 713 | |
| 714 void WebNotificationTray::DisableByUrl(const std::string& id) { | |
| 715 // See comment for DisableByExtension. | |
| 716 notification_list_->RemoveNotificationsBySource(id); | |
| 717 UpdateBubbleAndIcon(); | |
| 718 if (delegate_) | |
| 719 delegate_->DisableNotificationsFromSource(id); | |
| 720 } | |
| 721 | |
| 722 void WebNotificationTray::ShowBubble() { | |
| 723 if (bubble()) | |
| 724 return; | |
| 725 bubble_.reset(new Bubble(this)); | |
| 726 } | |
| 727 | |
| 728 void WebNotificationTray::HideBubble() { | |
| 729 bubble_.reset(); | |
| 730 } | |
| 731 | |
| 732 void WebNotificationTray::ShowSettings(const std::string& id) { | |
| 733 if (delegate_) | |
| 734 delegate_->ShowSettings(id); | |
| 735 } | |
| 736 | |
| 737 void WebNotificationTray::OnClicked(const std::string& id) { | |
| 738 if (delegate_) | |
| 739 delegate_->OnClicked(id); | |
| 740 } | |
| 741 | |
| 742 bool WebNotificationTray::PreHandleKeyEvent(aura::Window* target, | |
| 743 aura::KeyEvent* event) { | |
| 744 return false; | |
| 745 } | |
| 746 | |
| 747 bool WebNotificationTray::PreHandleMouseEvent(aura::Window* target, | |
| 748 aura::MouseEvent* event) { | |
| 749 if (event->type() == ui::ET_MOUSE_PRESSED) | |
| 750 return ProcessLocatedEvent(*event); | |
| 751 return false; | |
| 752 } | |
| 753 | |
| 754 ui::TouchStatus WebNotificationTray::PreHandleTouchEvent( | |
| 755 aura::Window* target, | |
| 756 aura::TouchEvent* event) { | |
| 757 if (event->type() != ui::ET_TOUCH_PRESSED) | |
| 758 return ui::TOUCH_STATUS_UNKNOWN; | |
| 759 if (ProcessLocatedEvent(*event)) | |
| 760 return ui::TOUCH_STATUS_END; | |
| 761 return ui::TOUCH_STATUS_UNKNOWN; | |
| 762 } | |
| 763 | |
| 764 ui::GestureStatus WebNotificationTray::PreHandleGestureEvent( | |
| 765 aura::Window* target, | |
| 766 aura::GestureEvent* event) { | |
| 767 return ui::GESTURE_STATUS_UNKNOWN; | |
| 768 } | |
| 769 | |
| 770 void WebNotificationTray::SetShelfAlignment(ShelfAlignment alignment) { | |
| 771 internal::TrayBackgroundView::SetShelfAlignment(alignment); | |
| 772 tray_container_->SetLayoutManager(new views::BoxLayout( | |
| 773 alignment == SHELF_ALIGNMENT_BOTTOM ? | |
| 774 views::BoxLayout::kHorizontal : views::BoxLayout::kVertical, | |
| 775 0, 0, 0)); | |
| 776 } | |
| 777 | |
| 778 bool WebNotificationTray::PerformAction(const views::Event& event) { | |
| 779 if (bubble()) { | |
| 780 status_area_widget_->HideWebNotificationBubble(); | |
| 781 } else { | |
| 782 status_area_widget_->ShowWebNotificationBubble( | |
| 783 internal::StatusAreaWidget::USER_ACTION); | |
| 784 } | |
| 785 return true; | |
| 786 } | |
| 787 | |
| 788 int WebNotificationTray::GetNotificationCount() const { | |
| 789 return notification_list()->notifications().size(); | |
| 790 } | |
| 791 | |
| 792 void WebNotificationTray::UpdateIcon() { | |
| 793 int count = GetNotificationCount(); | |
| 794 if (count == 0) { | |
| 795 SetVisible(false); | |
| 796 } else { | |
| 797 icon_->SetImage(gfx::ImageSkia(GetNotificationImage(count))); | |
| 798 SetVisible(true); | |
| 799 } | |
| 800 PreferredSizeChanged(); | |
| 801 } | |
| 802 | |
| 803 void WebNotificationTray::UpdateBubbleAndIcon() { | |
| 804 UpdateIcon(); | |
| 805 if (!bubble()) | |
| 806 return; | |
| 807 if (GetNotificationCount() == 0) | |
| 808 status_area_widget_->HideWebNotificationBubble(); | |
| 809 else | |
| 810 bubble_->Update(); | |
| 811 } | |
| 812 | |
| 813 bool WebNotificationTray::ProcessLocatedEvent(const aura::LocatedEvent& event) { | |
| 814 if (!bubble()) | |
| 815 return false; | |
| 816 gfx::Rect bounds = | |
| 817 bubble_->bubble_widget()->GetNativeWindow()->GetBoundsInRootWindow(); | |
| 818 if (bounds.Contains(event.root_location())) | |
| 819 return false; | |
| 820 status_area_widget_->HideWebNotificationBubble(); | |
| 821 // If the event occurred in the tray widget, don't process the click. | |
| 822 bounds = GetWidget()->GetNativeWindow()->GetBoundsInRootWindow(); | |
| 823 if (bounds.Contains(event.root_location())) | |
| 824 return true; | |
| 825 return false; | |
| 826 } | |
| 827 | |
| 828 } // namespace ash | |
| OLD | NEW |