| // Copyright 2015 The Chromium Authors | 
 | // Use of this source code is governed by a BSD-style license that can be | 
 | // found in the LICENSE file. | 
 |  | 
 | #include "ui/views/animation/ink_drop_highlight.h" | 
 |  | 
 | #include <memory> | 
 | #include <string> | 
 | #include <utility> | 
 |  | 
 | #include "base/functional/bind.h" | 
 | #include "third_party/skia/include/core/SkColor.h" | 
 | #include "ui/compositor/layer.h" | 
 | #include "ui/gfx/animation/animation.h" | 
 | #include "ui/gfx/geometry/insets.h" | 
 | #include "ui/gfx/geometry/size_conversions.h" | 
 | #include "ui/views/animation/animation_builder.h" | 
 | #include "ui/views/animation/ink_drop_highlight_observer.h" | 
 | #include "ui/views/animation/ink_drop_painted_layer_delegates.h" | 
 | #include "ui/views/animation/ink_drop_util.h" | 
 |  | 
 | namespace views { | 
 |  | 
 | namespace { | 
 |  | 
 | // The opacity of the highlight when it is not visible. | 
 | constexpr float kHiddenOpacity = 0.0f; | 
 |  | 
 | }  // namespace | 
 |  | 
 | std::string ToString(InkDropHighlight::AnimationType animation_type) { | 
 |   switch (animation_type) { | 
 |     case InkDropHighlight::AnimationType::kFadeIn: | 
 |       return std::string("FADE_IN"); | 
 |     case InkDropHighlight::AnimationType::kFadeOut: | 
 |       return std::string("FADE_OUT"); | 
 |   } | 
 | } | 
 |  | 
 | InkDropHighlight::InkDropHighlight( | 
 |     const gfx::PointF& center_point, | 
 |     std::unique_ptr<BasePaintedLayerDelegate> layer_delegate) | 
 |     : center_point_(center_point), | 
 |       layer_delegate_(std::move(layer_delegate)), | 
 |       layer_(std::make_unique<ui::Layer>()) { | 
 |   const gfx::RectF painted_bounds = layer_delegate_->GetPaintedBounds(); | 
 |   size_ = painted_bounds.size(); | 
 |  | 
 |   layer_->SetBounds(gfx::ToEnclosingRect(painted_bounds)); | 
 |   layer_->SetFillsBoundsOpaquely(false); | 
 |   layer_->set_delegate(layer_delegate_.get()); | 
 |   layer_->SetVisible(false); | 
 |   layer_->SetMasksToBounds(false); | 
 |   layer_->SetName("InkDropHighlight:layer"); | 
 | } | 
 |  | 
 | InkDropHighlight::InkDropHighlight(const gfx::SizeF& size, | 
 |                                    int corner_radius, | 
 |                                    const gfx::PointF& center_point, | 
 |                                    SkColor color) | 
 |     : InkDropHighlight( | 
 |           center_point, | 
 |           std::make_unique<RoundedRectangleLayerDelegate>(color, | 
 |                                                           size, | 
 |                                                           corner_radius)) { | 
 |   layer_->SetOpacity(visible_opacity_); | 
 | } | 
 |  | 
 | InkDropHighlight::InkDropHighlight(const gfx::Size& size, | 
 |                                    int corner_radius, | 
 |                                    const gfx::PointF& center_point, | 
 |                                    SkColor color) | 
 |     : InkDropHighlight(gfx::SizeF(size), corner_radius, center_point, color) {} | 
 |  | 
 | InkDropHighlight::InkDropHighlight(const gfx::SizeF& size, SkColor base_color) | 
 |     : size_(size), layer_(std::make_unique<ui::Layer>(ui::LAYER_SOLID_COLOR)) { | 
 |   layer_->SetColor(base_color); | 
 |   layer_->SetBounds(gfx::Rect(gfx::ToRoundedSize(size))); | 
 |   layer_->SetVisible(false); | 
 |   layer_->SetMasksToBounds(false); | 
 |   layer_->SetOpacity(visible_opacity_); | 
 |   layer_->SetName("InkDropHighlight:solid_color_layer"); | 
 | } | 
 |  | 
 | InkDropHighlight::~InkDropHighlight() { | 
 |   // Explicitly aborting all the animations ensures all callbacks are invoked | 
 |   // while this instance still exists. | 
 |   animation_abort_handle_.reset(); | 
 | } | 
 |  | 
 | bool InkDropHighlight::IsFadingInOrVisible() const { | 
 |   return last_animation_initiated_was_fade_in_; | 
 | } | 
 |  | 
 | void InkDropHighlight::FadeIn(const base::TimeDelta& duration) { | 
 |   layer_->SetOpacity(kHiddenOpacity); | 
 |   layer_->SetVisible(true); | 
 |   AnimateFade(AnimationType::kFadeIn, duration); | 
 | } | 
 |  | 
 | void InkDropHighlight::FadeOut(const base::TimeDelta& duration) { | 
 |   AnimateFade(AnimationType::kFadeOut, duration); | 
 | } | 
 |  | 
 | test::InkDropHighlightTestApi* InkDropHighlight::GetTestApi() { | 
 |   return nullptr; | 
 | } | 
 |  | 
 | void InkDropHighlight::AnimateFade(AnimationType animation_type, | 
 |                                    const base::TimeDelta& duration) { | 
 |   last_animation_initiated_was_fade_in_ = | 
 |       animation_type == AnimationType::kFadeIn; | 
 |  | 
 |   layer_->SetTransform(CalculateTransform()); | 
 |  | 
 |   const base::TimeDelta effective_duration = | 
 |       gfx::Animation::ShouldRenderRichAnimation() ? duration | 
 |                                                   : base::TimeDelta(); | 
 |   const float opacity = animation_type == AnimationType::kFadeIn | 
 |                             ? visible_opacity_ | 
 |                             : kHiddenOpacity; | 
 |   views::AnimationBuilder builder; | 
 |   if (effective_duration.is_positive()) | 
 |     animation_abort_handle_ = builder.GetAbortHandle(); | 
 |   builder | 
 |       .SetPreemptionStrategy( | 
 |           ui::LayerAnimator::IMMEDIATELY_ANIMATE_TO_NEW_TARGET) | 
 |       .OnStarted(base::BindOnce(&InkDropHighlight::AnimationStartedCallback, | 
 |                                 base::Unretained(this), animation_type)) | 
 |       .OnEnded(base::BindOnce(&InkDropHighlight::AnimationEndedCallback, | 
 |                               base::Unretained(this), animation_type, | 
 |                               InkDropAnimationEndedReason::SUCCESS)) | 
 |       .OnAborted(base::BindOnce(&InkDropHighlight::AnimationEndedCallback, | 
 |                                 base::Unretained(this), animation_type, | 
 |                                 InkDropAnimationEndedReason::PRE_EMPTED)) | 
 |       .Once() | 
 |       .SetDuration(effective_duration) | 
 |       .SetOpacity(layer_.get(), opacity, gfx::Tween::EASE_IN_OUT); | 
 | } | 
 |  | 
 | gfx::Transform InkDropHighlight::CalculateTransform() const { | 
 |   gfx::Transform transform; | 
 |   // No transform needed for a solid color layer. | 
 |   if (!layer_delegate_) | 
 |     return transform; | 
 |  | 
 |   transform.Translate(center_point_.x(), center_point_.y()); | 
 |   gfx::Vector2dF layer_offset = layer_delegate_->GetCenteringOffset(); | 
 |   transform.Translate(-layer_offset.x(), -layer_offset.y()); | 
 |  | 
 |   // Add subpixel correction to the transform. | 
 |   transform.PostConcat( | 
 |       GetTransformSubpixelCorrection(transform, layer_->device_scale_factor())); | 
 |  | 
 |   return transform; | 
 | } | 
 |  | 
 | void InkDropHighlight::AnimationStartedCallback(AnimationType animation_type) { | 
 |   if (observer_) | 
 |     observer_->AnimationStarted(animation_type); | 
 | } | 
 |  | 
 | void InkDropHighlight::AnimationEndedCallback( | 
 |     AnimationType animation_type, | 
 |     InkDropAnimationEndedReason reason) { | 
 |   // AnimationEndedCallback() may be invoked when this is being destroyed and | 
 |   // |layer_| may be null. | 
 |   if (animation_type == AnimationType::kFadeOut && layer_) | 
 |     layer_->SetVisible(false); | 
 |  | 
 |   if (observer_) | 
 |     observer_->AnimationEnded(animation_type, reason); | 
 | } | 
 |  | 
 | }  // namespace views |