Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Simulate skew on Android #42676

Open
wants to merge 4 commits into
base: main
Choose a base branch
from

Conversation

piaskowyk
Copy link
Contributor

@piaskowyk piaskowyk commented Jan 26, 2024

Summary:

This PR aims to address the issue of lacking a direct API for setting skew on native views in Android.

Why does this problem exist?

The lack of an API on Android to directly set skew on native views is the reason behind this problem. It is not a problem specific to React Native itself, but rather an Android platform limitation. Android restricts direct manipulation of transformations. Skew can only be applied via public methods like setRotation, setScale, or setTranslation.

How is this problem solved?

While we cannot directly set the fields in the transformation matrix responsible for skew, we can employ an alternative matrix that produces an equivalent result. The idea behind this PR is to utilize a 3D rotation without perspective to simulate the skew effect. Let's examine an example involving a wall's block with the number 3.

3D rotation with perspective

Screenshot 2024-01-26 at 10 22 17

3D rotation without perspective

Screenshot 2024-01-26 at 10 24 06

By applying the appropriate 3D rotation without perspective, it is possible to achieve the desired skew effect.

Mathematical background

Skew transformation 2D

The skew transformation is an affine 2D matrix.
image
image

Rotation transformation 3D

image image

Scale transformation 3D

image

Matrix multiplication

It is possible to use only Rotation and Scale transformation to achieve Skew 2D transformation.
image
image

image image

Skew is only 2D transformation so we only need to consider the 2D matrix when simulating the skew effect.

SkewX
image
SkewY
image

The only thing you need to do is solve the formulas and calculate the rotation and scale values.

Solution

The matrix has an infinite number of solutions, so we can assume that one of the rotation axes has an arbitrary value to simplify the calculations.

SkewX
image
SkewY
image

Limitations

It is worth noting, that this transformation is not completely equivalent to skew. The matrix that is created by those 3d rotations (and scaling) is equal to the skew matrix only on its 2x2 sub-matrix. The other values in this matrix are however not the same as if they would be in the affine matrix for the skew transformation. This implies that this transformation will not interact well with other 3d transformations, since those additional values in the matrix, will be taken into account when multiplying with other matrices, creating a completely different matrix for 3d vectors. However when 2d transformations are used, this will behave correctly, since 2d vectors won't interact with those values during multiplication. We think that this behaviour is beneficial in comparison to the current situation.
image

Changelog:

[ANDROID][FIXED] Skew transformation

Test Plan:

skewX before skewX after
skewX_before.mov
skewX_after.mov
skewY before skewY after
skewY_before.mov
skewY_after.mov

Example code:

code
import React from 'react';
import { Text, View, StyleSheet, TextInput } from 'react-native';
import Slider from '@react-native-community/slider';
import Animated, { 
  useAnimatedProps, 
  useAnimatedStyle, 
  useSharedValue
} from 'react-native-reanimated';

Animated.addWhitelistedNativeProps({ text: true });
const AnimatedTextInput = Animated.createAnimatedComponent(TextInput);

function App() {
  const skew = useSharedValue(0);
  const style = useAnimatedStyle(() => ({
    transform: [
      { skewX: `${skew.value}deg` },
    ],
  }));
  const animatedProps = useAnimatedProps(() => {
    const skewStr = Math.round(skew.value).toString();
    return { text: skewStr + '°', defaultValue: skewStr + '°' };
  });
  return (
    <View style={{
      flex: 1,
      justifyContent: 'center',
      alignItems: 'center',
      backgroundColor: '#fafafa',
    }}>
      <Animated.View style={[{
        width: 200, 
        height: 200, 
        margin: 50,
        backgroundColor: '#78c9af',
        alignItems: 'center',
        justifyContent: 'center',
      }, style]}>
        <Text style={{fontSize: 45, color: '#001a72'}}>
          SkewX
        </Text>
        <AnimatedTextInput
          style={{fontSize: 45, color: '#001a72'}}
          defaultValue={skew.value.toString() + '°'}
          value={skew.value.toString() + '°'}
          animatedProps={animatedProps}
        />
      </Animated.View>
      <Slider
        style={styles.slider}
        minimumValue={-45}
        maximumValue={45}
        value={0}
        onValueChange={(value) => {
          skew.value = value;
        }}
      />
    </View>
  );
}

