Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(354)

Unified Diff: services/view_manager/animation_runner_unittest.cc

Issue 772893004: Adds constants and runner for animations (Closed) Base URL: https://github.com/domokit/mojo.git@master
Patch Set: Merge Created 6 years ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
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
« no previous file with comments | « services/view_manager/animation_runner_observer.h ('k') | services/view_manager/scheduled_animation_group.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698