Index: chrome/browser/notifications/balloon_collection_impl.cc |
=================================================================== |
--- chrome/browser/notifications/balloon_collection_impl.cc (revision 263121) |
+++ chrome/browser/notifications/balloon_collection_impl.cc (working copy) |
@@ -1,463 +0,0 @@ |
-// 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 "chrome/browser/notifications/balloon_collection_impl.h" |
- |
-#include "base/bind.h" |
-#include "base/logging.h" |
-#include "base/stl_util.h" |
-#include "chrome/browser/chrome_notification_types.h" |
-#include "chrome/browser/notifications/balloon.h" |
-#include "chrome/browser/notifications/balloon_host.h" |
-#include "chrome/browser/notifications/notification.h" |
-#include "chrome/browser/ui/browser.h" |
-#include "chrome/browser/ui/panels/docked_panel_collection.h" |
-#include "chrome/browser/ui/panels/panel.h" |
-#include "chrome/browser/ui/panels/panel_manager.h" |
-#include "content/public/browser/notification_registrar.h" |
-#include "content/public/browser/notification_service.h" |
-#include "ui/gfx/rect.h" |
-#include "ui/gfx/screen.h" |
-#include "ui/gfx/size.h" |
- |
-// Portion of the screen allotted for notifications. When notification balloons |
-// extend over this, no new notifications are shown until some are closed. |
-const double kPercentBalloonFillFactor = 0.7; |
- |
-// Allow at least this number of balloons on the screen. |
-const int kMinAllowedBalloonCount = 2; |
- |
-// The spacing between the balloon and the panel. |
-const int kVerticalSpacingBetweenBalloonAndPanel = 5; |
- |
-// Delay from the mouse leaving the balloon collection before |
-// there is a relayout, in milliseconds. |
-const int kRepositionDelayMs = 300; |
- |
- |
-BalloonCollectionImpl::BalloonCollectionImpl() |
- : reposition_factory_(this), |
- added_as_message_loop_observer_(false) { |
- registrar_.Add(this, chrome::NOTIFICATION_PANEL_COLLECTION_UPDATED, |
- content::NotificationService::AllSources()); |
- registrar_.Add(this, chrome::NOTIFICATION_PANEL_CHANGED_EXPANSION_STATE, |
- content::NotificationService::AllSources()); |
- |
- SetPositionPreference(BalloonCollection::DEFAULT_POSITION); |
-} |
- |
-BalloonCollectionImpl::~BalloonCollectionImpl() { |
- RemoveMessageLoopObserver(); |
-} |
- |
-void BalloonCollectionImpl::AddImpl(const Notification& notification, |
- Profile* profile, |
- bool add_to_front) { |
- Balloon* new_balloon = MakeBalloon(notification, profile); |
- // The +1 on width is necessary because width is fixed on notifications, |
- // so since we always have the max size, we would always hit the scrollbar |
- // condition. We are only interested in comparing height to maximum. |
- new_balloon->set_min_scrollbar_size(gfx::Size(1 + layout_.max_balloon_width(), |
- layout_.max_balloon_height())); |
- new_balloon->SetPosition(layout_.OffScreenLocation(), false); |
- new_balloon->Show(); |
- int count = base_.count(); |
- if (count > 0 && layout_.RequiresOffsets()) |
- new_balloon->set_offset(base_.balloons()[count - 1]->offset()); |
- base_.Add(new_balloon, add_to_front); |
- PositionBalloons(false); |
- |
- // There may be no listener in a unit test. |
- if (space_change_listener_) |
- space_change_listener_->OnBalloonSpaceChanged(); |
- |
- // This is used only for testing. |
- if (!on_collection_changed_callback_.is_null()) |
- on_collection_changed_callback_.Run(); |
-} |
- |
-void BalloonCollectionImpl::Add(const Notification& notification, |
- Profile* profile) { |
- AddImpl(notification, profile, false); |
-} |
- |
-const Notification* BalloonCollectionImpl::FindById( |
- const std::string& id) const { |
- return base_.FindById(id); |
-} |
- |
-bool BalloonCollectionImpl::RemoveById(const std::string& id) { |
- return base_.CloseById(id); |
-} |
- |
-bool BalloonCollectionImpl::RemoveBySourceOrigin(const GURL& origin) { |
- return base_.CloseAllBySourceOrigin(origin); |
-} |
- |
-bool BalloonCollectionImpl::RemoveByProfile(Profile* profile) { |
- return base_.CloseAllByProfile(profile); |
-} |
- |
-void BalloonCollectionImpl::RemoveAll() { |
- base_.CloseAll(); |
-} |
- |
-bool BalloonCollectionImpl::HasSpace() const { |
- int count = base_.count(); |
- if (count < kMinAllowedBalloonCount) |
- return true; |
- |
- int max_balloon_size = 0; |
- int total_size = 0; |
- layout_.GetMaxLinearSize(&max_balloon_size, &total_size); |
- |
- int current_max_size = max_balloon_size * count; |
- int max_allowed_size = static_cast<int>(total_size * |
- kPercentBalloonFillFactor); |
- return current_max_size < max_allowed_size - max_balloon_size; |
-} |
- |
-void BalloonCollectionImpl::ResizeBalloon(Balloon* balloon, |
- const gfx::Size& size) { |
- balloon->set_content_size(Layout::ConstrainToSizeLimits(size)); |
- PositionBalloons(true); |
-} |
- |
-void BalloonCollectionImpl::DisplayChanged() { |
- layout_.RefreshSystemMetrics(); |
- PositionBalloons(true); |
-} |
- |
-void BalloonCollectionImpl::OnBalloonClosed(Balloon* source) { |
- // We want to free the balloon when finished. |
- const Balloons& balloons = base_.balloons(); |
- |
- Balloons::const_iterator it = balloons.begin(); |
- if (layout_.RequiresOffsets()) { |
- gfx::Vector2d offset; |
- bool apply_offset = false; |
- while (it != balloons.end()) { |
- if (*it == source) { |
- ++it; |
- if (it != balloons.end()) { |
- apply_offset = true; |
- offset.set_y((source)->offset().y() - (*it)->offset().y() + |
- (*it)->content_size().height() - source->content_size().height()); |
- } |
- } else { |
- if (apply_offset) |
- (*it)->add_offset(offset); |
- ++it; |
- } |
- } |
- // Start listening for UI events so we cancel the offset when the mouse |
- // leaves the balloon area. |
- if (apply_offset) |
- AddMessageLoopObserver(); |
- } |
- |
- base_.Remove(source); |
- PositionBalloons(true); |
- |
- // There may be no listener in a unit test. |
- if (space_change_listener_) |
- space_change_listener_->OnBalloonSpaceChanged(); |
- |
- // This is used only for testing. |
- if (!on_collection_changed_callback_.is_null()) |
- on_collection_changed_callback_.Run(); |
-} |
- |
-const BalloonCollection::Balloons& BalloonCollectionImpl::GetActiveBalloons() { |
- return base_.balloons(); |
-} |
- |
-void BalloonCollectionImpl::Observe( |
- int type, |
- const content::NotificationSource& source, |
- const content::NotificationDetails& details) { |
- gfx::Rect bounds; |
- switch (type) { |
- case chrome::NOTIFICATION_PANEL_COLLECTION_UPDATED: |
- case chrome::NOTIFICATION_PANEL_CHANGED_EXPANSION_STATE: |
- layout_.enable_computing_panel_offset(); |
- if (layout_.ComputeOffsetToMoveAbovePanels()) |
- PositionBalloons(true); |
- break; |
- default: |
- NOTREACHED(); |
- break; |
- } |
-} |
- |
-void BalloonCollectionImpl::PositionBalloonsInternal(bool reposition) { |
- const Balloons& balloons = base_.balloons(); |
- |
- layout_.RefreshSystemMetrics(); |
- gfx::Point origin = layout_.GetLayoutOrigin(); |
- for (Balloons::const_iterator it = balloons.begin(); |
- it != balloons.end(); |
- ++it) { |
- gfx::Point upper_left = layout_.NextPosition((*it)->GetViewSize(), &origin); |
- (*it)->SetPosition(upper_left, reposition); |
- } |
-} |
- |
-gfx::Rect BalloonCollectionImpl::GetBalloonsBoundingBox() const { |
- // Start from the layout origin. |
- gfx::Rect bounds = gfx::Rect(layout_.GetLayoutOrigin(), gfx::Size(0, 0)); |
- |
- // For each balloon, extend the rectangle. This approach is indifferent to |
- // the orientation of the balloons. |
- const Balloons& balloons = base_.balloons(); |
- Balloons::const_iterator iter; |
- for (iter = balloons.begin(); iter != balloons.end(); ++iter) { |
- gfx::Rect balloon_box = gfx::Rect((*iter)->GetPosition(), |
- (*iter)->GetViewSize()); |
- bounds.Union(balloon_box); |
- } |
- |
- return bounds; |
-} |
- |
-void BalloonCollectionImpl::AddMessageLoopObserver() { |
- if (!added_as_message_loop_observer_) { |
- base::MessageLoopForUI::current()->AddObserver(this); |
- added_as_message_loop_observer_ = true; |
- } |
-} |
- |
-void BalloonCollectionImpl::RemoveMessageLoopObserver() { |
- if (added_as_message_loop_observer_) { |
- base::MessageLoopForUI::current()->RemoveObserver(this); |
- added_as_message_loop_observer_ = false; |
- } |
-} |
- |
-void BalloonCollectionImpl::CancelOffsets() { |
- reposition_factory_.InvalidateWeakPtrs(); |
- |
- // Unhook from listening to all UI events. |
- RemoveMessageLoopObserver(); |
- |
- const Balloons& balloons = base_.balloons(); |
- for (Balloons::const_iterator it = balloons.begin(); |
- it != balloons.end(); |
- ++it) |
- (*it)->set_offset(gfx::Vector2d()); |
- |
- PositionBalloons(true); |
-} |
- |
-void BalloonCollectionImpl::HandleMouseMoveEvent() { |
- if (!IsCursorInBalloonCollection()) { |
- // Mouse has left the region. Schedule a reposition after |
- // a short delay. |
- if (!reposition_factory_.HasWeakPtrs()) { |
- base::MessageLoop::current()->PostDelayedTask( |
- FROM_HERE, |
- base::Bind(&BalloonCollectionImpl::CancelOffsets, |
- reposition_factory_.GetWeakPtr()), |
- base::TimeDelta::FromMilliseconds(kRepositionDelayMs)); |
- } |
- } else { |
- // Mouse moved back into the region. Cancel the reposition. |
- reposition_factory_.InvalidateWeakPtrs(); |
- } |
-} |
- |
-BalloonCollectionImpl::Layout::Layout() |
- : placement_(INVALID), |
- need_to_compute_panel_offset_(false), |
- offset_to_move_above_panels_(0) { |
- RefreshSystemMetrics(); |
-} |
- |
-void BalloonCollectionImpl::Layout::GetMaxLinearSize(int* max_balloon_size, |
- int* total_size) const { |
- DCHECK(max_balloon_size && total_size); |
- |
- // All placement schemes are vertical, so we only care about height. |
- *total_size = work_area_.height(); |
- *max_balloon_size = max_balloon_height(); |
-} |
- |
-gfx::Point BalloonCollectionImpl::Layout::GetLayoutOrigin() const { |
- // For lower-left and lower-right positioning, we need to add an offset |
- // to ensure balloons to stay on top of panels to avoid overlapping. |
- int x = 0; |
- int y = 0; |
- switch (placement_) { |
- case VERTICALLY_FROM_TOP_LEFT: { |
- x = work_area_.x() + HorizontalEdgeMargin(); |
- y = work_area_.y() + VerticalEdgeMargin() + offset_to_move_above_panels_; |
- break; |
- } |
- case VERTICALLY_FROM_TOP_RIGHT: { |
- x = work_area_.right() - HorizontalEdgeMargin(); |
- y = work_area_.y() + VerticalEdgeMargin() + offset_to_move_above_panels_; |
- break; |
- } |
- case VERTICALLY_FROM_BOTTOM_LEFT: |
- x = work_area_.x() + HorizontalEdgeMargin(); |
- y = work_area_.bottom() - VerticalEdgeMargin() - |
- offset_to_move_above_panels_; |
- break; |
- case VERTICALLY_FROM_BOTTOM_RIGHT: |
- x = work_area_.right() - HorizontalEdgeMargin(); |
- y = work_area_.bottom() - VerticalEdgeMargin() - |
- offset_to_move_above_panels_; |
- break; |
- default: |
- NOTREACHED(); |
- break; |
- } |
- return gfx::Point(x, y); |
-} |
- |
-gfx::Point BalloonCollectionImpl::Layout::NextPosition( |
- const gfx::Size& balloon_size, |
- gfx::Point* position_iterator) const { |
- DCHECK(position_iterator); |
- |
- int x = 0; |
- int y = 0; |
- switch (placement_) { |
- case VERTICALLY_FROM_TOP_LEFT: |
- x = position_iterator->x(); |
- y = position_iterator->y(); |
- position_iterator->set_y(position_iterator->y() + balloon_size.height() + |
- InterBalloonMargin()); |
- break; |
- case VERTICALLY_FROM_TOP_RIGHT: |
- x = position_iterator->x() - balloon_size.width(); |
- y = position_iterator->y(); |
- position_iterator->set_y(position_iterator->y() + balloon_size.height() + |
- InterBalloonMargin()); |
- break; |
- case VERTICALLY_FROM_BOTTOM_LEFT: |
- position_iterator->set_y(position_iterator->y() - balloon_size.height() - |
- InterBalloonMargin()); |
- x = position_iterator->x(); |
- y = position_iterator->y(); |
- break; |
- case VERTICALLY_FROM_BOTTOM_RIGHT: |
- position_iterator->set_y(position_iterator->y() - balloon_size.height() - |
- InterBalloonMargin()); |
- x = position_iterator->x() - balloon_size.width(); |
- y = position_iterator->y(); |
- break; |
- default: |
- NOTREACHED(); |
- break; |
- } |
- return gfx::Point(x, y); |
-} |
- |
-gfx::Point BalloonCollectionImpl::Layout::OffScreenLocation() const { |
- gfx::Point location = GetLayoutOrigin(); |
- switch (placement_) { |
- case VERTICALLY_FROM_TOP_LEFT: |
- case VERTICALLY_FROM_BOTTOM_LEFT: |
- location.Offset(0, kBalloonMaxHeight); |
- break; |
- case VERTICALLY_FROM_TOP_RIGHT: |
- case VERTICALLY_FROM_BOTTOM_RIGHT: |
- location.Offset(-kBalloonMaxWidth - BalloonView::GetHorizontalMargin(), |
- kBalloonMaxHeight); |
- break; |
- default: |
- NOTREACHED(); |
- break; |
- } |
- return location; |
-} |
- |
-bool BalloonCollectionImpl::Layout::RequiresOffsets() const { |
- // Layout schemes that grow up from the bottom require offsets; |
- // schemes that grow down do not require offsets. |
- bool offsets = (placement_ == VERTICALLY_FROM_BOTTOM_LEFT || |
- placement_ == VERTICALLY_FROM_BOTTOM_RIGHT); |
- return offsets; |
-} |
- |
-// static |
-gfx::Size BalloonCollectionImpl::Layout::ConstrainToSizeLimits( |
- const gfx::Size& size) { |
- // restrict to the min & max sizes |
- return gfx::Size( |
- std::max(min_balloon_width(), |
- std::min(max_balloon_width(), size.width())), |
- std::max(min_balloon_height(), |
- std::min(max_balloon_height(), size.height()))); |
-} |
- |
-bool BalloonCollectionImpl::Layout::ComputeOffsetToMoveAbovePanels() { |
- // If the offset is not enabled due to that we have not received a |
- // notification about panel, don't proceed because we don't want to call |
- // PanelManager::GetInstance() to create an instance when panel is not |
- // present. |
- if (!need_to_compute_panel_offset_) |
- return false; |
- |
- const DockedPanelCollection::Panels& panels = |
- PanelManager::GetInstance()->docked_collection()->panels(); |
- int offset_to_move_above_panels = 0; |
- |
- // The offset is the maximum height of panels that could overlap with the |
- // balloons. |
- if (NeedToMoveAboveLeftSidePanels()) { |
- for (DockedPanelCollection::Panels::const_reverse_iterator iter = |
- panels.rbegin(); |
- iter != panels.rend(); ++iter) { |
- // No need to check panels beyond the area occupied by the balloons. |
- if ((*iter)->GetBounds().x() >= work_area_.x() + max_balloon_width()) |
- break; |
- |
- int current_height = (*iter)->GetBounds().height(); |
- if (current_height > offset_to_move_above_panels) |
- offset_to_move_above_panels = current_height; |
- } |
- } else if (NeedToMoveAboveRightSidePanels()) { |
- for (DockedPanelCollection::Panels::const_iterator iter = panels.begin(); |
- iter != panels.end(); ++iter) { |
- // No need to check panels beyond the area occupied by the balloons. |
- if ((*iter)->GetBounds().right() <= |
- work_area_.right() - max_balloon_width()) |
- break; |
- |
- int current_height = (*iter)->GetBounds().height(); |
- if (current_height > offset_to_move_above_panels) |
- offset_to_move_above_panels = current_height; |
- } |
- } |
- |
- // Ensure that we have some sort of margin between the 1st balloon and the |
- // panel beneath it even the vertical edge margin is 0 as on Mac. |
- if (offset_to_move_above_panels && !VerticalEdgeMargin()) |
- offset_to_move_above_panels += kVerticalSpacingBetweenBalloonAndPanel; |
- |
- // If no change is detected, return false to indicate that we do not need to |
- // reposition balloons. |
- if (offset_to_move_above_panels_ == offset_to_move_above_panels) |
- return false; |
- |
- offset_to_move_above_panels_ = offset_to_move_above_panels; |
- return true; |
-} |
- |
-bool BalloonCollectionImpl::Layout::RefreshSystemMetrics() { |
- bool changed = false; |
- |
- // TODO(scottmg): NativeScreen is wrong. http://crbug.com/133312 |
- gfx::Rect new_work_area = |
- gfx::Screen::GetNativeScreen()->GetPrimaryDisplay().work_area(); |
- if (work_area_ != new_work_area) { |
- work_area_.SetRect(new_work_area.x(), new_work_area.y(), |
- new_work_area.width(), new_work_area.height()); |
- changed = true; |
- } |
- |
- return changed; |
-} |