| OLD | NEW |
| (Empty) |
| 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/tab_contents/tab_contents_wrapper.h" | |
| 9 #include "chrome/browser/ui/views/infobars/infobar.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(TabContentsWrapper* 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_->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 < 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 } | |
| OLD | NEW |