const styles = StyleSheet.create({
  container: {
    flex: 1,
    flexDirection: 'column',
    backgroundColor: '#232736',
    padding: 20,
  },
  slider: {
    height: 40,
    width: 300,
  },
});

export default App;

Co-authored with @bartlomiejbloniarz

Co-authored-by: Bartłomiej Błoniarz [email protected]

@facebook-github-bot facebook-github-bot added CLA Signed This label is managed by the Facebook bot. Authors need to sign the CLA before a PR can be reviewed. p: Software Mansion Partner: Software Mansion Partner Shared with Meta Applied via automation to indicate that an Issue or Pull Request has been shared with the team. labels Jan 26, 2024
@analysis-bot
Copy link

analysis-bot commented Jan 26, 2024

Platform Engine Arch Size (bytes) Diff
android hermes arm64-v8a 19,559,075 +8
android hermes armeabi-v7a n/a --
android hermes x86 n/a --
android hermes x86_64 n/a --
android jsc arm64-v8a 22,912,759 +9
android jsc armeabi-v7a n/a --
android jsc x86 n/a --
android jsc x86_64 n/a --

Base commit: ea3a714
Branch: main

@mrousavy
Copy link
Contributor

This is dope! Nice work @piaskowyk 🔥

KusStar added a commit to KusStar/react-native that referenced this pull request Jan 27, 2024
@NickGerleman
Copy link
Contributor

This is really neat 🙂.

Not having played much around this area before, something I noticed when reading was that Android Canvas exposed by overriding onDraw gives more flexibility here, allowing setting arbitrary matrix/skew in the drawing list (unlike RenderNode or View).

This seems kind of appealing, to not need to be constrained to the higher level view APIs, to simulate skew as different transforms, but I don't know if the API works, or results in any de-optimizations.

Curious if this is an approach that has been tried before.

@javache
Copy link
Member

javache commented Mar 14, 2024

This would be an alternative solution to #38494

@piaskowyk
Copy link
Contributor Author

@javache Do you require any additional information regarding this implementation from me? Or any way that I can help you. I aim to facilitate the merging process for you (of course if you want to merge it 😅)

@Wgga
Copy link

Wgga commented May 19, 2024

How to call after modifying the MatrixMathHelper.java file? Do you have the complete code?

@Wgga
Copy link

Wgga commented May 28, 2024

“修改MatrixMathHelper.java文件后调用”是什么意思?你能详细说明一下吗?

After modifying the MatrixMathHelper.java file, I don't know how to use it

@react-native-bot
Copy link
Collaborator

This PR is stale because it has been open 180 days with no activity. Remove stale label or comment or this will be closed in 7 days.

@react-native-bot react-native-bot added the Stale There has been a lack of activity on this issue and it may be closed soon. label Nov 25, 2024
@react-native-bot
Copy link
Collaborator

This PR is stale because it has been open 180 days with no activity. Remove stale label or comment or this will be closed in 7 days.

@piaskowyk
Copy link
Contributor Author

@javache, sorry for pinging 🙏 I just wanted to ask if there's a chance to merge this, or should we move the implementation to Reanimated?

@react-native-bot react-native-bot removed the Stale There has been a lack of activity on this issue and it may be closed soon. label Nov 30, 2024
@mateoguzmana
Copy link
Contributor

This is awesome!

If the new calculations concern any performance, perhaps it would be an idea to add a new Gradle property or feature flag to allow React Native developers to opt in to these changes, until there is an alternative or better solution (if there is).

(Fixes #27649)

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
CLA Signed This label is managed by the Facebook bot. Authors need to sign the CLA before a PR can be reviewed. p: Software Mansion Partner: Software Mansion Partner Shared with Meta Applied via automation to indicate that an Issue or Pull Request has been shared with the team.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

9 participants