Skip to content

Commit

Permalink
Fabric: Fixes animations strict weak ordering sorted check failed (#4…
Browse files Browse the repository at this point in the history
…6582)

Summary:
Fixes #46568 . cc cipolleschi

## Changelog:

[IOS] [FIXED] - Fabric: Fixes animations strict weak ordering sorted check failed

Pull Request resolved: #46582

Test Plan:
See issue in  #46568

## Repro steps
- Install Xcode 16.0
- navigate to react-native-github
- yarn install
- cd packages/rn-tester
- bundle install
- RCT_NEW_ARCH_ENABLED=1 bundle exec pod install
open RNTesterPods.xcworkspace to open Xcode

{F1885373361}

Testing with Reproducer from OSS
|  Paper |  Fabric (With Fix) |
|--------|-----------------|
| {F1885395747} | {F1885395870} |

Android - LayoutAnimation (Looks like it has been broken and not working way before this changes.)
 https://pxl.cl/5DGVv

Reviewed By: cipolleschi

Differential Revision: D63399017

Pulled By: realsoelynn

fbshipit-source-id: aaf4ac2884ccca2da7e90a52a8ef10df6ae4fc8a
  • Loading branch information
zhongwuzw authored and cipolleschi committed Oct 7, 2024
1 parent 0e677ed commit f18ed7b
Show file tree
Hide file tree
Showing 2 changed files with 46 additions and 4 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -834,10 +834,7 @@ LayoutAnimationKeyFrameManager::pullTransaction(
finalConflictingMutations.end(),
&shouldFirstComeBeforeSecondMutation);

std::stable_sort(
immediateMutations.begin(),
immediateMutations.end(),
&shouldFirstComeBeforeSecondRemovesOnly);
handleShouldFirstComeBeforeSecondRemovesOnly(immediateMutations);

animation.keyFrames = keyFramesToAnimate;
inflightAnimations_.push_back(std::move(animation));
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,40 @@ static inline bool shouldFirstComeBeforeSecondRemovesOnly(
(lhs.index > rhs.index);
}

static inline void handleShouldFirstComeBeforeSecondRemovesOnly(
ShadowViewMutation::List& list) noexcept {
std::unordered_map<std::string, std::vector<ShadowViewMutation>>
removeMutationsByTag;
ShadowViewMutation::List finalList;
for (auto& mutation : list) {
if (mutation.type == ShadowViewMutation::Type::Remove) {
auto key = std::to_string(mutation.parentShadowView.tag);
removeMutationsByTag[key].push_back(mutation);
} else {
finalList.push_back(mutation);
}
}

if (removeMutationsByTag.size() == 0) {
return;
}

for (auto& mutationsPair : removeMutationsByTag) {
if (mutationsPair.second.size() > 1) {
std::stable_sort(
mutationsPair.second.begin(),
mutationsPair.second.end(),
&shouldFirstComeBeforeSecondRemovesOnly);
}
finalList.insert(
finalList.begin(),
mutationsPair.second.begin(),
mutationsPair.second.end());
}

list = finalList;
}

static inline bool shouldFirstComeBeforeSecondMutation(
const ShadowViewMutation& lhs,
const ShadowViewMutation& rhs) noexcept {
Expand Down Expand Up @@ -55,6 +89,17 @@ static inline bool shouldFirstComeBeforeSecondMutation(
lhs.type == ShadowViewMutation::Type::Insert) {
return false;
}

// Remove comes before Update
if (lhs.type == ShadowViewMutation::Type::Remove &&
rhs.type == ShadowViewMutation::Type::Update) {
return true;
}
if (rhs.type == ShadowViewMutation::Type::Remove &&
lhs.type == ShadowViewMutation::Type::Update) {
return false;
}

} else {
// Make sure that removes on the same level are sorted - highest indices
// must come first.
Expand Down

0 comments on commit f18ed7b

Please sign in to comment.