React Native предоставляет Animated API который покрывает часть базовых потребностей вашего приложения, однако если требуется создавать сложные и производительные анимации на 60 fps, без сторонних библиотек не обойтись. Сегодня хотел бы рассказать о Reanimated 2 и продемонстрировать возможности библиотеки на примере создания компонента счетчика.
Reanimated
За анимации в React Native отвечает Animated API позволяющий реализовать множество различных сценариев. Однако у него одна проблема - большинство анимаций рассчитывается в JS потоке и только часть может быть перенесена на UI поток с помощью свойства useNativeDriver
Для того обхода этой проблемы инженеры Software Mansion разработали библиотеку Reanimated которая де-факто стала самым популярным решением для работы с анимацией в React Native. Чтобы освободить JS поток от расчета анимаций Reanimated запускает собственный JS поток, который занимается только расчетом анимации. Это накладывает некоторые ограничения на написание кода и вводит 2 концепции Worklets и Shared Values.
Worklets
Ворклеты в терминах Reanimated это небольшие JS функции которые запускаются в отдельном JS потоке для расчета анимаций. Единственное, что нужно чтобы функция стала ворклетом это пометить функцию директивой worklet
function someWorklet(greeting) {
'worklet';
console.log("Hey I'm running on the UI thread");
}
Ворклеты так же можно запустить и в обычном JS. По умолчанию ворклеты запускаются в основном JS потоке. Для того чтобы передать параметры можно воспользоваться параметрами функции или переменными из контекста в котором запускается ворклет. В этом случае эта переменная будет скопирована.
const width = 135.5;
function otherWorklet(greeting) {
'worklet';
console.log(greeting, 'From the UI thread');
console.log('Captured width is', width);
}
Так же внутри ворклета можно вызывать функции из главного потока React Native с помощью функции runOnJS
function callback(text) {
console.log('Running on the RN thread', text);
}
function someWorklet() {
'worklet';
console.log("I'm on UI but can call methods from the RN thread");
runOnJS(callback)('can pass arguments too');
}
На практике писать ворклеты придется не часто, так как библиотека предоставляет большое количество хуков для анимации которые под капотом уже используют ворклеты.
Shared values
Для того чтобы реализовать анимации одних только ворклетов недостаточно и второй важной концепцией библиотеки являются Shared values. Они решают несколько проблем, во-первых предоставляют механизм совместного использования памяти для нескольких JS потоков. Вторая функция это реактивность, Reanimated составляет граф зависимостей между Shared values и ворклетами в которых они используются. Далее если ворклет будет запускаться только при начальной инициализации или когда изменяется shared value связанное с этим ворклетом.
import Animated, {
useSharedValue,
useAnimatedStyle,
} from 'react-native-reanimated';
function Box() {
const offset = useSharedValue(0);
const animatedStyles = useAnimatedStyle(() => {
return {
transform: [{ translateX: offset.value }],
};
});
return (
<>
<Animated.View style={[styles.box, animatedStyles]} />
<Button onPress={() => (offset.value = Math.random() * 255)} title="Move" />
</>
);
}
useAnimatedStyle
под капотом запускает ворклет и каждый раз когда меняется значение offset.value
стили пересчитываются. Третьей функцией является управление анимацией. В примере выше View
будет менять положение моментально, но если мы хотим это анимировать можно использовать для этого специальные функции которые будут менять значение в shared values
с помощью time-based анимаций. Например, если добавить функцию withSpring
<Button
onPress={() => {
offset.value = withSpring(Math.random() * 255);
}}
title="Move"
/>
Реализация компонента с помощью Reanimated
Без учета стилей компонент получился в 70 строк. Под катом полный код и давайте реазберем его реализацию.
Реализация компонента
import {useState} from 'react';
import {Gesture, GestureDetector} from 'react-native-gesture-handler'
import MaskedView from '@react-native-masked-view/masked-view';
import Animated, {
withSequence,
useAnimatedStyle,
useSharedValue,
withSpring,
runOnJS,
} from 'react-native-reanimated';
import {StyleSheet, Text, View} from "react-native";
const Counter = ({initValue = 0}) => {
const [value, setValue] = useState(initValue)
const offset = useSharedValue(0);
const minusScale = useSharedValue(1);
const plusScale = useSharedValue(1);
const ballAnimatedStyles = useAnimatedStyle(() => ({
transform: [{translateX: offset.value }]
}));
const minusAnimatedStyles = useAnimatedStyle(() => ({
transform: [{scale: minusScale.value }]
}));
const plusAnimatedStyles = useAnimatedStyle(() => ({
transform: [{scale: plusScale.value }]
}));
const tap = Gesture.Pan().onUpdate(e => {
if (e.translationX > 0 && e.translationX < 75) {
offset.value = e.translationX
} else if (value > 0 && e.translationX > -75 && e.translationX < 0) {
offset.value = e.translationX
} else if (value === 0 && e.translationX > -25 && e.translationX < 0) {
offset.value = e.translationX
}
}).onEnd((e) => {
offset.value = withSpring(0)
if (e.translationX > 30) {
plusScale.value = withSequence(withSpring(1.2), withSpring(1))
runOnJS(setValue)(value + 1)
}
if (e.translationX < -30 && value > 0) {
minusScale.value = withSequence(withSpring(1.2), withSpring(1))
runOnJS(setValue)(value - 1)
}
if (e.translationX < 0 && value === 0) {
minusScale.value = withSequence(withSpring(1.2), withSpring(1))
}
})
return <MaskedView
style={styles.maskContainer}
androidRenderingMode="software"
maskElement={<View style={styles.maskElementContainer} />}
>
<View style={[styles.controlContainer]}>
<Animated.View style={[styles.leftContainer, minusAnimatedStyles]}>
<Text style={[styles.minusText, ...value <= 0 ? [styles.disabledText] : [] ]}>-</Text>
</Animated.View>
<GestureDetector gesture={tap}>
<Animated.View style={[styles.ballContainer, ballAnimatedStyles]}>
<Text style={styles.counterText}>{value}</Text>
</Animated.View>
</GestureDetector>
<Animated.View style={[styles.rightContainer, plusAnimatedStyles]}>
<Text style={styles.signText}>+</Text>
</Animated.View>
</View>
</MaskedView>
}
const styles = StyleSheet.create({
maskContainer: {
width: 180,
height: 80,
},
maskElementContainer: {
width: 180,
height: 80,
borderRadius: 40,
backgroundColor: '#7775ff',
zIndex: 0
},
controlContainer: {
flexDirection: 'row',
alignItems: 'center',
justifyContent: 'space-between',
width: 180,
height: 80,
borderRadius: 40,
backgroundColor: '#7775ff',
},
ballContainer: {
alignItems: 'center',
justifyContent: 'center',
width: 80,
height: 80,
borderRadius: 40,
backgroundColor: '#fff',
zIndex: 20
},
signText: {
fontSize: 36,
color: '#fff',
fontFamily: 'monospace',
},
minusText: {
fontSize: 36,
color: '#fff',
fontFamily: 'monospace',
},
disabledText: {
color: 'rgba(256, 256, 256, 0.4)',
},
leftContainer: {
flex: 1,
alignItems: 'center',
},
rightContainer: {
flex: 1,
alignItems: 'center',
},
counterText: {
fontSize: 36,
color: '#7775ff',
fontFamily: 'monospace',
fontWeight: 'bold'
},
});
export default Counter
Помимо Reanimated используются Gesture Handler и библиотека для создания маски Masked View.
<MaskedView
style={styles.maskContainer}
androidRenderingMode="software"
maskElement={<View style={styles.maskElementContainer} />}
>
<View style={[styles.controlContainer]}>
<Animated.View style={[styles.leftContainer, minusAnimatedStyles]}>
<Text style={[styles.minusText, ...value <= 0 ? [styles.disabledText] : [] ]}>-</Text>
</Animated.View>
<GestureDetector gesture={tap}>
<Animated.View style={[styles.ballContainer, ballAnimatedStyles]}>
<Text style={styles.counterText}>{value}</Text>
</Animated.View>
</GestureDetector>
<Animated.View style={[styles.rightContainer, plusAnimatedStyles]}>
<Text style={styles.signText}>+</Text>
</Animated.View>
</View>
</MaskedView>
Компонент состоит из контейнера который представляет собой MaskedView
и трех Animated.View
которые будем анимировать - это значки плюса и минуса и контролер со значением внутри. Для обработки жеста обернем контролер с помощью GestureDetector
.
// ...
const offset = useSharedValue(0);
const ballAnimatedStyles = useAnimatedStyle(() => ({
transform: [{translateX: offset.value }]
}));
// ...
const tap = Gesture.Pan().onUpdate(e => {
if (e.translationX > 0 && e.translationX < 75) {
offset.value = e.translationX
} else if (value > 0 && e.translationX > -75 && e.translationX < 0) {
offset.value = e.translationX
}
}).onEnd((e) => {
offset.value = withSpring(0)
if (e.translationX > 30) {
runOnJS(setValue)(value + 1)
}
if (e.translationX < -30 && value > 0) {
runOnJS(setValue)(value - 1)
}
})
// ...
<GestureDetector gesture={tap}>
<Animated.View style={[styles.ballContainer, ballAnimatedStyles]}>
<Text style={styles.counterText}>{value}</Text>
</Animated.View>
</GestureDetector>
// ...
Для начала анимируем движение контролера. Для этого понадобится создать shared value
с помощью которой будем анимировать движение контролера и добавить обработчик события. Так же используем хук useAnimatedStyle
создающий ассоциацию между значением shared value
и свойствами View с которыми его ассоциируем. В нашем случае это трансформация по оси X которую передаем во View
с помощью свойства style
. В обработчике касания используется 2 метода onUpdate и onEnd
. В onUpdate
фиксируем изменение положения по оси X и соответственно меняем значение shared value
ассоциированного с useAnimatedStyle
ворклетом. Обработчик onEnd
срабатывает при завершении жеста, и с помощью функции withSpring возвращаем значение shared value
в 0 а так же увеличиваем или уменьшаем значение нашего счетчика. Важно что делаем это с помощью функции runOnJS
так как GestureDetector
так же запускается на UI потоке.
const minusScale = useSharedValue(1);
const plusScale = useSharedValue(1);
// ...
const minusAnimatedStyles = useAnimatedStyle(() => ({
transform: [{scale: minusScale.value }]
}));
const plusAnimatedStyles = useAnimatedStyle(() => ({
transform: [{scale: plusScale.value }]
}));
const tap = Gesture.Pan().onUpdate(e => {
// .....
if (value === 0 && e.translationX > -25 && e.translationX < 0) {
offset.value = e.translationX
}
}).onEnd((e) => {
// ...
if (e.translationX > 30) {
plusScale.value = withSequence(withSpring(1.2), withSpring(1))
// ...
}
if (e.translationX < -30 && value > 0) {
minusScale.value = withSequence(withSpring(1.2), withSpring(1))
// ...
}
if (e.translationX < 0 && value === 0) {
minusScale.value = withSequence(withSpring(1.2), withSpring(1))
}
})
// ...
<Animated.View style={[styles.leftContainer, minusAnimatedStyles]}>
<Text style={[styles.minusText, ...value <= 0 ? [styles.disabledText] : [] ]}>-</Text>
</Animated.View>
// ...
<Animated.View style={[styles.rightContainer, plusAnimatedStyles]}>
<Text style={styles.signText}>+</Text>
</Animated.View>
Для анимации значков плюс и минус так же заведем 2 shared values
и хуки для контроля масштаба связанные с этими значениями. Изменяем значения после завершения жеста и используем для этого функцию withSequence
которая выполняет несколько анимаций, в нашем случае увеличив, а потом вернув масштаб.
Дальше нам остается только разместить компонент в прилоежнии:
import 'react-native-gesture-handler'
import {StyleSheet} from 'react-native';
import {StatusBar} from 'expo-status-bar';
import {GestureHandlerRootView} from 'react-native-gesture-handler'
import Counter from "./src/Component";
export default function App() {
return (
<GestureHandlerRootView style={styles.container}>
<StatusBar style="auto"/>
<Counter initValue={0} />
</GestureHandlerRootView>
);
}
const styles = StyleSheet.create({
container: {
flex: 1,
backgroundColor: '#5a56fd',
alignItems: 'center',
justifyContent: 'center',
},
});
И протестировать его работу:
Видно, что анимация никак не влияет на JS или UI потоки и это одно из основных преимуществ Reanimated.
В статье я постарался рассмотреть лишь базовые понятия этой библиотеки и если она вас заинтересовала, то можно посмотреть реализацию более сложных анимаций на ютуб каналах: Catalin Miron, Reactiive и William Candillon. Правда William Candillon последнее время сосредоточился на разработке React Native Skia о которой я писал в прошлой статье.
olololeg
Mой концепт, btw. Использовал FramerJS.
kirill3333 Автор
жаль ссылка на FramerJS не работает, ее можно оживить?
olololeg
ага, нашел одну из итераций на Дропбоксе. Стили отвалились, но идея работает.