summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--chrome/browser/views/browser_actions_container.cc209
-rw-r--r--chrome/browser/views/browser_actions_container.h17
2 files changed, 109 insertions, 117 deletions
diff --git a/chrome/browser/views/browser_actions_container.cc b/chrome/browser/views/browser_actions_container.cc
index 15bb3e7..7d5686c 100644
--- a/chrome/browser/views/browser_actions_container.cc
+++ b/chrome/browser/views/browser_actions_container.cc
@@ -285,27 +285,26 @@ BrowserActionsContainer::BrowserActionsContainer(
toolbar_(toolbar),
popup_(NULL),
popup_button_(NULL),
+ model_(NULL),
resize_gripper_(NULL),
chevron_(NULL),
suppress_chevron_(false),
resize_amount_(0),
animation_target_size_(0),
ALLOW_THIS_IN_INITIALIZER_LIST(task_factory_(this)) {
+ SetID(VIEW_ID_BROWSER_ACTION_TOOLBAR);
+
ExtensionsService* extension_service = profile->GetExtensionsService();
if (!extension_service) // The |extension_service| can be NULL in Incognito.
return;
- registrar_.Add(this, NotificationType::EXTENSION_LOADED,
- Source<Profile>(profile_));
- registrar_.Add(this, NotificationType::EXTENSION_UNLOADED,
- Source<Profile>(profile_));
- registrar_.Add(this, NotificationType::EXTENSION_UNLOADED_DISABLED,
- Source<Profile>(profile_));
registrar_.Add(this, NotificationType::EXTENSION_HOST_VIEW_SHOULD_CLOSE,
Source<Profile>(profile_));
- resize_animation_.reset(new SlideAnimation(this));
+ model_ = extension_service->toolbar_model();
+ model_->AddObserver(this);
+ resize_animation_.reset(new SlideAnimation(this));
resize_gripper_ = new views::ResizeGripper(this);
resize_gripper_->SetVisible(false);
AddChildView(resize_gripper_);
@@ -323,11 +322,11 @@ BrowserActionsContainer::BrowserActionsContainer(
int predefined_width =
profile_->GetPrefs()->GetInteger(prefs::kBrowserActionContainerWidth);
container_size_ = gfx::Size(predefined_width, kButtonSize);
-
- SetID(VIEW_ID_BROWSER_ACTION_TOOLBAR);
}
BrowserActionsContainer::~BrowserActionsContainer() {
+ if (model_)
+ model_->RemoveObserver(this);
CloseOverflowMenu();
HidePopup();
DeleteBrowserActionViews();
@@ -363,88 +362,6 @@ void BrowserActionsContainer::RefreshBrowserActionViews() {
browser_action_views_[i]->button()->UpdateState();
}
-void BrowserActionsContainer::AddBrowserAction(Extension* extension) {
-#if defined(DEBUG)
- for (size_t i = 0; i < browser_action_views_.size(); ++i) {
- DCHECK(browser_action_views_[i]->button()->extension() != extension) <<
- "Asked to add a browser action view for an extension that already "
- "exists.";
- }
-#endif
- if (!extension->browser_action())
- return;
-
- // Before we change anything, determine the number of visible browser actions.
- size_t visible_actions = VisibleBrowserActions();
-
- // Add the new browser action to the vector and the view hierarchy.
- BrowserActionView* view = new BrowserActionView(extension, this);
- browser_action_views_.push_back(view);
- AddChildView(view);
-
- // For details on why we do the following see the class comments in the
- // header.
-
- // Determine if we need to increase (we only do that if the container was
- // showing all icons before the addition of this icon). We use -1 because
- // we don't want to count the view that we just added.
- if (visible_actions < browser_action_views_.size() - 1) {
- // Some icons were hidden, don't increase the size of the container.
- OnBrowserActionVisibilityChanged();
- } else {
- // Container was at max, increase the size of it by one icon.
- animation_target_size_ = IconCountToWidth(visible_actions + 1);
-
- // We don't want the chevron to appear while we animate. See documentation
- // in the header for why we do this.
- suppress_chevron_ = !chevron_->IsVisible();
-
- // Animate!
- resize_animation_->Reset();
- resize_animation_->SetTweenType(SlideAnimation::NONE);
- resize_animation_->Show();
- }
-}
-
-void BrowserActionsContainer::RemoveBrowserAction(Extension* extension) {
- if (!extension->browser_action())
- return;
-
- if (popup_ && popup_->host()->extension() == extension) {
- HidePopup();
- }
-
- // Before we change anything, determine the number of visible browser actions.
- int visible_actions = VisibleBrowserActions();
-
- for (std::vector<BrowserActionView*>::iterator iter =
- browser_action_views_.begin(); iter != browser_action_views_.end();
- ++iter) {
- if ((*iter)->button()->extension() == extension) {
- RemoveChildView(*iter);
- delete *iter;
- browser_action_views_.erase(iter);
-
- // For details on why we do the following see the class comments in the
- // header.
-
- // Calculate the target size we'll animate to (end state). This might be
- // the same size (if the icon we are removing is in the overflow bucket
- // and there are other icons there). We don't decrement visible_actions
- // because we want the container to stay the same size (clamping will take
- // care of shrinking the container if there aren't enough icons to show).
- animation_target_size_ =
- ClampToNearestIconCount(IconCountToWidth(visible_actions));
-
- // Animate!
- resize_animation_->Reset();
- resize_animation_->SetTweenType(SlideAnimation::EASE_OUT);
- resize_animation_->Show();
- return;
- }
- }
-}
-
void BrowserActionsContainer::CloseOverflowMenu() {
// Close the overflow menu if open (and the context menu off of that).
if (overflow_menu_.get())
@@ -637,15 +554,21 @@ void BrowserActionsContainer::Paint(gfx::Canvas* canvas) {
void BrowserActionsContainer::ViewHierarchyChanged(bool is_add,
views::View* parent,
views::View* child) {
+ // No extensions (e.g., incognito).
+ if (!model_)
+ return;
+
if (is_add && child == this) {
+ // Initial toolbar button creation and placement in the widget hierarchy.
// We do this here instead of in the constructor because AddBrowserAction
// calls Layout on the Toolbar, which needs this object to be constructed
// before its Layout function is called.
- ExtensionsService* extension_service = profile_->GetExtensionsService();
- if (!extension_service)
- return; // The |extension_service| can be NULL in Incognito.
- for (size_t i = 0; i < extension_service->extensions()->size(); ++i)
- AddBrowserAction(extension_service->extensions()->at(i));
+ for (ExtensionList::iterator iter = model_->begin();
+ iter != model_->end(); ++iter) {
+ BrowserActionView* view = new BrowserActionView(*iter, this);
+ browser_action_views_.push_back(view);
+ AddChildView(view);
+ }
}
}
@@ -653,19 +576,6 @@ void BrowserActionsContainer::Observe(NotificationType type,
const NotificationSource& source,
const NotificationDetails& details) {
switch (type.value) {
- case NotificationType::EXTENSION_LOADED:
- CloseOverflowMenu();
- AddBrowserAction(Details<Extension>(details).ptr());
- OnBrowserActionVisibilityChanged();
- break;
-
- case NotificationType::EXTENSION_UNLOADED:
- case NotificationType::EXTENSION_UNLOADED_DISABLED:
- CloseOverflowMenu();
- RemoveBrowserAction(Details<Extension>(details).ptr());
- OnBrowserActionVisibilityChanged();
- break;
-
case NotificationType::EXTENSION_HOST_VIEW_SHOULD_CLOSE:
// If we aren't the host of the popup, then disregard the notification.
if (!popup_ || Details<ExtensionHost>(popup_->host()) != details)
@@ -765,6 +675,87 @@ int BrowserActionsContainer::ClampToNearestIconCount(int pixelWidth) const {
return returning;
}
+void BrowserActionsContainer::BrowserActionAdded(Extension* extension,
+ int index) {
+#if defined(DEBUG)
+ for (size_t i = 0; i < browser_action_views_.size(); ++i) {
+ DCHECK(browser_action_views_[i]->button()->extension() != extension) <<
+ "Asked to add a browser action view for an extension that already "
+ "exists.";
+ }
+#endif
+
+ CloseOverflowMenu();
+
+ // Before we change anything, determine the number of visible browser actions.
+ size_t visible_actions = VisibleBrowserActions();
+
+ // Add the new browser action to the vector and the view hierarchy.
+ BrowserActionView* view = new BrowserActionView(extension, this);
+ browser_action_views_.push_back(view);
+ AddChildView(index, view);
+
+ // For details on why we do the following see the class comments in the
+ // header.
+
+ // Determine if we need to increase (we only do that if the container was
+ // showing all icons before the addition of this icon). We use -1 because
+ // we don't want to count the view that we just added.
+ if (visible_actions < browser_action_views_.size() - 1) {
+ // Some icons were hidden, don't increase the size of the container.
+ OnBrowserActionVisibilityChanged();
+ } else {
+ // Container was at max, increase the size of it by one icon.
+ animation_target_size_ = IconCountToWidth(visible_actions + 1);
+
+ // We don't want the chevron to appear while we animate. See documentation
+ // in the header for why we do this.
+ suppress_chevron_ = !chevron_->IsVisible();
+
+ // Animate!
+ resize_animation_->Reset();
+ resize_animation_->SetTweenType(SlideAnimation::NONE);
+ resize_animation_->Show();
+ }
+}
+
+void BrowserActionsContainer::BrowserActionRemoved(Extension* extension) {
+ CloseOverflowMenu();
+
+ if (popup_ && popup_->host()->extension() == extension)
+ HidePopup();
+
+ // Before we change anything, determine the number of visible browser actions.
+ size_t visible_actions = VisibleBrowserActions();
+
+ for (std::vector<BrowserActionView*>::iterator iter =
+ browser_action_views_.begin(); iter != browser_action_views_.end();
+ ++iter) {
+ if ((*iter)->button()->extension() == extension) {
+ RemoveChildView(*iter);
+ delete *iter;
+ browser_action_views_.erase(iter);
+
+ // For details on why we do the following see the class comments in the
+ // header.
+
+ // Calculate the target size we'll animate to (end state). This might be
+ // the same size (if the icon we are removing is in the overflow bucket
+ // and there are other icons there). We don't decrement visible_actions
+ // because we want the container to stay the same size (clamping will take
+ // care of shrinking the container if there aren't enough icons to show).
+ animation_target_size_ =
+ ClampToNearestIconCount(IconCountToWidth(visible_actions));
+
+ // Animate!
+ resize_animation_->Reset();
+ resize_animation_->SetTweenType(SlideAnimation::EASE_OUT);
+ resize_animation_->Show();
+ return;
+ }
+ }
+}
+
int BrowserActionsContainer::WidthOfNonIconArea() const {
int chevron_size = (chevron_->IsVisible()) ?
chevron_->GetPreferredSize().width() : 0;
diff --git a/chrome/browser/views/browser_actions_container.h b/chrome/browser/views/browser_actions_container.h
index 3f769b7..9adb6e7 100644
--- a/chrome/browser/views/browser_actions_container.h
+++ b/chrome/browser/views/browser_actions_container.h
@@ -8,6 +8,7 @@
#include <vector>
#include "base/task.h"
+#include "chrome/browser/extensions/extension_toolbar_model.h"
#include "chrome/browser/extensions/image_loading_tracker.h"
#include "chrome/browser/views/browser_bubble.h"
#include "chrome/browser/views/extensions/extension_action_context_menu.h"
@@ -217,7 +218,8 @@ class BrowserActionsContainer
public BrowserBubble::Delegate,
public views::ViewMenuDelegate,
public views::ResizeGripper::ResizeGripperDelegate,
- public AnimationDelegate {
+ public AnimationDelegate,
+ public ExtensionToolbarModel::Observer {
public:
BrowserActionsContainer(Profile* profile, ToolbarView* toolbar);
virtual ~BrowserActionsContainer();
@@ -297,13 +299,9 @@ class BrowserActionsContainer
ExtensionPopup* TestGetPopup() { return popup_; }
private:
- // Adds a browser action view for the extension if it needs one. DCHECK if
- // it has already been added.
- void AddBrowserAction(Extension* extension);
-
- // Removes the browser action view for an extension if it has one. DCHECK if
- // no such view.
- void RemoveBrowserAction(Extension* extension);
+ // ExtensionToolbarModel::Observer implementation.
+ virtual void BrowserActionAdded(Extension* extension, int index);
+ virtual void BrowserActionRemoved(Extension* extension);
// Closes the overflow menu if open.
void CloseOverflowMenu();
@@ -346,6 +344,9 @@ class BrowserActionsContainer
// from browser_action_views_).
BrowserActionButton* popup_button_;
+ // The model that tracks the order of the toolbar icons.
+ ExtensionToolbarModel* model_;
+
// The current size of the container.
gfx::Size container_size_;