// 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_started_animation.h" #include "chrome/browser/tab_contents/tab_contents.h" #include "chrome/common/notification_service.h" #include "chrome/common/resource_bundle.h" #include "chrome/views/widget_win.h" #include "grit/theme_resources.h" // How long to spend moving downwards and fading out after waiting. static const int kMoveTimeMs = 600; // The animation framerate. static const int kFrameRateHz = 60; // What fraction of the frame height to move downward from the frame center. // Note that setting this greater than 0.5 will mean moving past the bottom of // the frame. static const double kMoveFraction = 1.0 / 3.0; DownloadStartedAnimation::DownloadStartedAnimation(TabContents* tab_contents) : Animation(kMoveTimeMs, kFrameRateHz, NULL), popup_(NULL), tab_contents_(tab_contents) { static SkBitmap* kDownloadImage = NULL; if (!kDownloadImage) { kDownloadImage = ResourceBundle::GetSharedInstance().GetBitmapNamed( IDR_DOWNLOAD_ANIMATION_BEGIN); } // If we're too small to show the download image, then don't bother - // the shelf will be enough. tab_contents_->GetContainerBounds(&tab_contents_bounds_); if (tab_contents_bounds_.height() < kDownloadImage->height()) return; NotificationService::current()->AddObserver( this, NotificationType::TAB_CONTENTS_HIDDEN, Source(tab_contents_)); NotificationService::current()->AddObserver( this, NotificationType::TAB_CONTENTS_DESTROYED, Source(tab_contents_)); SetImage(kDownloadImage); gfx::Rect rc(0, 0, 0, 0); popup_ = new views::WidgetWin; popup_->set_window_style(WS_POPUP); popup_->set_window_ex_style(WS_EX_LAYERED | WS_EX_TOOLWINDOW | WS_EX_TRANSPARENT); popup_->SetLayeredAlpha(0x00); popup_->Init(tab_contents_->GetNativeView(), rc, false); popup_->SetContentsView(this); Reposition(); popup_->Show(); Start(); } void DownloadStartedAnimation::Reposition() { if (!tab_contents_) return; // Align the image with the bottom left of the web contents (so that it // points to the newly created download). gfx::Size size = GetPreferredSize(); popup_->MoveWindow( tab_contents_bounds_.x(), static_cast(tab_contents_bounds_.bottom() - size.height() - size.height() * (1 - GetCurrentValue())), size.width(), size.height()); } void DownloadStartedAnimation::Close() { if (!tab_contents_) return; NotificationService::current()->RemoveObserver( this, NotificationType::TAB_CONTENTS_HIDDEN, Source(tab_contents_)); NotificationService::current()->RemoveObserver( this, NotificationType::TAB_CONTENTS_DESTROYED, Source(tab_contents_)); tab_contents_ = NULL; popup_->Close(); } void DownloadStartedAnimation::AnimateToState(double state) { if (state >= 1.0) { Close(); } else { Reposition(); // Start at zero, peak halfway and end at zero. double opacity = std::min(1.0 - pow(GetCurrentValue() - 0.5, 2) * 4.0, static_cast(1.0)); popup_->SetLayeredAlpha( static_cast(opacity * 255.0)); SchedulePaint(); // Reposition() calls MoveWindow() which never picks up // alpha changes, so we need to force a paint. } } void DownloadStartedAnimation::Observe(NotificationType type, const NotificationSource& source, const NotificationDetails& details) { Close(); }