Index: services/view_manager/animation_runner_unittest.cc |
diff --git a/services/view_manager/animation_runner_unittest.cc b/services/view_manager/animation_runner_unittest.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..15d16c4d174dffec7e7da521b150c1ce92f6a0e5 |
--- /dev/null |
+++ b/services/view_manager/animation_runner_unittest.cc |
@@ -0,0 +1,505 @@ |
+// Copyright 2014 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 "services/view_manager/animation_runner.h" |
+ |
+#include "base/strings/stringprintf.h" |
+#include "mojo/converters/geometry/geometry_type_converters.h" |
+#include "mojo/services/public/interfaces/view_manager/view_manager_constants.mojom.h" |
+#include "services/view_manager/animation_runner_observer.h" |
+#include "services/view_manager/scheduled_animation_group.h" |
+#include "services/view_manager/server_view.h" |
+#include "services/view_manager/test_server_view_delegate.h" |
+#include "testing/gtest/include/gtest/gtest.h" |
+ |
+namespace mojo { |
+namespace service { |
+namespace { |
+ |
+class TestAnimationRunnerObserver : public AnimationRunnerObserver { |
+ public: |
+ TestAnimationRunnerObserver() {} |
+ ~TestAnimationRunnerObserver() override {} |
+ |
+ std::vector<std::string>* changes() { return &changes_; } |
+ std::vector<uint32_t>* change_ids() { return &change_ids_; } |
+ |
+ void clear_changes() { |
+ changes_.clear(); |
+ change_ids_.clear(); |
+ } |
+ |
+ // AnimationRunnerDelgate: |
+ void OnAnimationScheduled(const ServerView* view, uint32_t id) override { |
+ change_ids_.push_back(id); |
+ changes_.push_back(base::StringPrintf( |
+ "scheduled view=%d,%d", view->id().connection_id, view->id().view_id)); |
+ } |
+ void OnAnimationDone(const ServerView* view, uint32_t id) override { |
+ change_ids_.push_back(id); |
+ changes_.push_back(base::StringPrintf( |
+ "done view=%d,%d", view->id().connection_id, view->id().view_id)); |
+ } |
+ void OnAnimationInterrupted(const ServerView* view, uint32_t id) override { |
+ change_ids_.push_back(id); |
+ changes_.push_back(base::StringPrintf( |
+ "interruped view=%d,%d", view->id().connection_id, view->id().view_id)); |
+ } |
+ void OnAnimationCanceled(const ServerView* view, uint32_t id) override { |
+ change_ids_.push_back(id); |
+ changes_.push_back(base::StringPrintf( |
+ "canceled view=%d,%d", view->id().connection_id, view->id().view_id)); |
+ } |
+ |
+ private: |
+ std::vector<uint32_t> change_ids_; |
+ std::vector<std::string> changes_; |
+ |
+ DISALLOW_COPY_AND_ASSIGN(TestAnimationRunnerObserver); |
+}; |
+ |
+} // namespace |
+ |
+class AnimationRunnerTest : public testing::Test { |
+ public: |
+ AnimationRunnerTest() |
+ : initial_time_(base::TimeTicks::Now()), runner_(initial_time_) { |
+ runner_.AddObserver(&runner_observer_); |
+ } |
+ ~AnimationRunnerTest() override { runner_.RemoveObserver(&runner_observer_); } |
+ |
+ protected: |
+ const base::TimeTicks initial_time_; |
+ TestAnimationRunnerObserver runner_observer_; |
+ AnimationRunner runner_; |
+ |
+ private: |
+ DISALLOW_COPY_AND_ASSIGN(AnimationRunnerTest); |
+}; |
+ |
+// Opacity from 1 to .5 over 1000. |
+TEST_F(AnimationRunnerTest, SingleProperty) { |
+ TestServerViewDelegate view_delegate; |
+ ServerView view(&view_delegate, ViewId()); |
+ |
+ AnimationGroup group; |
+ group.view_id = ViewIdToTransportId(view.id()); |
+ group.sequences.push_back(AnimationSequence::New()); |
+ AnimationSequence& sequence = *(group.sequences[0]); |
+ sequence.cycle_count = 1; |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element = *(sequence.elements[0]); |
+ element.property = ANIMATION_PROPERTY_OPACITY; |
+ element.duration = 1000; |
+ element.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element.target_value = AnimationValue::New(); |
+ element.target_value->float_value = .5; |
+ |
+ const uint32_t animation_id = runner_.Schedule(&view, group); |
+ |
+ ASSERT_EQ(1u, runner_observer_.changes()->size()); |
+ EXPECT_EQ("scheduled view=0,0", runner_observer_.changes()->at(0)); |
+ EXPECT_EQ(animation_id, runner_observer_.change_ids()->at(0)); |
+ runner_observer_.clear_changes(); |
+ |
+ EXPECT_TRUE(runner_.HasAnimations()); |
+ |
+ // Opacity should still be 1 (the initial value). |
+ EXPECT_EQ(1.f, view.opacity()); |
+ |
+ // Animate half way. |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(500)); |
+ |
+ EXPECT_EQ(.75f, view.opacity()); |
+ EXPECT_TRUE(runner_observer_.changes()->empty()); |
+ |
+ // Run well past the end. Value should progress to end and delegate should |
+ // be notified. |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromSeconds(10)); |
+ EXPECT_EQ(.5f, view.opacity()); |
+ |
+ ASSERT_EQ(1u, runner_observer_.changes()->size()); |
+ EXPECT_EQ("done view=0,0", runner_observer_.changes()->at(0)); |
+ EXPECT_EQ(animation_id, runner_observer_.change_ids()->at(0)); |
+ |
+ EXPECT_FALSE(runner_.HasAnimations()); |
+} |
+ |
+// Opacity from 1 to .5, followed by transform from identity to 2x,3x. |
+TEST_F(AnimationRunnerTest, TwoPropertiesInSequence) { |
+ TestServerViewDelegate view_delegate; |
+ ServerView view(&view_delegate, ViewId()); |
+ |
+ AnimationGroup group; |
+ group.view_id = ViewIdToTransportId(view.id()); |
+ group.sequences.push_back(AnimationSequence::New()); |
+ AnimationSequence& sequence = *(group.sequences[0]); |
+ sequence.cycle_count = 1; |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element = *(sequence.elements[0]); |
+ element.property = ANIMATION_PROPERTY_OPACITY; |
+ element.duration = 1000; |
+ element.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element.target_value = AnimationValue::New(); |
+ element.target_value->float_value = .5; |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element2 = *(sequence.elements[1]); |
+ element2.property = ANIMATION_PROPERTY_TRANSFORM; |
+ element2.duration = 2000; |
+ element2.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element2.target_value = AnimationValue::New(); |
+ gfx::Transform done_transform; |
+ done_transform.Scale(2, 4); |
+ element2.target_value->transform = Transform::From(done_transform); |
+ |
+ const uint32_t animation_id = runner_.Schedule(&view, group); |
+ runner_observer_.clear_changes(); |
+ |
+ // Nothing in the view should have changed yet. |
+ EXPECT_EQ(1.f, view.opacity()); |
+ EXPECT_TRUE(view.transform().IsIdentity()); |
+ |
+ // Animate half way from through opacity animation. |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(500)); |
+ |
+ EXPECT_EQ(.75f, view.opacity()); |
+ EXPECT_TRUE(view.transform().IsIdentity()); |
+ |
+ // Finish first element (opacity). |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(1000)); |
+ EXPECT_EQ(.5f, view.opacity()); |
+ EXPECT_TRUE(view.transform().IsIdentity()); |
+ |
+ // Half way through second (transform). |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(2000)); |
+ EXPECT_EQ(.5f, view.opacity()); |
+ gfx::Transform half_way_transform; |
+ half_way_transform.Scale(1.5, 2.5); |
+ EXPECT_EQ(half_way_transform, view.transform()); |
+ |
+ EXPECT_TRUE(runner_observer_.changes()->empty()); |
+ |
+ // To end. |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(3500)); |
+ EXPECT_EQ(.5f, view.opacity()); |
+ EXPECT_EQ(done_transform, view.transform()); |
+ |
+ ASSERT_EQ(1u, runner_observer_.changes()->size()); |
+ EXPECT_EQ("done view=0,0", runner_observer_.changes()->at(0)); |
+ EXPECT_EQ(animation_id, runner_observer_.change_ids()->at(0)); |
+} |
+ |
+// Opacity from .5 to 1 over 1000, transform to 2x,3x over 500. |
+TEST_F(AnimationRunnerTest, TwoPropertiesInParallel) { |
+ TestServerViewDelegate view_delegate; |
+ ServerView view(&view_delegate, ViewId(1, 1)); |
+ |
+ AnimationGroup group; |
+ group.view_id = ViewIdToTransportId(view.id()); |
+ group.sequences.push_back(AnimationSequence::New()); |
+ AnimationSequence& sequence = *(group.sequences[0]); |
+ sequence.cycle_count = 1; |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element = *(sequence.elements[0]); |
+ element.property = ANIMATION_PROPERTY_OPACITY; |
+ element.duration = 1000; |
+ element.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element.start_value = AnimationValue::New(); |
+ element.start_value->float_value = .5; |
+ element.target_value = AnimationValue::New(); |
+ element.target_value->float_value = 1; |
+ group.sequences.push_back(AnimationSequence::New()); |
+ AnimationSequence& sequence2 = *(group.sequences[1]); |
+ sequence2.cycle_count = 1; |
+ sequence2.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element2 = *(sequence2.elements[0]); |
+ element2.property = ANIMATION_PROPERTY_TRANSFORM; |
+ element2.duration = 500; |
+ element2.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element2.target_value = AnimationValue::New(); |
+ gfx::Transform done_transform; |
+ done_transform.Scale(2, 4); |
+ element2.target_value->transform = Transform::From(done_transform); |
+ |
+ const uint32_t animation_id = runner_.Schedule(&view, group); |
+ |
+ runner_observer_.clear_changes(); |
+ |
+ // Nothing in the view should have changed yet. |
+ EXPECT_EQ(1.f, view.opacity()); |
+ EXPECT_TRUE(view.transform().IsIdentity()); |
+ |
+ // Animate to 250, which is 1/4 way through opacity and half way through |
+ // transform. |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(250)); |
+ |
+ EXPECT_EQ(.625f, view.opacity()); |
+ gfx::Transform half_way_transform; |
+ half_way_transform.Scale(1.5, 2.5); |
+ EXPECT_EQ(half_way_transform, view.transform()); |
+ |
+ // Animate to 500, which is 1/2 way through opacity and transform done. |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(500)); |
+ EXPECT_EQ(.75f, view.opacity()); |
+ EXPECT_EQ(done_transform, view.transform()); |
+ |
+ // Animate to 750, which is 3/4 way through opacity and transform done. |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(750)); |
+ EXPECT_EQ(.875f, view.opacity()); |
+ EXPECT_EQ(done_transform, view.transform()); |
+ |
+ EXPECT_TRUE(runner_observer_.changes()->empty()); |
+ |
+ // To end. |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(3500)); |
+ EXPECT_EQ(1.f, view.opacity()); |
+ EXPECT_EQ(done_transform, view.transform()); |
+ |
+ ASSERT_EQ(1u, runner_observer_.changes()->size()); |
+ EXPECT_EQ("done view=1,1", runner_observer_.changes()->at(0)); |
+ EXPECT_EQ(animation_id, runner_observer_.change_ids()->at(0)); |
+} |
+ |
+// Opacity from .5 to 1 over 1000, pause for 500, 1 to .5 over 500, with a cycle |
+// count of 3. |
+TEST_F(AnimationRunnerTest, Cycles) { |
+ TestServerViewDelegate view_delegate; |
+ ServerView view(&view_delegate, ViewId(1, 2)); |
+ |
+ view.SetOpacity(.5f); |
+ |
+ AnimationGroup group; |
+ group.view_id = ViewIdToTransportId(view.id()); |
+ group.sequences.push_back(AnimationSequence::New()); |
+ AnimationSequence& sequence = *(group.sequences[0]); |
+ sequence.cycle_count = 3; |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element = *(sequence.elements[0]); |
+ element.property = ANIMATION_PROPERTY_OPACITY; |
+ element.duration = 1000; |
+ element.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element.target_value = AnimationValue::New(); |
+ element.target_value->float_value = 1; |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element2 = *(sequence.elements[1]); |
+ element2.property = ANIMATION_PROPERTY_NONE; |
+ element2.duration = 500; |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element3 = *(sequence.elements[2]); |
+ element3.property = ANIMATION_PROPERTY_OPACITY; |
+ element3.duration = 500; |
+ element3.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element3.target_value = AnimationValue::New(); |
+ element3.target_value->float_value = .5f; |
+ |
+ runner_.Schedule(&view, group); |
+ runner_observer_.clear_changes(); |
+ |
+ // Nothing in the view should have changed yet. |
+ EXPECT_EQ(.5f, view.opacity()); |
+ |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(500)); |
+ EXPECT_EQ(.75f, view.opacity()); |
+ |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(1250)); |
+ EXPECT_EQ(1.f, view.opacity()); |
+ |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(1750)); |
+ EXPECT_EQ(.75f, view.opacity()); |
+ |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(2500)); |
+ EXPECT_EQ(.75f, view.opacity()); |
+ |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(3250)); |
+ EXPECT_EQ(1.f, view.opacity()); |
+ |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(3750)); |
+ EXPECT_EQ(.75f, view.opacity()); |
+ |
+ // Animate to the end. |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(6500)); |
+ EXPECT_EQ(.5f, view.opacity()); |
+ |
+ ASSERT_EQ(1u, runner_observer_.changes()->size()); |
+ EXPECT_EQ("done view=1,2", runner_observer_.changes()->at(0)); |
+} |
+ |
+// Verifies scheduling the same view twice sends an interrupt. |
+TEST_F(AnimationRunnerTest, ScheduleTwice) { |
+ TestServerViewDelegate view_delegate; |
+ ServerView view(&view_delegate, ViewId(1, 2)); |
+ |
+ AnimationGroup group; |
+ group.view_id = ViewIdToTransportId(view.id()); |
+ group.sequences.push_back(AnimationSequence::New()); |
+ AnimationSequence& sequence = *(group.sequences[0]); |
+ sequence.cycle_count = 1; |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element = *(sequence.elements[0]); |
+ element.property = ANIMATION_PROPERTY_OPACITY; |
+ element.duration = 1000; |
+ element.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element.target_value = AnimationValue::New(); |
+ element.target_value->float_value = .5f; |
+ |
+ const uint32_t animation_id = runner_.Schedule(&view, group); |
+ runner_observer_.clear_changes(); |
+ |
+ // Animate half way. |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(500)); |
+ |
+ EXPECT_EQ(.75f, view.opacity()); |
+ EXPECT_TRUE(runner_observer_.changes()->empty()); |
+ |
+ // Schedule again. We should get an interrupt, but opacity shouldn't change. |
+ const uint32_t animation2_id = runner_.Schedule(&view, group); |
+ |
+ // Id should have changed. |
+ EXPECT_NE(animation_id, animation2_id); |
+ |
+ EXPECT_EQ(nullptr, runner_.GetViewForAnimation(animation_id)); |
+ EXPECT_EQ(&view, runner_.GetViewForAnimation(animation2_id)); |
+ |
+ EXPECT_EQ(.75f, view.opacity()); |
+ EXPECT_EQ(2u, runner_observer_.changes()->size()); |
+ EXPECT_EQ("interruped view=1,2", runner_observer_.changes()->at(0)); |
+ EXPECT_EQ(animation_id, runner_observer_.change_ids()->at(0)); |
+ EXPECT_EQ("scheduled view=1,2", runner_observer_.changes()->at(1)); |
+ EXPECT_EQ(animation2_id, runner_observer_.change_ids()->at(1)); |
+ runner_observer_.clear_changes(); |
+ |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(1000)); |
+ EXPECT_EQ(.625f, view.opacity()); |
+ EXPECT_TRUE(runner_observer_.changes()->empty()); |
+ |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(2000)); |
+ EXPECT_EQ(.5f, view.opacity()); |
+ EXPECT_EQ(1u, runner_observer_.changes()->size()); |
+ EXPECT_EQ("done view=1,2", runner_observer_.changes()->at(0)); |
+ EXPECT_EQ(animation2_id, runner_observer_.change_ids()->at(0)); |
+} |
+ |
+// Verifies Remove() works. |
+TEST_F(AnimationRunnerTest, CancelAnimationForView) { |
+ // Create an animation and advance it part way. |
+ TestServerViewDelegate view_delegate; |
+ ServerView view(&view_delegate, ViewId()); |
+ AnimationGroup group; |
+ group.view_id = ViewIdToTransportId(view.id()); |
+ group.sequences.push_back(AnimationSequence::New()); |
+ AnimationSequence& sequence = *(group.sequences[0]); |
+ sequence.cycle_count = 1; |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element = *(sequence.elements[0]); |
+ element.property = ANIMATION_PROPERTY_OPACITY; |
+ element.duration = 1000; |
+ element.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element.target_value = AnimationValue::New(); |
+ element.target_value->float_value = .5; |
+ |
+ const uint32_t animation_id = runner_.Schedule(&view, group); |
+ runner_observer_.clear_changes(); |
+ EXPECT_EQ(&view, runner_.GetViewForAnimation(animation_id)); |
+ |
+ EXPECT_TRUE(runner_.HasAnimations()); |
+ |
+ // Animate half way. |
+ runner_.Tick(initial_time_ + base::TimeDelta::FromMicroseconds(500)); |
+ EXPECT_EQ(.75f, view.opacity()); |
+ EXPECT_TRUE(runner_observer_.changes()->empty()); |
+ |
+ // Cancel the animation. |
+ runner_.CancelAnimationForView(&view); |
+ |
+ EXPECT_FALSE(runner_.HasAnimations()); |
+ EXPECT_EQ(nullptr, runner_.GetViewForAnimation(animation_id)); |
+ |
+ EXPECT_EQ(.75f, view.opacity()); |
+ |
+ EXPECT_EQ(1u, runner_observer_.changes()->size()); |
+ EXPECT_EQ("canceled view=0,0", runner_observer_.changes()->at(0)); |
+ EXPECT_EQ(animation_id, runner_observer_.change_ids()->at(0)); |
+} |
+ |
+// Verifies a tick with a very large delta and a sequence that repeats forever |
+// doesn't take a long time. |
+TEST_F(AnimationRunnerTest, InfiniteRepeatWithHugeGap) { |
+ TestServerViewDelegate view_delegate; |
+ ServerView view(&view_delegate, ViewId(1, 2)); |
+ |
+ view.SetOpacity(.5f); |
+ |
+ AnimationGroup group; |
+ group.view_id = ViewIdToTransportId(view.id()); |
+ group.sequences.push_back(AnimationSequence::New()); |
+ AnimationSequence& sequence = *(group.sequences[0]); |
+ sequence.cycle_count = 0; // Repeats forever. |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element = *(sequence.elements[0]); |
+ element.property = ANIMATION_PROPERTY_OPACITY; |
+ element.duration = 500; |
+ element.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element.target_value = AnimationValue::New(); |
+ element.target_value->float_value = 1; |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element2 = *(sequence.elements[1]); |
+ element2.property = ANIMATION_PROPERTY_OPACITY; |
+ element2.duration = 500; |
+ element2.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element2.target_value = AnimationValue::New(); |
+ element2.target_value->float_value = .5f; |
+ |
+ runner_.Schedule(&view, group); |
+ runner_observer_.clear_changes(); |
+ |
+ runner_.Tick(initial_time_ + |
+ base::TimeDelta::FromMicroseconds(1000000000750)); |
+ |
+ EXPECT_EQ(.75f, view.opacity()); |
+ |
+ ASSERT_EQ(0u, runner_observer_.changes()->size()); |
+} |
+ |
+// Verifies a second schedule sets any properties that are no longer animating |
+// to their final value. |
+TEST_F(AnimationRunnerTest, RescheduleSetsPropertiesToFinalValue) { |
+ TestServerViewDelegate view_delegate; |
+ ServerView view(&view_delegate, ViewId()); |
+ |
+ AnimationGroup group; |
+ group.view_id = ViewIdToTransportId(view.id()); |
+ group.sequences.push_back(AnimationSequence::New()); |
+ AnimationSequence& sequence = *(group.sequences[0]); |
+ sequence.cycle_count = 1; |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element = *(sequence.elements[0]); |
+ element.property = ANIMATION_PROPERTY_OPACITY; |
+ element.duration = 1000; |
+ element.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element.target_value = AnimationValue::New(); |
+ element.target_value->float_value = .5; |
+ sequence.elements.push_back(AnimationElement::New()); |
+ AnimationElement& element2 = *(sequence.elements[1]); |
+ element2.property = ANIMATION_PROPERTY_TRANSFORM; |
+ element2.duration = 500; |
+ element2.tween_type = ANIMATION_TWEEN_TYPE_LINEAR; |
+ element2.target_value = AnimationValue::New(); |
+ gfx::Transform done_transform; |
+ done_transform.Scale(2, 4); |
+ element2.target_value->transform = Transform::From(done_transform); |
+ runner_.Schedule(&view, group); |
+ |
+ // Schedule() again, this time without animating opacity. |
+ element.property = ANIMATION_PROPERTY_NONE; |
+ runner_.Schedule(&view, group); |
+ |
+ // Opacity should go to final value. |
+ EXPECT_EQ(.5f, view.opacity()); |
+ // Transform shouldn't have changed since newly scheduled animation also has |
+ // transform in it. |
+ EXPECT_TRUE(view.transform().IsIdentity()); |
+} |
+ |
+} // namespace service |
+} // namespace mojo |