// Copyright (c) 2006-2008 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/views/download_shelf_view.h" #include #include "base/logging.h" #include "chrome/browser/browser.h" #include "chrome/browser/download/download_item_model.h" #include "chrome/browser/download/download_manager.h" #include "chrome/browser/tab_contents/navigation_entry.h" #include "chrome/browser/tab_contents/tab_contents.h" #include "chrome/browser/views/download_item_view.h" #include "chrome/browser/views/download_tab_view.h" #include "chrome/common/gfx/chrome_canvas.h" #include "chrome/common/l10n_util.h" #include "chrome/common/resource_bundle.h" #include "chrome/views/background.h" #include "chrome/views/button.h" #include "chrome/views/image_view.h" #include "grit/generated_resources.h" #include "grit/theme_resources.h" // Max number of download views we'll contain. Any time a view is added and // we already have this many download views, one is removed. static const int kMaxDownloadViews = 15; // Padding from left edge and first download view. static const int kLeftPadding = 2; // Padding from right edge and close button/show downloads link. static const int kRightPadding = 10; // Padding between the show all link and close button. static const int kCloseAndLinkPadding = 14; // Padding between the download views. static const int kDownloadPadding = 10; // Padding between the top/bottom and the content. static const int kTopBottomPadding = 2; // Padding between the icon and 'show all downloads' link static const int kDownloadsTitlePadding = 4; // Default background color for the shelf. static const SkColor kBackgroundColor = SkColorSetRGB(230, 237, 244); // Border color. static const SkColor kBorderColor = SkColorSetRGB(214, 214, 214); // New download item animation speed in milliseconds. static const int kNewItemAnimationDurationMs = 800; // Shelf show/hide speed. static const int kShelfAnimationDurationMs = 120; namespace { // Sets size->width() to view's preferred width + size->width().s // Sets size->height() to the max of the view's preferred height and // size->height(); void AdjustSize(views::View* view, gfx::Size* size) { gfx::Size view_preferred = view->GetPreferredSize(); size->Enlarge(view_preferred.width(), 0); size->set_height(std::max(view_preferred.height(), size->height())); } int CenterPosition(int size, int target_size) { return std::max((target_size - size) / 2, kTopBottomPadding); } } // namespace DownloadShelfView::DownloadShelfView(TabContents* tab_contents) : tab_contents_(tab_contents) { Init(); } void DownloadShelfView::Init() { ResourceBundle &rb = ResourceBundle::GetSharedInstance(); arrow_image_ = new views::ImageView(); arrow_image_->SetImage(rb.GetBitmapNamed(IDR_DOWNLOADS_FAVICON)); AddChildView(arrow_image_); show_all_view_ = new views::Link(l10n_util::GetString(IDS_SHOW_ALL_DOWNLOADS)); show_all_view_->SetController(this); AddChildView(show_all_view_); close_button_ = new views::Button(); close_button_->SetImage(views::Button::BS_NORMAL, rb.GetBitmapNamed(IDR_CLOSE_BAR)); close_button_->SetImage(views::Button::BS_HOT, rb.GetBitmapNamed(IDR_CLOSE_BAR_H)); close_button_->SetImage(views::Button::BS_PUSHED, rb.GetBitmapNamed(IDR_CLOSE_BAR_P)); close_button_->SetListener(this, 0); AddChildView(close_button_); set_background(views::Background::CreateSolidBackground(kBackgroundColor)); new_item_animation_.reset(new SlideAnimation(this)); new_item_animation_->SetSlideDuration(kNewItemAnimationDurationMs); shelf_animation_.reset(new SlideAnimation(this)); shelf_animation_->SetSlideDuration(kShelfAnimationDurationMs); shelf_animation_->Show(); } void DownloadShelfView::AddDownloadView(View* view) { shelf_animation_->Show(); DCHECK(view); download_views_.push_back(view); AddChildView(view); if (download_views_.size() > kMaxDownloadViews) RemoveDownloadView(*download_views_.begin()); new_item_animation_->Reset(); new_item_animation_->Show(); } void DownloadShelfView::ChangeTabContents(TabContents* old_contents, TabContents* new_contents) { DCHECK(old_contents == tab_contents_); tab_contents_ = new_contents; } void DownloadShelfView::AddDownload(DownloadItem* download) { DownloadItemView* view = new DownloadItemView( download, this, new DownloadItemModel(download)); AddDownloadView(view); } void DownloadShelfView::RemoveDownloadView(View* view) { DCHECK(view); std::vector::iterator i = find(download_views_.begin(), download_views_.end(), view); DCHECK(i != download_views_.end()); download_views_.erase(i); RemoveChildView(view); delete view; if (download_views_.empty()) tab_contents_->SetDownloadShelfVisible(false); Layout(); SchedulePaint(); } void DownloadShelfView::Paint(ChromeCanvas* canvas) { PaintBackground(canvas); PaintBorder(canvas); } void DownloadShelfView::PaintBorder(ChromeCanvas* canvas) { canvas->FillRectInt(kBorderColor, 0, 0, width(), 1); } gfx::Size DownloadShelfView::GetPreferredSize() { gfx::Size prefsize(kRightPadding + kLeftPadding + kCloseAndLinkPadding, 0); AdjustSize(close_button_, &prefsize); AdjustSize(show_all_view_, &prefsize); // Add one download view to the preferred size. if (download_views_.size() > 0) { AdjustSize(*download_views_.begin(), &prefsize); prefsize.Enlarge(kDownloadPadding, 0); } prefsize.Enlarge(0, kTopBottomPadding + kTopBottomPadding); if (shelf_animation_->IsAnimating()) { prefsize.set_height(static_cast( static_cast(prefsize.height()) * shelf_animation_->GetCurrentValue())); } return prefsize; } void DownloadShelfView::AnimationProgressed(const Animation *animation) { if (animation == new_item_animation_.get()) { Layout(); SchedulePaint(); } else if (animation == shelf_animation_.get()) { // Force a re-layout of the parent, which will call back into // GetPreferredSize, where we will do our animation. In the case where the // animation is hiding, we do a full resize - the fast resizing would // otherwise leave blank white areas where the shelf was and where the // user's eye is. Thankfully bottom-resizing is a lot faster than // top-resizing. tab_contents_->ToolbarSizeChanged(shelf_animation_->IsShowing()); } } void DownloadShelfView::AnimationEnded(const Animation *animation) { if (animation == shelf_animation_.get()) { tab_contents_->SetDownloadShelfVisible(shelf_animation_->IsShowing()); } } void DownloadShelfView::Layout() { // When the download shelf is not visible it is not parented to anything, // which means it is not safe to lay out the controls, so we return early. // Otherwise, we can have problems when for example the user switches to // another tab (that doesn't have a download shelf) _before_ the download // has started and we'll crash when calling SetVisible() below because // the NativeControlContainer ctor tries to use the Container. if (!GetWidget()) return; // Let our base class layout our child views views::View::Layout(); gfx::Size image_size = arrow_image_->GetPreferredSize(); gfx::Size close_button_size = close_button_->GetPreferredSize(); gfx::Size show_all_size = show_all_view_->GetPreferredSize(); int max_download_x = std::max(0, width() - kRightPadding - close_button_size.width() - kCloseAndLinkPadding - show_all_size.width() - image_size.width() - kDownloadPadding); int next_x = max_download_x + kDownloadPadding; // Align vertically with show_all_view_. arrow_image_->SetBounds(next_x, CenterPosition(show_all_size.height(), height()), image_size.width(), image_size.height()); next_x += image_size.width() + kDownloadsTitlePadding; show_all_view_->SetBounds(next_x, CenterPosition(show_all_size.height(), height()), show_all_size.width(), show_all_size.height()); next_x += show_all_size.width() + kCloseAndLinkPadding; close_button_->SetBounds(next_x, CenterPosition(close_button_size.height(), height()), close_button_size.width(), close_button_size.height()); next_x = kLeftPadding; std::vector::reverse_iterator ri; for (ri = download_views_.rbegin(); ri != download_views_.rend(); ++ri) { gfx::Size view_size = (*ri)->GetPreferredSize(); int x = next_x; // Figure out width of item. int item_width = view_size.width(); if (new_item_animation_->IsAnimating() && ri == download_views_.rbegin()) { item_width = static_cast(static_cast(view_size.width()) * new_item_animation_->GetCurrentValue()); } next_x += (item_width + kDownloadPadding); // Make sure our item can be contained within the shelf. if (next_x < max_download_x) { (*ri)->SetVisible(true); (*ri)->SetBounds(x, CenterPosition(view_size.height(), height()), item_width, view_size.height()); } else { (*ri)->SetVisible(false); } } } // Open the download page. void DownloadShelfView::LinkActivated(views::Link* source, int event_flags) { int index; NavigationController* controller = tab_contents_->controller(); Browser* browser = Browser::GetBrowserForController(controller, &index); DCHECK(browser); browser->ShowDownloadsTab(); } void DownloadShelfView::ButtonPressed(views::BaseButton* button) { shelf_animation_->Hide(); } bool DownloadShelfView::IsShowing() const { return shelf_animation_->IsShowing(); }