В этой статье я расскажу о четырех библиотеках, которые сделают вашу жизнь проще. По крайней мере, я на это надеюсь
1) Nanoid
Ссылка на github
Это маленькая библиотека, которая имеет всего одну функцию — генерация уникального id. Она может быть полезна в массе случаев, как любой другой генератор случайных последовательностей символов. Неоспоримые плюсы: простота и крошечный размер — 143 байта.
Конечно, если вы используете, к примеру, lodash, то вы можете использовать метод uniqueId().
Использование максимально просто:
import nanoid from 'nanoid'
size = 8
id = nanoid(size) //cjF38yYc
UPD:
НИКОГДА не используйте nanoid() для индексации списков или любых других повторяющихся элементов с помощью key в ReactJS. Так как если использовать этот метод, каждый раз при обновлении компонента React будет думать, что все элементы новые и будет их перерендерить, что очень плохо скажется на производительность и вообще противоречит смыслу key.
Подробнее о key: reactjs.org/docs/lists-and-keys.html#keys
Еще подробнее о key: habr.com/company/hh/blog/352150
Пример очень плохого использования nanoid:
import nanoid from 'nanoid'
import React from 'react'
const ListDrinks = props=>{
const drinks = ['rum','bear','vodka']
return(
<ul>
{drinks.map((values)=>{
//Вот так делать нельзя!
return(
<li key={nanoid(8)}>{values}</li>
)
})}
</ul>
)
}
export default ListDrinks
Пример нормального использования nanoid:
import React, { Component } from 'react';
import nanoid from 'nanoid'
class InputWithLabel extends Component {
constructor(props){
super(props)
this.id = nanoid()
}
render() {
return (
<div>
<label htmlFor={this.id}>My label</label>
<input id={this.id} type="text"/>
</div>
);
}
}
export default InputWithLabel;
То, как делать нельзя я узнал от добрых людей в комментариях, за что им отдельное, большое спасибо!
2) Classnames
Ссылка на github
Эта библиотека для простого условного объединения имен классов. Пользоваться ей не намного сложнее, чем предыдущей библиотекой.
Пример простого использования:
import cn from 'classnames'
cn('menu','active')//'menu active'
let isActive = true
cn('menu',{'active':isActive})//'menu active'
isActive = false
cn('menu',{'active':isActive})//'menu'
Лично для меня эта библиотека является обязательной в любом React-приложении. Конечно, до того момента, пока я не найду более удобный инструмент.
3) Formik и Yup
Ссылка на github(Formik)
Ссылка на github(Yup)
В разговоре об упрощении чего-либо в React нельзя не упомянуть работу с формами. Наверное, каждый начинающий React-developer в один прекрасный момент понимал, как он ненавидит работу с формами. Когда приходит это понимание, стоит незамедлительно искать спасительную пилюлю.
Для меня этой пилюлей стали Formik и Yup.
Formik — библиотека, помогающая работать с формами. Она упрощает получение данных из формы, валидацию данных, вывод сообщений об ошибках и многое другое.
Yup — библиотека, которая является валидатором для модели, которую мы сами и создаем с помощью Yup.
Для сколько-нибудь полного описания этой связки нужна отдельная статья, но я попытаюсь показать с высоты птичьего полета, что они из себя представляют.
Код примера можно запустить тут: Пример
Первым делом создадим схему:
import * as Yup from "yup";
const BasicFormSchema = Yup.object().shape({
email: Yup.string()
//Проверяем, корректный ли адрес.
//Если нет, то выводится сообщение в скобках
.email("Invalid email address")
//не сабмитим, если поле не заполнено
.required("Required"),
username: Yup.string()
//минимальная длина - 2 символа
.min(2, "Must be longer than 2 characters")
//максимальная длина - 20 символов
.max(20, "Nice try, nobody has a first name that long")
.required("Required"),
password: Yup.string()
.min(8, "Must be longer than 8 characters")
.required("Required")
});
export default BasicFormSchema;
В коде выше мы определили схему, которая по сути — объект. Она имеет три поля: email, username и password. Каждому из полей мы определили некоторые проверки.
Одним из способов использования Formik является элемент <Formik/>, который имеет множество разных свойств, один из которых render.
import React from "react";
import { Formik, Field, Form } from "formik";
import BasicFormSchema from "./BasicFormSсhema";
const SignUp = () => (
<div className="container">
<h1>Sign up</h1>
<Formik
//инициализируем значения input-ов
initialValues={{
email: "",
username: "",
password: ""
}}
//подключаем схему валидации, которую описали выше
validationSchema={BasicFormSchema}
//определяем, что будет происходить при вызове onsubmit
onSubmit={values => {
setTimeout(() => {
alert(JSON.stringify(values, null, 2));
}, 500);
}}
//свойство, где описывыем нашу форму
//errors-ошибки валидации формы
//touched-поля формы, которые мы "затронули",
//то есть, в которых что-то ввели
render={({ errors, touched }) => (
<Form className="form-container">
<label htmlFor="email">Email</label>
<Field
name="email"
placeholder="mtarasov777@gmail.com"
type="email"
/>
{//если в этом поле возникла ошибка и
//если это поле "затронуто, то выводим ошибку
errors.email &&
touched.email && <div className="field-error">{errors.email}</div>}
<label htmlFor="username">Username</label>
<Field name="username" placeholder="snapoak" type="text" />
{errors.username &&
touched.username && (
<div className="field-error">{errors.username}</div>
)}
<label htmlFor="password">Password</label>
<Field name="password" placeholder="123456qwe" type="password" />
{errors.password &&
touched.password && (
<div className="field-error">{errors.password}</div>
)}
<button type="submit">Submit</button>
</Form>
)}
/>
</div>
);
export default SignUp;
Код простой, я снабдил его комментариями, поэтому, думаю, вопросов возникнуть не должно.
Если же они возникли, то в ГитХаб репозитории имеется отличная документация, также можете задавать вопросы в комментариях.
Вот и конец. Я знаю, что есть много отличных библиотек, для работы с формами, какие-то кажутся вам лучшими, какие-то худшими. Я выразил тут личное мнение.
Надеюсь, что эта статья кому-нибудь может. Можете писать свои примеры полезных библиотек в комментарии, буду рад узнать что-то новое.
vtvz_ru
Статья немного не формата хабра, но ладно… Немного моих мыслей.
Замечание по поводу первой библиотеки: использовать в списках ее категорически нельзя. Это может очень плохо сказаться на производительности, особенно на больших списках, т.к. ReactJS будет пересоздавать элементы на каждый чих. Тогда уж лучше вообще ключи не использовать, чем рандомные. На крайний случай, индекс массива, хотя это тоже не рекомендуется. А лучше всего использовать статичные IDшники, либо, как в Вашем примере, сам элемент:
faiwer
Народ наткнувшись на это "не рекомендуется" начинает бояться так делать, и начинает городить чёрт знает что. Топик-стартер вот генерирует новые ID при каждом render-е. А один мой коллега очень сильно извратил кодовую базу, добавляя ID в статичные списки сущностей.
На самом же деле надо просто понять проблему. И тогда оказывается, что в ряде случаев использование индекса в массиве в качестве key очень даже уместно. По сути, если:
… то мы можем смело использовать
key={index}
ни о чём не переживая. По правде говоря, даже не удовлетворяя этим двум пунктам иногда можно использоватьkey={index}
, но это уже совсем другая история (с).P.S. автор статьи по вашей ссылке не прав в пункте "the list and items are static–they are not computed and do not change" + "When all of them are met, you may safely use the index as a key.". Потому что "items" совершенно не обязательно должны быть статичными.
Dartess
Кажется, вы переизобрели react-bem-classes :) прекрасная штука, очень удобно работать в т.ч. с модификаторами, плюс миксовать блоки и элементы.
vtvz_ru
Ну почти переизобрел (Библиотека не моя, просто пользуюсь. Только декоратор самописный). Мне кажется, react-bem-helper немного более гибкий. По крайней мере я не заметил в Вашей библиотеке подобных конструкций:
Для меня замыкания не столь необходимы, но вот передавать объект с модификаторами, где все контролируется Boolean значениями, это просто мегаудобно. Прям как доктор прописал.
Посмотрите в сторону этой библиотеки. Там много чего интересного есть. А декоратор написать не сложно.
Dartess
Она не моя, просто остановился на ней, когда выбирал.
вот так работает, т.е. модификаторы либо булевы, либо со значением. В документации нет примера с модификаторами элемента, только пример с блоком, но работает так же (сама возможность передавать модификаторы объектом описана). Передачи коллбэка, вроде как, нет, и ИМХО это был бы лишний функционал.
Того что есть хватает за глаза. Подметил просто, что
react-bem-helper
с декоратором стал похож наreact-bem-classes
.snapoak Автор
Спасибо за подробный ответ! Узнал много нового!
faiwer
А что вы хотели показать примером для
Nanoid
? Вы всякий раз генерируете новый ID для элементов списка. Чего ради? По сути вы показываете React, что у вас всякий раз НОВЫЕ элементы. React не соотносит их со старыми instance-ми компонент, а генерирует новые. Полностью с нуля формирует всё, что касается VDom. Всякий раз. Даже если не поменялось вообще ничего. Зачем?kylt_lichnosti
Зато шторм не ругается.
Valery4
Имея в проекте lodash вообще не понимаю зачем бы мне понадобился nanoid.
Valery4
И да, я не про списки и Реакт. А вообще если бы мне понадобились уникальные ID.
Например когда я делал мок данных которые потом буду приходить с сервера с уникальными ID.
snapoak Автор
1КБ на дороге не валяется!
P.S. спасибо, никогда не смотрел в сторону lodash, так как приятель сказал, что ramda ramda ramda
Valery4
Не 2, а 2.2K — 1.2K = 1K.
А если в сжатом виде 915 — 638 = 277
Внимательно прочитайте ещё раз мой первый коммент.
Имея оверхед в 277 байт — я не буду подключать ради их экономии более лёгкую библиотеку.
Допустим tree shaking работает как надо.
snapoak Автор
Опечатался, прошу прощения.
Про размер ирония была(часто иронизирую про размер), конечно, если используете lodash, в nanoid смысла никакого нет
Valery4
А я про Ramda никогда не слышал.
snapoak Автор
Библиотека для функционального программирования на JS. Много интересных штук, позволяет писать чистый код (описание многих ФП библиотек на JS)
Blooderst
А чем Math.random не угодил? Для цели генерации случайных ключей в реактовском списке (вон из профессии!) полностью подходит — шанс повторения ничтожно мал, все генерируемые значения можно считать уникальными. Экономия в 638 (915) байт.
AxisPod
Nanoid, что за бред? Убить всю производительность React, пусть тормозит, процы ныне мощные, надо чем-то нагружать? Тут я бы ещё понял, если эта либа вычисляла хэш для объекта, а случайное число, это уже полный бред.
vlreshet
Если откинуть бредовость конкретно для реакта — Nanoid всё-равно остаётся дичью. Тянуть целую библиотеку (пускай и такую маленькую) вместо одной строчки vanilla js
UPD. досмотрел, они ещё обещают «secure», и, мол, чуток больший диапазон символов. Но тем не менее
Nookie-Grey
Пугает, что кто то еще ставит + к рейтингу для этой статьи…
faiwer
Что характерно — автор критику прочитал… и забил :)
snapoak Автор
Не забил ;(