Реагування станом на введення
React надає декларативний спосіб маніпулювання UI. Замість маніпулювання окремими шматочками UI безпосередньо, слід описувати різні стани, в яких може перебувати компонент, і перемикатися між ними у відповідь на введення користувачем. Це схоже на те, як UI уявляють дизайнери.
You will learn
- Як декларативне програмування UI відрізняється від імперативного
- Як перелічити різні візуальні стани, в яких може перебувати компонент
- Як у коді запустити зміни між різними візуальними станами
Як декларативний UI відрізняється від імперативного
Під час розробки взаємодій із UI, ймовірно, ви думаєте про те, як UI змінюється у відповідь на дії користувача. Уявіть форму, що дає користувачу змогу надіслати відповідь:
- Коли ви друкуєте щось у формі, кнопка “Надіслати” стає увімкненою.
- Коли ви натискаєте “Надіслати”, то і форма, і кнопка стають вимкненими, а натомість з’являється елемент індикації надсилання.
- Якщо мережевий запит успішний, то форма ховається, і з’являється повідомлення “Дякуємо”.
- Якщо мережевий запит невдалий, то з’являється повідомлення про помилку, а форма знову стає ввімкненою.
В імперативному програмуванні описане вище безпосередньо відповідає тому, як реалізується взаємодія. Доводиться писати прямі інструкції для маніпулювання UI, залежно від того, що відбувається. Ось іще один спосіб подумати про це: уявіть, що їдете з кимось в авто й керуєте поїздкою, називаючи кожний поворот.
Illustrated by Rachel Lee Nabors
Водій не знає, куди ви хочете потрапити, він просто виконує команди. (І якщо ви переплутаєте орієнтири, то опинитеся не там!) Це зветься імперативним, тому що доводиться “командувати” кожним елементом, від індикатора до кнопки, кажучи комп’ютеру, як саме оновлювати UI.
У цьому прикладі імперативного програмування UI форма створена без React. Вна використовує лише браузерний DOM:
async function handleFormSubmit(e) { e.preventDefault(); disable(textarea); disable(button); show(loadingMessage); hide(errorMessage); try { await submitForm(textarea.value); show(successMessage); hide(form); } catch (err) { show(errorMessage); errorMessage.textContent = err.message; } finally { hide(loadingMessage); enable(textarea); enable(button); } } function handleTextareaChange() { if (textarea.value.length === 0) { disable(button); } else { enable(button); } } function hide(el) { el.style.display = 'none'; } function show(el) { el.style.display = ''; } function enable(el) { el.disabled = false; } function disable(el) { el.disabled = true; } function submitForm(answer) { // Удаймо, що тут відбувається мережевий запит. return new Promise((resolve, reject) => { setTimeout(() => { if (answer.toLowerCase() === 'стамбул') { resolve(); } else { reject(new Error('Гарний варіант, але неправильна відповідь. Спробуйте ще!')); } }, 1500); }); } let form = document.getElementById('form'); let textarea = document.getElementById('textarea'); let button = document.getElementById('button'); let loadingMessage = document.getElementById('loading'); let errorMessage = document.getElementById('error'); let successMessage = document.getElementById('success'); form.onsubmit = handleFormSubmit; textarea.oninput = handleTextareaChange;
Імперативна маніпуляція UI добре працює в ізольованих прикладах, але її складність зростає експоненційно в складніших системах. Уявіть оновлення сторінки, сповненої різних форм, схожих на цю. Додавання нового елемента UI або нової взаємодії може вимагати ретельної перевірки всього наявного коду, аби пересвідчитись, що не з’явився якийсь дефект (наприклад, не забули показати чи приховати щось).
React створений для розв’язання цієї проблеми.
У React ви не маніпулюєте UI безпосередньо, тобто ви не вмикаєте, вимикаєте, показуєте чи приховуєте компоненти безпосередньо. Замість цього ви оголошуєте, що хочете показати, і React з’ясовує, як оновити UI. Уявіть, що ви ніби сідаєте в таксі й кажете водієві, куди хочете поїхати, але не керуєте кожним його поворотом. Це його робота — довезти вас туди, і він може навіть знати короткі шляхи, про які ви б не подумали!
Illustrated by Rachel Lee Nabors
Декларативне осмислення UI
Вище ви побачили, як реалізувати форму імперативно. Щоб краще зрозуміти, як мислити у стилі React, далі ми проведемо вас крізь повторну реалізацію того самого UI за допомогою React:
- З’ясуйте різні візуальні стани свого компонента
- Визначте, що збуджує ці зміни стану
- Представте стан у пам’яті за допомогою
useState
- Вилучіть усі несуттєві змінні стану
- Приєднайте обробники подій, щоб задати стан
Крок 1. З’ясуйте різні візуальні стани свого компонента
Досліджуючи комп’ютерні науки, ви могли чути про “скінченний автомат”, що перебуває в одному з декількох “станів”. Якщо ви працюєте разом із дизайнером, то могли бачити макети різних “візуальних станів”. React розташований на перетині дизайну та комп’ютерних наук, тож обидві ці ідеї — наші джерела натхнення.
По-перше, необхідно візуалізувати всі різні “стани” UI, які користувач може побачити:
- Порожній — форма має вимкнену кнопку “Надіслати”.
- Друкування — форма має ввімкнену кнопку “Надіслати”.
- Надсилання — форма повністю вимкнена. Показано індикатор надсилання.
- Успіх — замість форми показано повідомлення “Дякуємо”.
- Помилка — те саме, що для стану “Друкування”, але з додатковим повідомленням про помилку.
Як і дизайнеру, вам захочеться “макетувати” чи створити “макети” (“mocks”) різних станів, перш ніж додавати логіку. Наприклад, ось макет суто візуальної частини форми. Цей макет контролюється пропом status
, чиє усталене значення — 'empty'
:
export default function Form({ status = 'empty' }) { if (status === 'success') { return <h1>Правильно!</h1> } return ( <> <h2>Вікторина міст</h2> <p> У якому місті є білборд, що перетворює повітря на питну воду? </p> <form> <textarea /> <br /> <button> Надіслати </button> </form> </> ) }
Цей проп можна назвати як завгодно, назва тут неважлива. Спробуйте замінити status = 'empty'
на status = 'success'
, щоб побачити появу повідомлення про успіх. Макетування дає змогу швидко ітеруватися в розробці UI перед під’єднанням будь-якої логіки. Ось змістовніший прототип того самого компонента, так само “контрольований” пропом status
:
export default function Form({ // Спробуйте 'submitting', 'error', 'success': status = 'empty' }) { if (status === 'success') { return <h1>Правильно!</h1> } return ( <> <h2>Вікторина міст</h2> <p> У якому місті є білборд, що перетворює повітря на питну воду? </p> <form> <textarea disabled={ status === 'submitting' } /> <br /> <button disabled={ status === 'empty' || status === 'submitting' }> Надіслати </button> {status === 'error' && <p className="Error"> Гарний варіант, але неправильна відповідь. Спробуйте ще! </p> } </form> </> ); }
Deep Dive
Якщо компонент має багато візуальних станів, було б зручніше показати їх на одній сторінці:
import Form from './Form.js'; let statuses = [ 'empty', 'typing', 'submitting', 'success', 'error', ]; export default function App() { return ( <> {statuses.map(status => ( <section key={status}> <h4>Форма ({status}):</h4> <Form status={status} /> </section> ))} </> ); }
Сторінки, схожі на цю, нерідко звуть “живими стилістичними настановами” (“living styleguides”) або “сторибуками” (“storybooks”).
Крок 2. Визначте, що збуджує ці зміни стану
Збудити зміни стану можна у відповідь на два види введення:
- Людське введення, наприклад, клацання кнопки, друкування в полі, перехід за посиланням.
- Комп’ютерне введення, наприклад, надходження мережевої відповіді, завершення таймера, завантаження зображення.
Illustrated by Rachel Lee Nabors
В обох випадках необхідно задати змінні стану, щоб UI оновився. У формі, що ми розробляємо, необхідно змінити стан у відповідь на кілька різних введень:
- Зміни в текстовому полі (людське) повинні перемкнути форму зі стану Порожній до стану Друкування та навпаки, залежно від того, чи є текстове поле порожнім.
- Клацання кнопки “Надіслати” (людське) повинно перемкнути форму до стану Надсилання.
- Успішна мережева відповідь (комп’ютерне) повинна перемикати її до стану Успіх.
- Невдала мережева відповідь (комп’ютерне) повинна перемикати її до стану Помилка з відповідним повідомленням про помилку.
Щоб легше візуалізувати ці переходи, спробуйте намалювати кожний стан на папері як підписане коло, а кожну зміну між двома станами — стрілкою. Так ви можете накреслити чимало переходів і знайти дефекти задовго до початку реалізації.
Крок 3. Представте стан у пам’яті за допомогою useState
Далі необхідно представити візуальні стани свого компонента у пам’яті, використовуючи useState
. Ключовою в цій справі є простота: кожна дрібка стану — це “рухома деталь”, і вам краще мати якомога менше таких “рухомих деталей”. Більше складності — більше помилок!
Почніть зі стану, який безсумнівно повинен бути присутній. Наприклад, необхідно зберігати answer
— значення поля, а також за наявності error
— останню помилку:
const [answer, setAnswer] = useState('');
const [error, setError] = useState(null);
Далі потрібна змінна стану, що представляє те, який із візуальних станів має бути виведений. Зазвичай є більш ніж один спосіб представити це в пам’яті, тому доведеться з цим поекспериментувати.
Якщо вам важко одразу вигадати найкращий спосіб, почніть із додавання такої кількості стану, щоб напевно були покриті всі можливі візуальні стани:
const [isEmpty, setIsEmpty] = useState(true);
const [isTyping, setIsTyping] = useState(false);
const [isSubmitting, setIsSubmitting] = useState(false);
const [isSuccess, setIsSuccess] = useState(false);
const [isError, setIsError] = useState(false);
Ваш перший варіант навряд буде найкращим із можливих, але це нормально: рефакторинг стану — частина процесу розробки!
Крок 4. Вилучіть усі несуттєві змінні стану
Краще уникати дублювання вмісту стану, щоб відстежувати лише те, що суттєво. Якщо витратити трохи часу на рефакторинг структури стану, то компоненти стануть легшими для розуміння, зменшиться дублювання, буде менше плутанини. Ваша мета — уникнути випадків, коли стан у пам’яті не представляє жодного валідного UI, який ви хочете показати користувачу. (Наприклад, ви не хочете, щоб водночас можна було побачити повідомлення про помилку та вимкнене поле, бо тоді користувач не зможе виправити помилку!)
Ось кілька питань, котрі можна поставити, щодо змінних стану:
- Чи призводить цей стан до парадоксів? Наприклад,
isTyping
іisSubmitting
не можуть водночас бутиtrue
. Парадокс зазвичай означає, що на стан накладено недостатньо обмежень. Є чотири можливі комбінації двох булевих змінних, але лише три з них відповідають валідним станам. Щоб позбавитися “неможливого” стану, можна поєднати ці змінні в одну зміннуstatus
, яка повинна мати одне з трьох значень:'typing'
,'submitting'
або'success'
. - Чи доступна та сама інформація в іншій змінній стану? Ще один парадокс:
isEmpty
йisTyping
не можуть водночас мати значенняtrue
. Їхнє розділення загрожує можливою розсинхронізацією та породженням помилок. На щастя, можна вилучитиisEmpty
, а натомість перевірятиanswer.length === 0
. - Чи можна отримати ту саму інформацію через інверсію іншої змінної стану? Змінна
isError
не потрібна, тому що можна натомість перевіритиerror !== null
.
Після такого прибирання залишаються 3 (із 7 на початку!) суттєві змінні стану:
const [answer, setAnswer] = useState('');
const [error, setError] = useState(null);
const [status, setStatus] = useState('typing'); // 'typing', 'submitting' або 'success'
Вони суттєві, тому що жодну з них не можна вилучити, не зламавши функціональність.
Deep Dive
Ці три змінні доволі добре представляють стан цієї форми. Проте є деякі проміжні стани, що не зовсім мають зміст. Наприклад, ненульове значення error
не має змісту, коли status
має значення success
. Щоб точніше змоделювати стан, його можна виокремити в редюсер. Редюсери дають змогу уніфікувати кілька змінних стану в один об’єкт, а також скріпити всю пов’язану з ними логіку!
Крок 5. Приєднайте обробники подій, щоб задати стан
Врешті-решт, створімо обробники подій, що оновлюють стан. Нижче — остаточний вигляд форми, де під’єднані всі обробники подій:
import { useState } from 'react'; export default function Form() { const [answer, setAnswer] = useState(''); const [error, setError] = useState(null); const [status, setStatus] = useState('typing'); if (status === 'success') { return <h1>Правильно!</h1> } async function handleSubmit(e) { e.preventDefault(); setStatus('submitting'); try { await submitForm(answer); setStatus('success'); } catch (err) { setStatus('typing'); setError(err); } } function handleTextareaChange(e) { setAnswer(e.target.value); } return ( <> <h2>Вікторина міст</h2> <p> У якому місті є білборд, що перетворює повітря на питну воду? </p> <form onSubmit={handleSubmit}> <textarea value={answer} onChange={handleTextareaChange} disabled={status === 'submitting'} /> <br /> <button disabled={ answer.length === 0 || status === 'submitting' }> Надіслати </button> {error !== null && <p className="Error"> {error.message} </p> } </form> </> ); } function submitForm(answer) { // Удаймо, що тут звертання до мережі. return new Promise((resolve, reject) => { setTimeout(() => { let shouldError = answer.toLowerCase() !== 'lima' if (shouldError) { reject(new Error('Гарний варіант, але неправильна відповідь. Спробуйте ще!')); } else { resolve(); } }, 1500); }); }
Попри те, що цей код не перевищує за розміром вихідний імперативний приклад, він значно надійніший. Вираження всіх взаємодій як змін до стану дає змогу пізніше додавати нові візуальні стани, не ламаючи наявних. Також це дає змогу змінювати те, що повинно виводитися в кожному стані, не змінюючи логіки самої взаємодії.
Recap
- Декларативне програмування означає описувати UI для кожного візуального стану, а не займатися мікроменеджментом UI (імперативним стилем).
- Для розробки компонента:
- З’ясуйте всі його візуальні стани.
- Визначте людські та комп’ютерні тригери змін стану.
- Змоделюйте стан за допомогою
useState
. - Вилучіть несуттєві частини стану, щоб уникнути помилок і парадоксів.
- Під’єднайте обробників подій, що задаватимуть значення стану.
Challenge 1 of 3: Додавання та вилучення класу CSS
Зробіть так, щоб клацання картинки вилучало клас CSS background--active
із зовнішнього <div>
, але додавало клас picture--active
до <img>
. Повторне клацання фону повинно відновлювати вихідні класи CSS.
Візуально слід очікувати, що клацання картинки вилучить фіолетовий фон і виділить межі картинки. Клацання поза картинкою виділяє фон, але вилучає виділення меж картинки.
export default function Picture() { return ( <div className="background background--active"> <img className="picture" alt="Веселкові будинки в Кампунг Пелангі, Індонезія" src="https://i.imgur.com/5qwVYb1.jpeg" /> </div> ); }