542 lines
18 KiB
C++
542 lines
18 KiB
C++
/*
|
|
* Copyright (c) Meta Platforms, Inc. and affiliates.
|
|
*
|
|
* This source code is licensed under the MIT license found in the
|
|
* LICENSE file in the root directory of this source tree.
|
|
*/
|
|
|
|
#include <vector>
|
|
|
|
#include <glog/logging.h>
|
|
#include <gtest/gtest.h>
|
|
|
|
#include <ReactCommon/RuntimeExecutor.h>
|
|
#include <react/renderer/animations/LayoutAnimationDriver.h>
|
|
#include <react/renderer/componentregistry/ComponentDescriptorProvider.h>
|
|
#include <react/renderer/componentregistry/ComponentDescriptorProviderRegistry.h>
|
|
#include <react/renderer/componentregistry/ComponentDescriptorRegistry.h>
|
|
#include <react/renderer/components/root/RootComponentDescriptor.h>
|
|
#include <react/renderer/components/view/ViewComponentDescriptor.h>
|
|
#include <react/renderer/core/PropsParserContext.h>
|
|
#include <react/renderer/mounting/Differentiator.h>
|
|
#include <react/renderer/mounting/ShadowViewMutation.h>
|
|
|
|
#include <react/renderer/mounting/stubs.h>
|
|
#include <react/test_utils/Entropy.h>
|
|
#include <react/test_utils/MockClock.h>
|
|
#include <react/test_utils/shadowTreeGeneration.h>
|
|
|
|
// Uncomment when random test blocks are uncommented below.
|
|
// #include <algorithm>
|
|
// #include <random>
|
|
|
|
MockClock::time_point MockClock::time_ = {};
|
|
|
|
namespace facebook::react {
|
|
|
|
static void testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
uint_fast32_t seed,
|
|
int treeSize,
|
|
int repeats,
|
|
int stages,
|
|
int animation_duration,
|
|
int animation_frames,
|
|
int delay_ms_between_frames,
|
|
int delay_ms_between_stages,
|
|
int delay_ms_between_repeats,
|
|
bool commits_conflicting_mutations = false,
|
|
int final_animation_delay = 0) {
|
|
auto entropy = seed == 0 ? Entropy() : Entropy(seed);
|
|
|
|
auto eventDispatcher = EventDispatcher::Shared{};
|
|
auto contextContainer = std::make_shared<ContextContainer const>();
|
|
auto componentDescriptorParameters =
|
|
ComponentDescriptorParameters{eventDispatcher, contextContainer, nullptr};
|
|
auto viewComponentDescriptor =
|
|
ViewComponentDescriptor(componentDescriptorParameters);
|
|
auto rootComponentDescriptor =
|
|
RootComponentDescriptor(componentDescriptorParameters);
|
|
auto noopEventEmitter =
|
|
std::make_shared<ViewEventEmitter const>(nullptr, -1, eventDispatcher);
|
|
|
|
PropsParserContext parserContext{-1, *contextContainer};
|
|
|
|
// Create a RuntimeExecutor
|
|
RuntimeExecutor runtimeExecutor =
|
|
[](std::function<void(jsi::Runtime &)> const & /*unused*/) {};
|
|
|
|
// Create component descriptor registry for animation driver
|
|
auto providerRegistry =
|
|
std::make_shared<ComponentDescriptorProviderRegistry>();
|
|
auto componentDescriptorRegistry =
|
|
providerRegistry->createComponentDescriptorRegistry(
|
|
componentDescriptorParameters);
|
|
providerRegistry->add(
|
|
concreteComponentDescriptorProvider<ViewComponentDescriptor>());
|
|
|
|
// Create Animation Driver
|
|
auto animationDriver = std::make_shared<LayoutAnimationDriver>(
|
|
runtimeExecutor, contextContainer, nullptr);
|
|
animationDriver->setComponentDescriptorRegistry(componentDescriptorRegistry);
|
|
|
|
// Mock animation timers
|
|
animationDriver->setClockNow([]() {
|
|
return std::chrono::duration_cast<std::chrono::milliseconds>(
|
|
MockClock::now().time_since_epoch())
|
|
.count();
|
|
});
|
|
|
|
auto allNodes = std::vector<ShadowNode::Shared>{};
|
|
|
|
for (int i = 0; i < repeats; i++) {
|
|
allNodes.clear();
|
|
|
|
int surfaceIdInt = 1;
|
|
auto surfaceId = SurfaceId(surfaceIdInt);
|
|
|
|
auto family = rootComponentDescriptor.createFamily(
|
|
{Tag(surfaceIdInt), surfaceId, nullptr}, nullptr);
|
|
|
|
// Creating an initial root shadow node.
|
|
auto emptyRootNode = std::const_pointer_cast<RootShadowNode>(
|
|
std::static_pointer_cast<RootShadowNode const>(
|
|
rootComponentDescriptor.createShadowNode(
|
|
ShadowNodeFragment{RootShadowNode::defaultSharedProps()},
|
|
family)));
|
|
|
|
// Applying size constraints.
|
|
emptyRootNode = emptyRootNode->clone(
|
|
parserContext,
|
|
LayoutConstraints{
|
|
Size{512, 0}, Size{512, std::numeric_limits<Float>::infinity()}},
|
|
LayoutContext{});
|
|
|
|
// Generation of a random tree.
|
|
auto singleRootChildNode =
|
|
generateShadowNodeTree(entropy, viewComponentDescriptor, treeSize);
|
|
|
|
// Injecting a tree into the root node.
|
|
auto currentRootNode = std::static_pointer_cast<RootShadowNode const>(
|
|
emptyRootNode->ShadowNode::clone(ShadowNodeFragment{
|
|
ShadowNodeFragment::propsPlaceholder(),
|
|
std::make_shared<ShadowNode::ListOfShared>(
|
|
ShadowNode::ListOfShared{singleRootChildNode})}));
|
|
|
|
// Building an initial view hierarchy.
|
|
auto viewTree = buildStubViewTreeWithoutUsingDifferentiator(*emptyRootNode);
|
|
viewTree.mutate(
|
|
calculateShadowViewMutations(*emptyRootNode, *currentRootNode));
|
|
|
|
for (int j = 0; j < stages; j++) {
|
|
auto nextRootNode = currentRootNode;
|
|
|
|
// Mutating the tree.
|
|
alterShadowTree(
|
|
entropy,
|
|
nextRootNode,
|
|
{
|
|
&messWithChildren,
|
|
&messWithYogaStyles,
|
|
&messWithLayoutableOnlyFlag,
|
|
});
|
|
|
|
std::vector<LayoutableShadowNode const *> affectedLayoutableNodes{};
|
|
affectedLayoutableNodes.reserve(1024);
|
|
|
|
// Laying out the tree.
|
|
std::const_pointer_cast<RootShadowNode>(nextRootNode)
|
|
->layoutIfNeeded(&affectedLayoutableNodes);
|
|
|
|
nextRootNode->sealRecursive();
|
|
allNodes.push_back(nextRootNode);
|
|
|
|
// Calculating mutations.
|
|
auto originalMutations =
|
|
calculateShadowViewMutations(*currentRootNode, *nextRootNode);
|
|
|
|
// If tree randomization produced no changes in the form of mutations,
|
|
// don't bother trying to animate because this violates a bunch of our
|
|
// assumptions in this test
|
|
if (originalMutations.empty()) {
|
|
continue;
|
|
}
|
|
|
|
// If we only mutated the root... also don't bother
|
|
if (originalMutations.size() == 1 &&
|
|
(originalMutations[0].oldChildShadowView.tag == 1 ||
|
|
originalMutations[0].newChildShadowView.tag == 1)) {
|
|
continue;
|
|
}
|
|
|
|
// Configure animation
|
|
animationDriver->uiManagerDidConfigureNextLayoutAnimation(
|
|
{surfaceId,
|
|
0,
|
|
false,
|
|
{(double)animation_duration,
|
|
{/* Create */ AnimationType::EaseInEaseOut,
|
|
AnimationProperty::Opacity,
|
|
(double)animation_duration,
|
|
0,
|
|
0,
|
|
0},
|
|
{/* Update */ AnimationType::EaseInEaseOut,
|
|
AnimationProperty::ScaleXY,
|
|
(double)animation_duration,
|
|
0,
|
|
0,
|
|
0},
|
|
{/* Delete */ AnimationType::EaseInEaseOut,
|
|
AnimationProperty::Opacity,
|
|
(double)animation_duration,
|
|
0,
|
|
0,
|
|
0}},
|
|
{},
|
|
{},
|
|
{}});
|
|
|
|
// Get mutations for each frame
|
|
for (int k = 0; k < animation_frames + 2; k++) {
|
|
auto mutationsInput = ShadowViewMutation::List{};
|
|
if (k == 0) {
|
|
mutationsInput = originalMutations;
|
|
}
|
|
|
|
if (k != (animation_frames + 1)) {
|
|
EXPECT_TRUE(animationDriver->shouldOverridePullTransaction());
|
|
} else if (!commits_conflicting_mutations) {
|
|
EXPECT_FALSE(animationDriver->shouldOverridePullTransaction());
|
|
}
|
|
|
|
auto telemetry = TransactionTelemetry{};
|
|
telemetry.willLayout();
|
|
telemetry.willCommit();
|
|
telemetry.willDiff();
|
|
|
|
auto transaction = animationDriver->pullTransaction(
|
|
surfaceId, 0, telemetry, mutationsInput);
|
|
|
|
EXPECT_TRUE(transaction.has_value() || k == animation_frames);
|
|
|
|
// We have something to validate.
|
|
if (transaction.has_value()) {
|
|
auto mutations = transaction->getMutations();
|
|
|
|
// Mutating the view tree.
|
|
viewTree.mutate(mutations);
|
|
|
|
// We don't do any validation on this until all animations are
|
|
// finished!
|
|
}
|
|
|
|
MockClock::advance_by(
|
|
std::chrono::milliseconds(delay_ms_between_frames));
|
|
}
|
|
|
|
// After the animation is completed...
|
|
// Build a view tree to compare with.
|
|
// After all the synthetic mutations, at the end of the animation,
|
|
// the mutated and newly-constructed trees should be identical.
|
|
if (!commits_conflicting_mutations) {
|
|
auto rebuiltViewTree =
|
|
buildStubViewTreeWithoutUsingDifferentiator(*nextRootNode);
|
|
|
|
// Comparing the newly built tree with the updated one.
|
|
if (rebuiltViewTree != viewTree) {
|
|
// Something went wrong.
|
|
|
|
LOG(ERROR)
|
|
<< "Entropy seed: " << entropy.getSeed()
|
|
<< ". To see why trees are different, define STUB_VIEW_TREE_VERBOSE and see logging in StubViewTree.cpp.\n";
|
|
|
|
EXPECT_TRUE(false);
|
|
}
|
|
}
|
|
|
|
currentRootNode = nextRootNode;
|
|
|
|
MockClock::advance_by(std::chrono::milliseconds(delay_ms_between_stages));
|
|
}
|
|
|
|
// Flush all remaining animations before validating trees
|
|
if (final_animation_delay > 0) {
|
|
MockClock::advance_by(std::chrono::milliseconds(final_animation_delay));
|
|
|
|
auto telemetry = TransactionTelemetry{};
|
|
telemetry.willLayout();
|
|
telemetry.willCommit();
|
|
telemetry.willDiff();
|
|
|
|
auto transaction =
|
|
animationDriver->pullTransaction(surfaceId, 0, telemetry, {});
|
|
// We have something to validate.
|
|
if (transaction.has_value()) {
|
|
auto mutations = transaction->getMutations();
|
|
|
|
// Mutating the view tree.
|
|
viewTree.mutate(mutations);
|
|
|
|
// We don't do any validation on this until all animations are
|
|
// finished!
|
|
}
|
|
}
|
|
|
|
// After all animations are completed...
|
|
// Build a view tree to compare with.
|
|
// After all the synthetic mutations, at the end of the animation,
|
|
// the mutated and newly-constructed trees should be identical.
|
|
if (commits_conflicting_mutations) {
|
|
auto rebuiltViewTree =
|
|
buildStubViewTreeWithoutUsingDifferentiator(*currentRootNode);
|
|
|
|
// Comparing the newly built tree with the updated one.
|
|
if (rebuiltViewTree != viewTree) {
|
|
// Something went wrong.
|
|
|
|
LOG(ERROR)
|
|
<< "Entropy seed: " << entropy.getSeed()
|
|
<< ". To see why trees are different, define STUB_VIEW_TREE_VERBOSE and see logging in StubViewTree.cpp.\n";
|
|
|
|
EXPECT_TRUE(false);
|
|
}
|
|
}
|
|
|
|
MockClock::advance_by(std::chrono::milliseconds(delay_ms_between_repeats));
|
|
}
|
|
|
|
SUCCEED();
|
|
}
|
|
|
|
} // namespace facebook::react
|
|
|
|
using namespace facebook::react;
|
|
|
|
TEST(
|
|
LayoutAnimationTest,
|
|
stableSmallerTreeFewRepeatsFewStages_NonOverlapping_2029343357) {
|
|
testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
/* seed */ 2029343357, /* working seed found 5-10-2021 */
|
|
/* size */ 128,
|
|
/* repeats */ 128,
|
|
/* stages */ 10,
|
|
/* animation_duration */ 1000,
|
|
/* animation_frames*/ 10,
|
|
/* delay_ms_between_frames */ 100,
|
|
/* delay_ms_between_stages */ 100,
|
|
/* delay_ms_between_repeats */ 2000);
|
|
}
|
|
|
|
TEST(
|
|
LayoutAnimationTest,
|
|
stableSmallerTreeFewRepeatsFewStages_NonOverlapping_3619914559) {
|
|
testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
/* seed */ 3619914559, /* working seed found 5-10-2021 */
|
|
/* size */ 128,
|
|
/* repeats */ 128,
|
|
/* stages */ 10,
|
|
/* animation_duration */ 1000,
|
|
/* animation_frames*/ 10,
|
|
/* delay_ms_between_frames */ 100,
|
|
/* delay_ms_between_stages */ 100,
|
|
/* delay_ms_between_repeats */ 2000);
|
|
}
|
|
|
|
TEST(
|
|
LayoutAnimationTest,
|
|
stableSmallerTreeFewRepeatsFewStages_NonOverlapping_597132284) {
|
|
testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
/* seed */ 597132284, /* failing seed found 5-10-2021 */
|
|
/* size */ 128,
|
|
/* repeats */ 128,
|
|
/* stages */ 10,
|
|
/* animation_duration */ 1000,
|
|
/* animation_frames*/ 10,
|
|
/* delay_ms_between_frames */ 100,
|
|
/* delay_ms_between_stages */ 100,
|
|
/* delay_ms_between_repeats */ 2000);
|
|
}
|
|
|
|
TEST(
|
|
LayoutAnimationTest,
|
|
stableSmallerTreeFewRepeatsFewStages_NonOverlapping_774986518) {
|
|
testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
/* seed */ 774986518, /* failing seed found 5-10-2021 */
|
|
/* size */ 128,
|
|
/* repeats */ 128,
|
|
/* stages */ 10,
|
|
/* animation_duration */ 1000,
|
|
/* animation_frames*/ 10,
|
|
/* delay_ms_between_frames */ 100,
|
|
/* delay_ms_between_stages */ 100,
|
|
/* delay_ms_between_repeats */ 2000);
|
|
}
|
|
|
|
TEST(
|
|
LayoutAnimationTest,
|
|
stableSmallerTreeFewRepeatsFewStages_NonOverlapping_1450614414) {
|
|
testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
/* seed */ 1450614414, /* failing seed found 5-10-2021 */
|
|
/* size */ 128,
|
|
/* repeats */ 128,
|
|
/* stages */ 10,
|
|
/* animation_duration */ 1000,
|
|
/* animation_frames*/ 10,
|
|
/* delay_ms_between_frames */ 100,
|
|
/* delay_ms_between_stages */ 100,
|
|
/* delay_ms_between_repeats */ 2000);
|
|
}
|
|
|
|
TEST(LayoutAnimationTest, stableBiggerTreeFewRepeatsFewStages_NonOverlapping) {
|
|
testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
/* seed */ 2029343357,
|
|
/* size */ 512,
|
|
/* repeats */ 32,
|
|
/* stages */ 10,
|
|
/* animation_duration */ 1000,
|
|
/* animation_frames*/ 10,
|
|
/* delay_ms_between_frames */ 100,
|
|
/* delay_ms_between_stages */ 100,
|
|
/* delay_ms_between_repeats */ 2000);
|
|
}
|
|
|
|
TEST(LayoutAnimationTest, stableBiggerTreeFewRepeatsManyStages_NonOverlapping) {
|
|
testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
/* seed */ 2029343357,
|
|
/* size */ 512,
|
|
/* repeats */ 32,
|
|
/* stages */ 128,
|
|
/* animation_duration */ 1000,
|
|
/* animation_frames*/ 10,
|
|
/* delay_ms_between_frames */ 100,
|
|
/* delay_ms_between_stages */ 100,
|
|
/* delay_ms_between_repeats */ 2000);
|
|
}
|
|
|
|
// You may uncomment this - locally only! - to generate failing seeds.
|
|
// TEST(LayoutAnimationTest, stableSmallerTreeFewRepeatsFewStages_Random) {
|
|
// std::random_device device;
|
|
// for (int i = 0; i < 10; i++) {
|
|
// uint_fast32_t seed = device();
|
|
// LOG(ERROR) << "Seed: " << seed;
|
|
// testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
// /* seed */ seed,
|
|
// /* size */ 128,
|
|
// /* repeats */ 128,
|
|
// /* stages */ 10,
|
|
// /* animation_duration */ 1000,
|
|
// /* animation_frames*/ 10,
|
|
// /* delay_ms_between_frames */ 100,
|
|
// /* delay_ms_between_stages */ 100,
|
|
// /* delay_ms_between_repeats */ 2000);
|
|
// }
|
|
// // Fail if you want output to get seeds
|
|
// LOG(ERROR) << "ALL RUNS SUCCESSFUL";
|
|
// // react_native_assert(false);
|
|
// }
|
|
|
|
//
|
|
// These tests are "overlapping", meaning that mutations will be committed
|
|
// before the previous animation completes.
|
|
//
|
|
|
|
TEST(
|
|
LayoutAnimationTest,
|
|
stableSmallerTreeFewRepeatsFewStages_Overlapping_2029343357) {
|
|
testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
/* seed */ 2029343357,
|
|
/* size */ 128,
|
|
/* repeats */ 128,
|
|
/* stages */ 10,
|
|
/* animation_duration */ 1000,
|
|
/* animation_frames*/ 9, // an animation completes in 10 frames, so this
|
|
// causes conflicts
|
|
/* delay_ms_between_frames */ 100,
|
|
/* delay_ms_between_stages */ 100,
|
|
/* delay_ms_between_repeats */ 2000,
|
|
/* commits_conflicting_mutations */ true,
|
|
/* final_animation_delay */ 10000 + 1);
|
|
}
|
|
|
|
TEST(
|
|
LayoutAnimationTest,
|
|
stableSmallerTreeFewRepeatsFewStages_Overlapping_597132284) {
|
|
testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
/* seed */ 597132284,
|
|
/* size */ 128,
|
|
/* repeats */ 128,
|
|
/* stages */ 10,
|
|
/* animation_duration */ 1000,
|
|
/* animation_frames*/ 9, // an animation completes in 10 frames, so this
|
|
// causes conflicts
|
|
/* delay_ms_between_frames */ 100,
|
|
/* delay_ms_between_stages */ 100,
|
|
/* delay_ms_between_repeats */ 2000,
|
|
/* commits_conflicting_mutations */ true,
|
|
/* final_animation_delay */ 10000 + 1);
|
|
}
|
|
|
|
TEST(
|
|
LayoutAnimationTest,
|
|
stableSmallerTreeFewRepeatsFewStages_Overlapping_ManyConflicts_597132284) {
|
|
testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
/* seed */ 597132284,
|
|
/* size */ 128,
|
|
/* repeats */ 128,
|
|
/* stages */ 50,
|
|
/* animation_duration */ 1000,
|
|
/* animation_frames*/ 5, // an animation completes in 10 frames, so this
|
|
// causes conflicts. We only animate 5 frames,
|
|
// but have 50 stages, so conflicts stack up
|
|
// quickly.
|
|
/* delay_ms_between_frames */ 100,
|
|
/* delay_ms_between_stages */ 100,
|
|
/* delay_ms_between_repeats */ 2000,
|
|
/* commits_conflicting_mutations */ true,
|
|
/* final_animation_delay */ 50000 + 1);
|
|
}
|
|
|
|
TEST(
|
|
LayoutAnimationTest,
|
|
stableBiggerTreeFewRepeatsManyStages_Overlapping_ManyConflicts_2029343357) {
|
|
testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
/* seed */ 2029343357,
|
|
/* size */ 512,
|
|
/* repeats */ 32,
|
|
/* stages */ 128,
|
|
/* animation_duration */ 1000,
|
|
/* animation_frames*/ 10,
|
|
/* delay_ms_between_frames */ 10,
|
|
/* delay_ms_between_stages */ 10,
|
|
/* delay_ms_between_repeats */ 2000,
|
|
/* commits_conflicting_mutations */ true,
|
|
/* final_animation_delay */ (128 * 1000 + 100));
|
|
}
|
|
|
|
// You may uncomment this -
|
|
// locally only !-to generate failing seeds.
|
|
// TEST(
|
|
// LayoutAnimationTest,
|
|
// stableSmallerTreeFewRepeatsFewStages_Overlapping_Random) {
|
|
// std::random_device device;
|
|
// for (int i = 0; i < 10; i++) {
|
|
// uint_fast32_t seed = device();
|
|
// LOG(ERROR) << "Seed: " << seed;
|
|
// testShadowNodeTreeLifeCycleLayoutAnimations(
|
|
// /* seed */ seed,
|
|
// /* size */ 512,
|
|
// /* repeats */ 32,
|
|
// /* stages */ 128,
|
|
// /* animation_duration */ 1000,
|
|
// /* animation_frames*/ 10,
|
|
// /* delay_ms_between_frames */ 10,
|
|
// /* delay_ms_between_stages */ 10,
|
|
// /* delay_ms_between_repeats */ 2000,
|
|
// /* commits_conflicting_mutations */ true,
|
|
// /* final_animation_delay */ (128 * 1000 + 100));
|
|
// }
|
|
// // Fail if you want output to get seeds
|
|
// LOG(ERROR) << "ALL RUNS SUCCESSFUL";
|
|
// // react_native_assert(false);
|
|
// }
|