Restyle как новый стандарт для создания UI в React Native
- пятница, 29 сентября 2023 г. в 12:52:53
Меня зовут Павленко Виталий, я Team Lead в команде UI-kit в Профи.
В какой-то момент мы хотели решить несколько проблем с UI-kit в наших React Native приложениях:
UI-kit был слишком медленный. И здесь важно уточнить, что он был сделан с использованием styled-components.
В UI-kit не было возможности темизации, потому что токены импортировались как модуль в каждом компоненте. И как следствие, в приложении невозможно было сделать темную тему, потому что при смене токенов оно бы просто не перерендерилось.
Спойлер. Restyle решает обе проблемы, а еще дает дополнительные классные плюшки, о которых мы обязательно поговорим дальше в статье.
The Restyle library provides a type-enforced system for building UI components in React Native with TypeScript. It's a library for building UI libraries, with themability as the core focus.
Restyle предоставляет набор инструментов, которые позволят строить UI продуктов на основе токенов. Особенно очень хорошо такой подход залетает, когда в продукте есть Дизайн Система и этот набор токенов действительно определен.
Одно из основных преимуществ библиотеки Restyle по сравнению с той же styled-system, в том, что она не использует styled-components под капотом. В основе Restyle используются абстрактные стили StyleSheet, поэтому ожидается более высокая производительность.
Что мы понимаем под токенами? Это цвета, отступы, радиусы и так далее – все строгие ограничения, касающиеся стилей компонентов.
У нас будет несколько групп токенов:
{
"profiBrand": "rgba(224, 25, 53, 1)",
"error500": "rgba(224, 25, 53, 1)",
"error300": "rgba(231, 78, 99, 1)",
"error100": "rgba(255, 242, 244, 1)",
"warning500": "rgba(255, 184, 0, 1)",
"warning100": "rgba(255, 246, 221, 1)",
"success500": "rgba(46, 181, 24, 1)",
"success100": "rgba(234, 248, 232, 1)",
"g100": "rgba(255, 255, 255, 1)",
"g200": "rgba(244, 245, 248, 1)",
"g300": "rgba(208, 211, 221, 1)",
"g400": "rgba(164, 166, 178, 1)",
"g500": "rgba(24, 24, 24, 1)",
}
{
"1": 4,
"2": 8,
"3": 12,
"4": 16,
"5": 20,
"6": 24,
"7": 28,
"8": 32,
"9": 36,
"10": 40
}
{
"xxs": 4,
"xs": 8,
"s": 12,
"m": 18,
"l": 20,
"xl": 24,
"xxl": 32
}
{
"headingXL": {
"fontSize": 40,
"lineHeight": 44,
"letterSpacing": -0.4
},
"headingL": {
"fontSize": 36,
"lineHeight": 40,
"letterSpacing": -0.36
},
"headingM": {
"fontSize": 28,
"lineHeight": 32,
"letterSpacing": -0.28
},
"headingS": {
"fontSize": 22,
"lineHeight": 26,
"letterSpacing": 0
},
"headingXS": {
"fontSize": 17,
"lineHeight": 22,
"letterSpacing": 0
},
"bodyL": {
"fontSize": 17,
"lineHeight": 24,
"letterSpacing": -0.34
},
"bodyM": {
"fontSize": 15,
"lineHeight": 22,
"letterSpacing": -0.3
},
"bodyS": {
"fontSize": 13,
"lineHeight": 16,
"letterSpacing": -0.26
}
}
{
"s": 0,
"m": 360
}
А дальше эти токены нужно расшарить на всё приложение, для этого библиотека Restyle предоставляет ThemeProvider
:
import {ThemeProvider, createTheme} from '@shopify/restyle';
import {
colors,
spacing,
breakpoints,
textVariants,
borderRadii
} from './tokens';
import {Box,Text} from './';
const theme = createTheme({
colors,
spacing,
breakpoints,
textVariants,
borderRadii,
});
export const App = () => {
<ThemeProvider theme={theme}>
<Box
flex={1}
background="g200"
margin={2}
>
<Text variant="bodyM" color="g500">Привет!</Text>
</Box>
</ThemeProvider>
}
Компоненты Box
и Text
являются основными строительными блоками для UI. Box
создается на основе View
и нужен для оберток, Text
нужен для текстовых элементов соответственно. Они создаются один раз, и дальше их просто можно переиспользовать.
import {createBox, createText} from '@shopify/restyle';
export const Box = createBox();
export const Text = createText();
export const myComponent = () => {
return (
<Box
flex={1}
background="g200"
margin={2}
>
<Text variant="bodyM" color="g500">Привет!</Text>
</Box>
)
};
Text
имеет свой собственный проп variant
, где он как раз принимает значение токена типографики textVariant
.
Нет необходимости выносить стили отдельно или создавать отельные файлы. Стили можно добавлять прям в компоненте, используя Restyle компоненты Box
и Text
.
Да, все правильно, мы будем писать стили прям в JSX. А почему это хорошо?
Не нужно создавать бесконечные названия блоков (Container, Wrapper и т.д)
Мы сразу видим всю картину целиком, за что отвечает каждый блок
Стилей не так уж и много обычно
Мы все равно часто добавляем стили в JSX, будет все однообразно
Styled обертки сильно ухудшают производительность
Но все эти плюсы будут работать, если вы будете декомпозировать сложные компоненты на небольшие модули. В полотне верстки это будет выглядеть плохо.
Цвета можно пробрасывать во все цветовые пропсы, используя ключи из темы (например g200
, g300
, g400
).
backgroundColor [bg]
color
shadowColor
textShadowColor
borderColor
В квадратных скобках сокращенный вариант пропса
Посмотрим на пример, как это может использоваться:
<Box
backgroundColor="g300"
borderColor="g400"
>
<Text
variant="bodyS"
color="g200"
>
Any text
</Text>
</Box>
Все отступы тоже можно пробрасывать прям в компоненты, используя специальные пропсы для этого. В нашем случае они добавляют отступы, кратные 4px
(так было решено в нашей ДС).
margin [m], marginTop [mt], marginRight [mr], marginBottom [mb], marginLeft [ml], marginStart [ms], marginEnd[me], marginHorizontal [mx], marginVertical [my], padding [p], paddingTop [pt], paddingRight [pr], paddingBottom [pb], paddingLeft [pl], paddingStart [ps], paddingEnd [pe], paddingHorizontal [px], paddingVertical [py], gap [g], rowGap [rG], columnGap [cG]
В квадратных скобках сокращенный вариант пропса
// будет добавлен marginTop=12px и padding=8px
<Box
marginTop={3}
padding={2}
/>
Радиус тоже берется из темы и задается следующим образом:
<Box
borderColor="g400"
borderStyle="solid"
borderWidth={1}
borderRadius="xs"
/>
Через свойство style
можно пробросить и обычные стили. Таким образом можно добавить разные исключения, например, если нужен отступ некратный 4 или цвет не из ДС.
// добавится отступ 5px и кастомный цвет
<Box style={{ padding: 5, backgroundColor: '#ccc' }} />
В нашей ДС определены два размера девайсов, что позволяет указывать разные стили следующим образом:
// s: от 0px до 360px
// m: больше 360px
// для s девайсов будет 4px отступ, а для m девайсов 8px
<Box marginTop={{ s: 1, m: 2 }} />
При помощи такой конструкции можно пробрасывать любые свойства, не только отступы.
Если написать просто marginTop={1}
, то стили применятся для всех устройств (s
и m
).
Если нам нужно написать какую-то логику в компоненте, опираясь на размеры девайсов, то можно использовать хук useResponsiveProp()
:
import {useResponsiveProp} from '@shopify/restyle';
export const MyComponent = () => {
const value = useResponsiveProp({
s: 'Сейчас S устройство',
m: 'Сейчса M устройство'
});
return <Text>{value}</Text>;
};
Нам не нужно импортировать нужные токены из темы, можно нужно просто достать их используя хук useTheme()
.
import {useTheme} from '@shopify/restyle';
export const MyComponent = () => {
const {colors: {g300}} = useTheme();
return <OutsideComponent color={g300}>Привет!</OutsideComponent>
};
Это может пригодиться, если нам нужно пробросить именно значение цвета в сторонний компонент.
Мы произвели исследование по производительности 3 вариантов и получили интересные результаты!
Для теста был создан следующий компонент:
Компонент содержал 100 строк
В каждой строке был номер, созданный шрифтом из ДС
В каждой строке было 20 блоков, с радиусом из ДС и рандомным цветом из ДС
Проводилось 10 замеров рендера для каждого варианта, затем для результата бралось среднее значение.
Измерялось время рендера следующим образом:
export const Component = () => {
const start = new Date().getTime();
useEffect(() => {
const renderTime = new Date().getTime() - start;
console.log(renderTime);
}, []);
return (
...
);
};
Вариант UI | Время рендера (милисекунды) |
StyleSheet | 408 |
Restyle | 499 |
Styled-components | 632 |
Оказалось, что Restyle на порядок быстрее Styled-components, но естественно немного медленнее вообще нативного решения, потому что содержит дополнительные обертки и парсер атрибутов в стили.
Что мы получаем от Restyle:
Удобные “строительные блоки” Box
и Text
Удобная работа с токенами ДС через атрибуты
Все токены достаются через хуки из Restyle контекста
Удобочитаемые композиции верстки сразу со стилями
Понятная и однозначная концепция для построения UI
Эта библиотека точно заслуживает внимания. Не вижу причин не брать ее на вооружение прям сейчас! Спасибо за внимание 🙂