Home | History | Annotate | Download | only in infobars
      1 // Copyright (c) 2011 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 "chrome/browser/ui/views/infobars/infobar_container.h"
      6 
      7 #include "chrome/browser/tab_contents/infobar_delegate.h"
      8 #include "chrome/browser/ui/views/infobars/infobar.h"
      9 #include "content/browser/tab_contents/tab_contents.h"
     10 #include "content/common/notification_details.h"
     11 #include "content/common/notification_source.h"
     12 #include "ui/base/animation/slide_animation.h"
     13 
     14 InfoBarContainer::Delegate::~Delegate() {
     15 }
     16 
     17 InfoBarContainer::InfoBarContainer(Delegate* delegate)
     18     : delegate_(delegate),
     19       tab_contents_(NULL),
     20       top_arrow_target_height_(InfoBar::kDefaultArrowTargetHeight) {
     21 }
     22 
     23 InfoBarContainer::~InfoBarContainer() {
     24   // RemoveAllInfoBarsForDestruction() should have already cleared our infobars.
     25   DCHECK(infobars_.empty());
     26 }
     27 
     28 void InfoBarContainer::ChangeTabContents(TabContents* contents) {
     29   registrar_.RemoveAll();
     30 
     31   while (!infobars_.empty()) {
     32     InfoBar* infobar = infobars_.front();
     33     // NULL the container pointer first so that if the infobar is currently
     34     // animating, OnInfoBarAnimated() won't get called; we'll manually trigger
     35     // this once for the whole set of changes below.  This also prevents
     36     // InfoBar::MaybeDelete() from calling RemoveInfoBar() a second time if the
     37     // infobar happens to be at zero height (dunno if this can actually happen).
     38     infobar->set_container(NULL);
     39     RemoveInfoBar(infobar);
     40   }
     41 
     42   tab_contents_ = contents;
     43   if (tab_contents_) {
     44     Source<TabContents> tc_source(tab_contents_);
     45     registrar_.Add(this, NotificationType::TAB_CONTENTS_INFOBAR_ADDED,
     46                    tc_source);
     47     registrar_.Add(this, NotificationType::TAB_CONTENTS_INFOBAR_REMOVED,
     48                    tc_source);
     49     registrar_.Add(this, NotificationType::TAB_CONTENTS_INFOBAR_REPLACED,
     50                    tc_source);
     51 
     52     for (size_t i = 0; i < tab_contents_->infobar_count(); ++i) {
     53       // As when we removed the infobars above, we prevent callbacks to
     54       // OnInfoBarAnimated() for each infobar.
     55       AddInfoBar(tab_contents_->GetInfoBarDelegateAt(i)->CreateInfoBar(), false,
     56                  NO_CALLBACK);
     57     }
     58   }
     59 
     60   // Now that everything is up to date, signal the delegate to re-layout.
     61   OnInfoBarStateChanged(false);
     62 }
     63 
     64 int InfoBarContainer::GetVerticalOverlap(int* total_height) {
     65   // Our |total_height| is the sum of the preferred heights of the InfoBars
     66   // contained within us plus the |vertical_overlap|.
     67   int vertical_overlap = 0;
     68   int next_infobar_y = 0;
     69 
     70   for (InfoBars::iterator i(infobars_.begin()); i != infobars_.end(); ++i) {
     71     InfoBar* infobar = *i;
     72     next_infobar_y -= infobar->arrow_height();
     73     vertical_overlap = std::max(vertical_overlap, -next_infobar_y);
     74     next_infobar_y += infobar->total_height();
     75   }
     76 
     77   if (total_height)
     78     *total_height = next_infobar_y + vertical_overlap;
     79   return vertical_overlap;
     80 }
     81 
     82 void InfoBarContainer::SetMaxTopArrowHeight(int height) {
     83   // Decrease the height by the arrow stroke thickness, which is the separator
     84   // line height, because the infobar arrow target heights are without-stroke.
     85   top_arrow_target_height_ = std::min(
     86       std::max(height - InfoBar::kSeparatorLineHeight, 0),
     87       InfoBar::kMaximumArrowTargetHeight);
     88   UpdateInfoBarArrowTargetHeights();
     89 }
     90 
     91 void InfoBarContainer::OnInfoBarStateChanged(bool is_animating) {
     92   if (delegate_)
     93     delegate_->InfoBarContainerStateChanged(is_animating);
     94 }
     95 
     96 void InfoBarContainer::RemoveDelegate(InfoBarDelegate* delegate) {
     97   tab_contents_->RemoveInfoBar(delegate);
     98 }
     99 
    100 void InfoBarContainer::RemoveInfoBar(InfoBar* infobar) {
    101   InfoBars::iterator infobar_iterator(std::find(infobars_.begin(),
    102                                                 infobars_.end(), infobar));
    103   DCHECK(infobar_iterator != infobars_.end());
    104   PlatformSpecificRemoveInfoBar(infobar);
    105   infobars_.erase(infobar_iterator);
    106 }
    107 
    108 void InfoBarContainer::RemoveAllInfoBarsForDestruction() {
    109   // Before we remove any children, we reset |delegate_|, so that no removals
    110   // will result in us trying to call
    111   // delegate_->InfoBarContainerStateChanged().  This is important because at
    112   // this point |delegate_| may be shutting down, and it's at best unimportant
    113   // and at worst disastrous to call that.
    114   delegate_ = NULL;
    115   ChangeTabContents(NULL);
    116 }
    117 
    118 void InfoBarContainer::Observe(NotificationType type,
    119                                const NotificationSource& source,
    120                                const NotificationDetails& details) {
    121   switch (type.value) {
    122     case NotificationType::TAB_CONTENTS_INFOBAR_ADDED:
    123       AddInfoBar(Details<InfoBarDelegate>(details)->CreateInfoBar(), true,
    124                  WANT_CALLBACK);
    125       break;
    126 
    127     case NotificationType::TAB_CONTENTS_INFOBAR_REMOVED:
    128       RemoveInfoBar(Details<InfoBarDelegate>(details).ptr(), true);
    129       break;
    130 
    131     case NotificationType::TAB_CONTENTS_INFOBAR_REPLACED: {
    132       typedef std::pair<InfoBarDelegate*, InfoBarDelegate*> InfoBarPair;
    133       InfoBarPair* infobar_pair = Details<InfoBarPair>(details).ptr();
    134       RemoveInfoBar(infobar_pair->first, false);
    135       AddInfoBar(infobar_pair->second->CreateInfoBar(), false, WANT_CALLBACK);
    136       break;
    137     }
    138 
    139     default:
    140       NOTREACHED();
    141       break;
    142   }
    143 }
    144 
    145 void InfoBarContainer::RemoveInfoBar(InfoBarDelegate* delegate,
    146                                      bool use_animation) {
    147   // Search for the infobar associated with |delegate|.  We cannot search for
    148   // |delegate| in |tab_contents_|, because an InfoBar remains alive until its
    149   // close animation completes, while the delegate is removed from the tab
    150   // immediately.
    151   for (InfoBars::iterator i(infobars_.begin()); i != infobars_.end(); ++i) {
    152     InfoBar* infobar = *i;
    153     if (infobar->delegate() == delegate) {
    154       // We merely need hide the infobar; it will call back to RemoveInfoBar()
    155       // itself once it's hidden.
    156       infobar->Hide(use_animation);
    157       UpdateInfoBarArrowTargetHeights();
    158       break;
    159     }
    160   }
    161 }
    162 
    163 void InfoBarContainer::AddInfoBar(InfoBar* infobar,
    164                                   bool animate,
    165                                   CallbackStatus callback_status) {
    166   DCHECK(std::find(infobars_.begin(), infobars_.end(), infobar) ==
    167       infobars_.end());
    168   infobars_.push_back(infobar);
    169   UpdateInfoBarArrowTargetHeights();
    170   PlatformSpecificAddInfoBar(infobar);
    171   if (callback_status == WANT_CALLBACK)
    172     infobar->set_container(this);
    173   infobar->Show(animate);
    174   if (callback_status == NO_CALLBACK)
    175     infobar->set_container(this);
    176 }
    177 
    178 void InfoBarContainer::UpdateInfoBarArrowTargetHeights() {
    179   for (size_t i = 0; i < infobars_.size(); ++i)
    180     infobars_[i]->SetArrowTargetHeight(ArrowTargetHeightForInfoBar(i));
    181 }
    182 
    183 int InfoBarContainer::ArrowTargetHeightForInfoBar(size_t infobar_index) const {
    184   if (!delegate_->DrawInfoBarArrows(NULL))
    185     return 0;
    186   if (infobar_index == 0)
    187     return top_arrow_target_height_;
    188   const ui::SlideAnimation* first_infobar_animation =
    189       const_cast<const InfoBar*>(infobars_.front())->animation();
    190   if ((infobar_index > 1) || first_infobar_animation->IsShowing())
    191     return InfoBar::kDefaultArrowTargetHeight;
    192   // When the first infobar is animating closed, we animate the second infobar's
    193   // arrow target height from the default to the top target height.  Note that
    194   // the animation values here are going from 1.0 -> 0.0 as the top bar closes.
    195   return top_arrow_target_height_ + static_cast<int>(
    196       (InfoBar::kDefaultArrowTargetHeight - top_arrow_target_height_) *
    197           first_infobar_animation->GetCurrentValue());
    198 }
    199