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/tray/system_tray.h" | |
| 9 #include "ash/system/tray/system_tray_bubble_view.h" | |
| 10 #include "ash/system/tray/tray_constants.h" | |
| 11 #include "ash/system/tray/tray_views.h" | |
| 12 #include "grit/ash_strings.h" | |
| 13 #include "grit/ui_resources.h" | |
| 14 #include "grit/ui_resources_standard.h" | |
| 15 #include "ui/aura/event.h" | |
| 16 #include "ui/aura/window.h" | |
| 17 #include "ui/base/l10n/l10n_util.h" | |
| 18 #include "ui/base/models/simple_menu_model.h" | |
| 19 #include "ui/base/resource/resource_bundle.h" | |
| 20 #include "ui/views/controls/button/button.h" | |
| 21 #include "ui/views/controls/button/menu_button.h" | |
| 22 #include "ui/views/controls/button/menu_button_listener.h" | |
| 23 #include "ui/views/controls/label.h" | |
| 24 #include "ui/views/controls/menu/menu_model_adapter.h" | |
| 25 #include "ui/views/controls/menu/menu_runner.h" | |
| 26 #include "ui/views/layout/box_layout.h" | |
| 27 #include "ui/views/layout/fill_layout.h" | |
| 28 #include "ui/views/layout/grid_layout.h" | |
| 29 | |
| 30 namespace { | |
| 31 | |
| 32 const int kIconBorder = 4; | |
| 33 const int kImageWidth = 40; | |
| 34 const int kImageHeight = 25; | |
| 35 const int kWebNotificationBubbleMinHeight = 50; | |
| 36 const int kWebNotificationBubbleMaxHeight = 400; | |
| 37 const int kWebNotificationWidth = 400; | |
| 38 const int kWebNotificationButtonWidth = 32; | |
| 39 | |
| 40 // The image has three icons: 1 notifiaction, 2 notifications, and 3+. | |
| 41 SkBitmap GetNotificationImage(int notification_count) { | |
| 42 SkBitmap image; | |
| 43 gfx::Image all = ui::ResourceBundle::GetSharedInstance().GetImageNamed( | |
| 44 IDR_AURA_UBER_TRAY_WEB_NOTIFICATON); | |
| 45 | |
| 46 int image_index = notification_count - 1; | |
| 47 image_index = std::max(0, std::min(image_index, 2)); | |
| 48 | |
| 49 SkIRect region = SkIRect::MakeXYWH( | |
| 50 0, image_index * kImageHeight, kImageWidth, kImageHeight); | |
| 51 all.ToSkBitmap()->extractSubset(&image, region); | |
| 52 return image; | |
| 53 } | |
| 54 | |
| 55 } // namespace | |
| 56 | |
| 57 namespace ash { | |
| 58 | |
| 59 namespace internal { | |
| 60 | |
| 61 struct WebNotification { | |
| 62 WebNotification(const std::string& i, | |
| 63 const string16& t, | |
| 64 const string16& m, | |
| 65 const string16& s, | |
| 66 const std::string& e) : | |
|
sadrul
2012/06/06 19:19:19
the ':' should be in the next line
stevenjb
2012/06/06 23:12:15
Done.
| |
| 67 id(i), | |
| 68 title(t), | |
| 69 message(m), | |
| 70 display_source(s), | |
| 71 extension_id(e) { | |
| 72 } | |
| 73 std::string id; | |
|
sadrul
2012/06/06 19:19:19
newline between 72/73
stevenjb
2012/06/06 23:12:15
Done.
| |
| 74 string16 title; | |
| 75 string16 message; | |
| 76 string16 display_source; | |
| 77 std::string extension_id; | |
| 78 SkBitmap image; | |
| 79 }; | |
| 80 | |
| 81 // A helper class to manage the list of notifications. | |
| 82 class WebNotificationList { | |
| 83 public: | |
| 84 typedef std::list<WebNotification> Notifications; | |
| 85 | |
| 86 WebNotificationList() { | |
| 87 } | |
| 88 | |
| 89 void AddNotification(const std::string& id, | |
| 90 const string16& title, | |
| 91 const string16& message, | |
| 92 const string16& source, | |
| 93 const std::string& extension) { | |
| 94 Notifications::iterator iter = GetNotification(id); | |
| 95 if (iter != notifications_.end()) { | |
| 96 iter->title = title; | |
| 97 iter->message = message; | |
| 98 iter->display_source = source; | |
| 99 iter->extension_id = extension; | |
| 100 } else { | |
| 101 notifications_.push_back( | |
| 102 WebNotification(id, title, message, source, extension)); | |
| 103 } | |
| 104 } | |
| 105 | |
| 106 bool RemoveNotification(const std::string& id) { | |
| 107 Notifications::iterator iter = GetNotification(id); | |
| 108 if (iter == notifications_.end()) | |
| 109 return false; | |
| 110 notifications_.erase(iter); | |
| 111 return true; | |
| 112 } | |
| 113 | |
| 114 void RemoveNotificationsBySource(const std::string& id) { | |
| 115 Notifications::iterator source_iter = GetNotification(id); | |
| 116 if (source_iter == notifications_.end()) | |
| 117 return; | |
| 118 string16 display_source = source_iter->display_source; | |
| 119 for (Notifications::iterator loopiter = notifications_.begin(); | |
| 120 loopiter != notifications_.end(); ) { | |
|
sadrul
2012/06/06 19:19:19
Can you just ++loopiter here instead of below?
..
stevenjb
2012/06/06 23:12:15
Correct.
| |
| 121 Notifications::iterator curiter = loopiter++; | |
| 122 if (curiter->display_source == display_source) | |
| 123 notifications_.erase(curiter); | |
| 124 } | |
| 125 } | |
| 126 | |
| 127 void RemoveNotificationsByExtension(const std::string& id) { | |
| 128 Notifications::iterator source_iter = GetNotification(id); | |
| 129 if (source_iter == notifications_.end()) | |
| 130 return; | |
| 131 std::string extension_id = source_iter->extension_id; | |
| 132 for (Notifications::iterator loopiter = notifications_.begin(); | |
| 133 loopiter != notifications_.end(); ) { | |
| 134 Notifications::iterator curiter = loopiter++; | |
| 135 if (curiter->extension_id == extension_id) | |
| 136 notifications_.erase(curiter); | |
| 137 } | |
| 138 } | |
| 139 | |
| 140 bool SetNotificationImage(const std::string& id, | |
| 141 const SkBitmap& image) { | |
| 142 Notifications::iterator iter = GetNotification(id); | |
| 143 if (iter == notifications_.end()) | |
| 144 return false; | |
| 145 iter->image = image; | |
| 146 return true; | |
| 147 } | |
| 148 | |
| 149 const Notifications& notifications() const { return notifications_; } | |
| 150 | |
| 151 private: | |
| 152 Notifications::iterator GetNotification(const std::string& id) { | |
| 153 for (Notifications::iterator iter = notifications_.begin(); | |
| 154 iter != notifications_.end(); ++iter) { | |
| 155 if (iter->id == id) | |
| 156 return iter; | |
| 157 } | |
| 158 return notifications_.end(); | |
| 159 } | |
| 160 | |
| 161 Notifications notifications_; | |
| 162 | |
| 163 DISALLOW_COPY_AND_ASSIGN(WebNotificationList); | |
| 164 }; | |
| 165 | |
| 166 // A simple view for the text (title and message) of a notification. | |
| 167 class WebNotificationMessageView : public views::View { | |
| 168 public: | |
| 169 explicit WebNotificationMessageView(const WebNotification& notification) { | |
| 170 views::Label* title = new views::Label(notification.title); | |
| 171 title->SetHorizontalAlignment(views::Label::ALIGN_LEFT); | |
| 172 title->SetFont(title->font().DeriveFont(0, gfx::Font::BOLD)); | |
| 173 views::Label* message = new views::Label(notification.message); | |
| 174 message->SetHorizontalAlignment(views::Label::ALIGN_LEFT); | |
| 175 message->SetMultiLine(true); | |
| 176 | |
| 177 SetLayoutManager( | |
| 178 new views::BoxLayout(views::BoxLayout::kVertical, 0, 0, 1)); | |
| 179 AddChildView(title); | |
| 180 AddChildView(message); | |
| 181 } | |
| 182 | |
| 183 virtual ~WebNotificationMessageView() { | |
| 184 } | |
| 185 | |
| 186 private: | |
| 187 DISALLOW_COPY_AND_ASSIGN(WebNotificationMessageView); | |
| 188 }; | |
| 189 | |
| 190 const int kTogglePermissionCommand = 0; | |
| 191 const int kToggleExtensionCommand = 1; | |
| 192 const int kShowSettingsCommand = 2; | |
|
sadrul
2012/06/06 19:19:19
These could be in the anonymous namespace
stevenjb
2012/06/06 23:12:15
Done.
| |
| 193 | |
| 194 // A dropdown menu for notifications. | |
| 195 class WebNotificationMenuModel : public ui::SimpleMenuModel, | |
| 196 public ui::SimpleMenuModel::Delegate { | |
| 197 public: | |
| 198 explicit WebNotificationMenuModel(WebNotificationTray* tray, | |
| 199 const WebNotification& notification) | |
| 200 : ALLOW_THIS_IN_INITIALIZER_LIST(ui::SimpleMenuModel(this)), | |
| 201 tray_(tray), | |
| 202 notification_(notification) { | |
| 203 // Add 'disable notifications' menu item. | |
| 204 if (!notification.extension_id.empty()) { | |
| 205 AddItem(kToggleExtensionCommand, | |
| 206 GetLabelForCommandId(kToggleExtensionCommand)); | |
| 207 } else if (!notification.display_source.empty()) { | |
| 208 AddItem(kTogglePermissionCommand, | |
| 209 GetLabelForCommandId(kTogglePermissionCommand)); | |
| 210 } | |
| 211 // Add settings menu item. | |
| 212 if (!notification.display_source.empty()) { | |
| 213 AddItem(kShowSettingsCommand, | |
| 214 GetLabelForCommandId(kShowSettingsCommand)); | |
| 215 } | |
| 216 } | |
| 217 | |
| 218 virtual ~WebNotificationMenuModel() { | |
| 219 } | |
| 220 | |
| 221 // Overridden from ui::SimpleMenuModel: | |
| 222 virtual string16 GetLabelForCommandId(int command_id) const OVERRIDE { | |
| 223 switch (command_id) { | |
| 224 case kToggleExtensionCommand: | |
| 225 return l10n_util::GetStringUTF16( | |
| 226 IDS_ASH_WEB_NOTFICATION_TRAY_EXTENSIONS_DISABLE); | |
| 227 case kTogglePermissionCommand: | |
| 228 return l10n_util::GetStringFUTF16( | |
| 229 IDS_ASH_WEB_NOTFICATION_TRAY_SITE_DISABLE, | |
| 230 notification_.display_source); | |
| 231 case kShowSettingsCommand: | |
| 232 return l10n_util::GetStringUTF16( | |
| 233 IDS_ASH_WEB_NOTFICATION_TRAY_SETTINGS); | |
| 234 default: | |
| 235 NOTREACHED(); | |
| 236 } | |
| 237 return string16(); | |
| 238 } | |
| 239 | |
| 240 // Overridden from ui::SimpleMenuModel::Delegate: | |
| 241 virtual bool IsCommandIdChecked(int command_id) const OVERRIDE { | |
| 242 return false; | |
| 243 } | |
| 244 | |
| 245 virtual bool IsCommandIdEnabled(int command_id) const OVERRIDE { | |
| 246 return true; | |
| 247 } | |
| 248 | |
| 249 virtual bool GetAcceleratorForCommandId( | |
| 250 int command_id, | |
| 251 ui::Accelerator* accelerator) OVERRIDE { | |
| 252 return false; | |
| 253 } | |
| 254 | |
| 255 virtual void ExecuteCommand(int command_id) OVERRIDE { | |
| 256 switch (command_id) { | |
| 257 case kToggleExtensionCommand: | |
| 258 tray_->DisableByExtension(notification_.id); | |
| 259 break; | |
| 260 case kTogglePermissionCommand: | |
| 261 tray_->DisableByUrl(notification_.id); | |
| 262 break; | |
| 263 case kShowSettingsCommand: | |
| 264 tray_->ShowSettings(notification_.id); | |
| 265 break; | |
| 266 default: | |
| 267 NOTREACHED(); | |
| 268 } | |
| 269 } | |
| 270 | |
| 271 private: | |
| 272 WebNotificationTray* tray_; | |
| 273 WebNotification notification_; | |
| 274 | |
| 275 DISALLOW_COPY_AND_ASSIGN(WebNotificationMenuModel); | |
| 276 }; | |
| 277 | |
| 278 // The view for a notification entry (icon + message + buttons). | |
| 279 class WebNotificationView : public views::View, | |
| 280 public views::ButtonListener, | |
| 281 public views::MenuButtonListener { | |
| 282 public: | |
| 283 WebNotificationView(WebNotificationTray* tray, | |
| 284 const WebNotification& notification) | |
| 285 : tray_(tray), | |
| 286 notification_(notification), | |
| 287 icon_(NULL), | |
| 288 menu_button_(NULL), | |
| 289 close_button_(NULL) { | |
| 290 InitView(tray, notification); | |
| 291 } | |
| 292 | |
| 293 virtual ~WebNotificationView() { | |
| 294 } | |
| 295 | |
| 296 void InitView(WebNotificationTray* tray, | |
| 297 const WebNotification& notification) { | |
| 298 set_border(views::Border::CreateSolidSidedBorder( | |
| 299 1, 0, 0, 0, kBorderLightColor)); | |
| 300 set_background(views::Background::CreateSolidBackground(kBackgroundColor)); | |
| 301 | |
| 302 icon_ = new views::ImageView; | |
| 303 icon_->SetImage(notification.image); | |
| 304 | |
| 305 WebNotificationMessageView* message_view | |
| 306 = new WebNotificationMessageView(notification); | |
| 307 | |
| 308 close_button_ = new views::ImageButton(this); | |
| 309 close_button_->SetImage( | |
| 310 views::CustomButton::BS_NORMAL, | |
| 311 ResourceBundle::GetSharedInstance().GetImageSkiaNamed( | |
| 312 IDR_AURA_WINDOW_CLOSE)); | |
| 313 | |
| 314 menu_button_ = new views::MenuButton(NULL, string16(), this, true); | |
| 315 menu_button_->set_border(NULL); | |
| 316 | |
| 317 views::GridLayout* layout = new views::GridLayout(this); | |
| 318 SetLayoutManager(layout); | |
| 319 | |
| 320 views::ColumnSet* columns = layout->AddColumnSet(0); | |
| 321 | |
| 322 columns->AddPaddingColumn(0, kTrayPopupPaddingHorizontal/2); | |
| 323 | |
| 324 // Notification Icon. | |
| 325 columns->AddColumn(views::GridLayout::CENTER, views::GridLayout::CENTER, | |
| 326 0, /* resize percent */ | |
| 327 views::GridLayout::FIXED, | |
| 328 kNotificationIconWidth, kNotificationIconWidth); | |
| 329 | |
| 330 columns->AddPaddingColumn(0, kTrayPopupPaddingHorizontal/2); | |
| 331 | |
| 332 // Notification message text. | |
| 333 columns->AddColumn(views::GridLayout::FILL, views::GridLayout::FILL, | |
| 334 100, /* resize percent */ | |
| 335 views::GridLayout::USE_PREF, 0, 0); | |
| 336 | |
| 337 columns->AddPaddingColumn(0, kTrayPopupPaddingHorizontal/2); | |
| 338 | |
| 339 // Close and menu buttons. | |
| 340 columns->AddColumn(views::GridLayout::CENTER, views::GridLayout::CENTER, | |
| 341 0, /* resize percent */ | |
| 342 views::GridLayout::FIXED, | |
| 343 kWebNotificationButtonWidth, | |
| 344 kWebNotificationButtonWidth); | |
| 345 | |
| 346 columns->AddPaddingColumn(0, kTrayPopupPaddingHorizontal/2); | |
| 347 | |
| 348 // Layout rows | |
| 349 layout->AddPaddingRow(0, kTrayPopupPaddingBetweenItems); | |
| 350 | |
| 351 layout->StartRow(0, 0); | |
| 352 layout->AddView(icon_, 1, 2); | |
| 353 layout->AddView(message_view, 1, 2); | |
| 354 layout->AddView(close_button_); | |
| 355 | |
| 356 layout->StartRow(0, 0); | |
| 357 layout->SkipColumns(4); | |
| 358 layout->AddView(menu_button_); | |
| 359 | |
| 360 layout->AddPaddingRow(0, kTrayPopupPaddingBetweenItems); | |
| 361 } | |
| 362 | |
| 363 // Overridden from ButtonListener. | |
| 364 virtual void ButtonPressed(views::Button* sender, | |
| 365 const views::Event& event) OVERRIDE { | |
| 366 if (sender == close_button_) | |
| 367 tray_->RemoveNotification(notification_.id); | |
| 368 } | |
| 369 | |
| 370 // Overridden from MenuButtonListener. | |
| 371 virtual void OnMenuButtonClicked(View* source, const gfx::Point& point) { | |
| 372 if (source != menu_button_) | |
| 373 return; | |
| 374 WebNotificationMenuModel menu_model(tray_, notification_); | |
| 375 views::MenuModelAdapter menu_model_adapter(&menu_model); | |
| 376 views::MenuRunner menu_runner(menu_model_adapter.CreateMenu()); | |
| 377 | |
| 378 gfx::Point screen_location; | |
| 379 views::View::ConvertPointToScreen(menu_button_, &screen_location); | |
| 380 if (menu_runner.RunMenuAt( | |
| 381 source->GetWidget()->GetTopLevelWidget(), | |
| 382 menu_button_, | |
| 383 gfx::Rect(screen_location, menu_button_->size()), | |
| 384 views::MenuItemView::TOPRIGHT, | |
| 385 views::MenuRunner::HAS_MNEMONICS) | |
| 386 == views::MenuRunner::MENU_DELETED) { | |
|
sadrul
2012/06/06 19:19:19
== should be in the line above
stevenjb
2012/06/06 23:12:15
Used ignore_result instead.
| |
| 387 return; | |
| 388 } | |
| 389 } | |
| 390 | |
| 391 | |
| 392 private: | |
| 393 WebNotificationTray* tray_; | |
| 394 WebNotification notification_; | |
| 395 views::ImageView* icon_; | |
| 396 views::MenuButton* menu_button_; | |
| 397 views::ImageButton* close_button_; | |
| 398 | |
| 399 DISALLOW_COPY_AND_ASSIGN(WebNotificationView); | |
| 400 }; | |
| 401 | |
| 402 } // namespace internal | |
| 403 | |
| 404 using internal::WebNotificationList; | |
| 405 using internal::WebNotificationView; | |
| 406 | |
| 407 class WebNotificationTray::BubbleContentsView : public views::View { | |
| 408 public: | |
| 409 explicit BubbleContentsView(WebNotificationTray* tray) | |
| 410 : tray_(tray) { | |
| 411 SetLayoutManager(new views::FillLayout); | |
| 412 set_background(views::Background::CreateSolidBackground(kBackgroundColor)); | |
| 413 | |
| 414 scroll_content_ = new views::View; | |
| 415 scroll_content_->SetLayoutManager( | |
| 416 new views::BoxLayout(views::BoxLayout::kVertical, 0, 0, 1)); | |
| 417 scroller_ = new internal::FixedSizedScrollView; | |
| 418 scroller_->SetContentsView(scroll_content_); | |
| 419 | |
| 420 AddChildView(scroller_); | |
| 421 } | |
| 422 | |
| 423 void Update(const WebNotificationList::Notifications& notifications) { | |
| 424 scroll_content_->RemoveAllChildViews(true); | |
| 425 for (WebNotificationList::Notifications::const_iterator iter = | |
| 426 notifications.begin(); iter != notifications.end(); ++iter) { | |
| 427 WebNotificationView* view = new WebNotificationView(tray_, *iter); | |
| 428 scroll_content_->AddChildView(view); | |
| 429 } | |
| 430 scroller_->Layout(); | |
| 431 SizeToPreferredSize(); | |
| 432 Layout(); | |
| 433 PreferredSizeChanged(); | |
| 434 SchedulePaint(); | |
| 435 } | |
| 436 | |
| 437 // views::View overrides. | |
| 438 virtual gfx::Size GetPreferredSize() { | |
| 439 gfx::Size preferred_size = scroll_content_->GetPreferredSize(); | |
| 440 int height = std::min( | |
| 441 std::max(preferred_size.height(), kWebNotificationBubbleMinHeight), | |
| 442 kWebNotificationBubbleMaxHeight); | |
| 443 preferred_size.set_height(height); | |
| 444 return preferred_size; | |
| 445 } | |
| 446 | |
| 447 private: | |
| 448 WebNotificationTray* tray_; | |
| 449 internal::FixedSizedScrollView* scroller_; | |
| 450 views::View* scroll_content_; | |
| 451 | |
| 452 DISALLOW_COPY_AND_ASSIGN(BubbleContentsView); | |
| 453 }; | |
| 454 | |
| 455 class WebNotificationTray::Bubble | |
| 456 : public internal::SystemTrayBubbleView::Host { | |
| 457 public: | |
| 458 explicit Bubble(WebNotificationTray* tray) : | |
|
sadrul
2012/06/06 19:19:19
: should be in the next line
stevenjb
2012/06/06 23:12:15
Done.
| |
| 459 tray_(tray), | |
| 460 bubble_view_(NULL), | |
| 461 bubble_widget_(NULL), | |
| 462 contents_view_(NULL) { | |
| 463 views::View* anchor = tray->tray_container(); | |
| 464 views::BubbleBorder::ArrowLocation arrow_location; | |
| 465 int arrow_offset = 0; | |
| 466 if (tray_->shelf_alignment() == SHELF_ALIGNMENT_BOTTOM) { | |
| 467 arrow_location = views::BubbleBorder::BOTTOM_RIGHT; | |
| 468 arrow_offset = anchor->GetContentsBounds().width() / 2; | |
| 469 } else if (tray_->shelf_alignment() == SHELF_ALIGNMENT_LEFT) { | |
| 470 arrow_location = views::BubbleBorder::LEFT_BOTTOM; | |
| 471 } else { | |
| 472 arrow_location = views::BubbleBorder::RIGHT_BOTTOM; | |
| 473 } | |
| 474 bubble_view_ = new internal::SystemTrayBubbleView( | |
| 475 anchor, arrow_location, this, false); | |
| 476 bubble_view_->SetMaxHeight(kWebNotificationBubbleMaxHeight); | |
| 477 bubble_view_->set_bubble_width(kWebNotificationWidth); | |
| 478 | |
| 479 bubble_widget_ = views::BubbleDelegateView::CreateBubble(bubble_view_); | |
| 480 | |
| 481 bubble_view_->SetAlignment(views::BubbleBorder::ALIGN_EDGE_TO_ANCHOR_EDGE); | |
| 482 bubble_widget_->non_client_view()->frame_view()->set_background(NULL); | |
| 483 bubble_view_->SetBubbleBorder(arrow_offset); | |
| 484 | |
| 485 contents_view_ = new BubbleContentsView(tray); | |
| 486 bubble_view_->AddChildView(contents_view_); | |
| 487 | |
| 488 Update(); | |
| 489 bubble_view_->Show(); | |
| 490 | |
| 491 // Don't show any system notifications while we are viewing web | |
| 492 // notifications. As soon as the web notification tray looses focus, | |
| 493 // this will be destroyed and any hidden or suppressed system notifications | |
| 494 // will be displayed (see ~Bubble below). | |
| 495 ash::Shell::GetInstance()->system_tray()->SetHideNotifications(true); | |
| 496 } | |
| 497 | |
| 498 ~Bubble() { | |
|
sadrul
2012/06/06 19:19:19
virtual?
stevenjb
2012/06/06 23:12:15
Oops! Done.
| |
| 499 if (bubble_view_) | |
| 500 bubble_view_->reset_host(); | |
| 501 if (bubble_widget_) | |
| 502 bubble_widget_->Close(); | |
| 503 | |
| 504 // Show any hidden or suppressed system notifications. | |
| 505 ash::Shell::GetInstance()->system_tray()->SetHideNotifications(false); | |
| 506 } | |
| 507 | |
| 508 // Overridden from SystemTrayBubbleView::Host. | |
| 509 virtual void BubbleViewDestroyed() OVERRIDE { | |
| 510 } | |
| 511 | |
| 512 virtual gfx::Rect GetAnchorRect() const OVERRIDE { | |
| 513 gfx::Rect anchor_rect = tray_->tray_container()->GetScreenBounds(); | |
| 514 return anchor_rect; | |
| 515 } | |
| 516 | |
| 517 virtual void OnMouseEnteredView() OVERRIDE { | |
| 518 } | |
| 519 | |
| 520 virtual void OnMoiseExitedView() OVERRIDE { | |
| 521 } | |
| 522 | |
| 523 void Update() { | |
| 524 contents_view_->Update(tray_->notification_list()->notifications()); | |
| 525 bubble_view_->Layout(); | |
| 526 bubble_view_->SchedulePaint(); | |
| 527 } | |
| 528 | |
| 529 views::Widget* bubble_widget() const { return bubble_widget_; } | |
| 530 | |
| 531 private: | |
| 532 WebNotificationTray* tray_; | |
| 533 internal::SystemTrayBubbleView* bubble_view_; | |
| 534 views::Widget* bubble_widget_; | |
| 535 BubbleContentsView* contents_view_; | |
| 536 }; | |
| 537 | |
| 538 WebNotificationTray::WebNotificationTray() | |
| 539 : notification_list_(new WebNotificationList()), | |
| 540 tray_container_(NULL), | |
| 541 icon_(NULL), | |
| 542 delegate_(NULL) { | |
| 543 tray_container_ = new views::View; | |
| 544 tray_container_->set_border(views::Border::CreateEmptyBorder( | |
| 545 kIconBorder, kIconBorder, kIconBorder, kIconBorder)); | |
| 546 SetShelfAlignment(shelf_alignment()); | |
| 547 | |
| 548 icon_ = new views::ImageView; | |
| 549 tray_container_->AddChildView(icon_); | |
| 550 UpdateIcon(); // Hides the tray initially. | |
| 551 | |
| 552 SetContents(tray_container_); | |
| 553 | |
| 554 Shell::GetInstance()->AddEnvEventFilter(this); | |
| 555 } | |
| 556 | |
| 557 WebNotificationTray::~WebNotificationTray() { | |
| 558 Shell::GetInstance()->RemoveEnvEventFilter(this); | |
| 559 } | |
| 560 | |
| 561 bool WebNotificationTray::PreHandleKeyEvent(aura::Window* target, | |
| 562 aura::KeyEvent* event) { | |
| 563 return false; | |
| 564 } | |
| 565 | |
| 566 bool WebNotificationTray::PreHandleMouseEvent(aura::Window* target, | |
| 567 aura::MouseEvent* event) { | |
| 568 if (event->type() == ui::ET_MOUSE_PRESSED) | |
| 569 ProcessLocatedEvent(*event); | |
| 570 return false; | |
| 571 } | |
| 572 | |
| 573 ui::TouchStatus WebNotificationTray::PreHandleTouchEvent(aura::Window* target, | |
| 574 aura::TouchEvent* event) { | |
| 575 if (event->type() == ui::ET_TOUCH_PRESSED) | |
| 576 ProcessLocatedEvent(*event); | |
| 577 return ui::TOUCH_STATUS_UNKNOWN; | |
| 578 } | |
| 579 | |
| 580 ui::GestureStatus WebNotificationTray::PreHandleGestureEvent( | |
| 581 aura::Window* target, | |
| 582 aura::GestureEvent* event) { | |
| 583 return ui::GESTURE_STATUS_UNKNOWN; | |
| 584 } | |
| 585 | |
| 586 void WebNotificationTray::SetShelfAlignment(ShelfAlignment alignment) { | |
| 587 internal::TrayBackgroundView::SetShelfAlignment(alignment); | |
| 588 tray_container_->SetLayoutManager(new views::BoxLayout( | |
| 589 alignment == SHELF_ALIGNMENT_BOTTOM ? | |
| 590 views::BoxLayout::kHorizontal : views::BoxLayout::kVertical, | |
| 591 0, 0, 0)); | |
| 592 } | |
| 593 | |
| 594 bool WebNotificationTray::PerformAction(const views::Event& event) { | |
| 595 if (bubble()) | |
| 596 bubble_.reset(); | |
| 597 else | |
| 598 bubble_.reset(new Bubble(this)); | |
| 599 return true; | |
| 600 } | |
| 601 | |
| 602 void WebNotificationTray::SetDelegate(Delegate* delegate) { | |
| 603 DCHECK(!delegate_); | |
| 604 delegate_ = delegate; | |
| 605 } | |
| 606 | |
| 607 void WebNotificationTray::AddNotification(const std::string& id, | |
| 608 const string16& title, | |
| 609 const string16& message, | |
| 610 const string16& source, | |
| 611 const std::string& extension) { | |
| 612 notification_list_->AddNotification(id, title, message, source, extension); | |
| 613 UpdateIcon(); | |
| 614 if (bubble()) { | |
| 615 bubble_->Update(); | |
| 616 } else { | |
| 617 // Only show the web notification bubble if the system tray is not shown. | |
| 618 if (!ash::Shell::GetInstance()->system_tray()->IsBubbleVisible()) | |
| 619 bubble_.reset(new Bubble(this)); | |
|
sadrul
2012/06/06 19:19:19
This seems unfortunate. It would be nice to say 's
stevenjb
2012/06/06 23:12:15
I can put this in the status area. I was going to
| |
| 620 } | |
| 621 } | |
| 622 | |
| 623 void WebNotificationTray::RemoveNotification(const std::string& id) { | |
| 624 if (!notification_list_->RemoveNotification(id)) | |
| 625 return; | |
| 626 if (delegate_) | |
| 627 delegate_->NotificationRemoved(id); | |
| 628 UpdateBubbleAndIcon(); | |
| 629 } | |
| 630 | |
| 631 void WebNotificationTray::SetNotificationImage(const std::string& id, | |
| 632 const SkBitmap& image) { | |
| 633 if (notification_list_->SetNotificationImage(id, image)) { | |
| 634 if (bubble()) | |
|
sadrul
2012/06/06 19:19:19
if (notification_list_ ... && bubble())
bubble_-
stevenjb
2012/06/06 23:12:15
I personally dislike combining functions that do s
| |
| 635 bubble_->Update(); | |
| 636 } | |
| 637 } | |
| 638 | |
| 639 void WebNotificationTray::DisableByExtension(const std::string& id) { | |
| 640 notification_list_->RemoveNotificationsByExtension(id); | |
| 641 UpdateBubbleAndIcon(); | |
| 642 if (delegate_) | |
| 643 delegate_->DisableExtension(id); | |
| 644 } | |
| 645 | |
| 646 void WebNotificationTray::DisableByUrl(const std::string& id) { | |
| 647 notification_list_->RemoveNotificationsBySource(id); | |
| 648 UpdateBubbleAndIcon(); | |
| 649 if (delegate_) | |
| 650 delegate_->DisableNotificationsFromSource(id); | |
| 651 } | |
| 652 | |
| 653 void WebNotificationTray::ShowSettings(const std::string& id) { | |
| 654 if (delegate_) | |
| 655 delegate_->ShowSettings(id); | |
| 656 } | |
| 657 | |
| 658 int WebNotificationTray::GetNotificationCount() const { | |
| 659 return notification_list()->notifications().size(); | |
| 660 } | |
| 661 | |
| 662 void WebNotificationTray::UpdateIcon() { | |
| 663 int count = GetNotificationCount(); | |
| 664 if (count == 0) { | |
| 665 SetVisible(false); | |
| 666 } else { | |
| 667 icon_->SetImage(GetNotificationImage(count)); | |
| 668 SetVisible(true); | |
| 669 } | |
| 670 PreferredSizeChanged(); | |
| 671 } | |
| 672 | |
| 673 void WebNotificationTray::UpdateBubbleAndIcon() { | |
| 674 UpdateIcon(); | |
| 675 if (bubble()) { | |
| 676 if (GetNotificationCount() == 0) | |
| 677 bubble_.reset(); | |
| 678 else | |
| 679 bubble_->Update(); | |
| 680 } | |
| 681 } | |
| 682 | |
| 683 void WebNotificationTray::ProcessLocatedEvent(const aura::LocatedEvent& event) { | |
| 684 if (!bubble()) | |
| 685 return; | |
| 686 gfx::Rect bounds = | |
| 687 bubble_->bubble_widget()->GetNativeWindow()->GetBoundsInRootWindow(); | |
| 688 if (!bounds.Contains(event.root_location())) | |
| 689 bubble_.reset(); | |
| 690 } | |
| 691 | |
| 692 } // namespace ash | |
| OLD | NEW |