Напиши застосунок зберігання контактів телефонної книги.
Застосунок повинен складатися з форми і списку контактів. На поточному кроці реалізуй додавання імені контакту та відображення списку контактів. Застосунок не повинен зберігати контакти між різними сесіями (оновлення сторінки).
Використовуйте цю розмітку інпуту з вбудованою валідацією для імені контакту.
<input
type="text"
name="name"
pattern="^[a-zA-Zа-яА-Я]+(([' -][a-zA-Zа-яА-Я ])?[a-zA-Zа-яА-Я]*)*$"
title="Name may contain only letters, apostrophe, dash and spaces. For example Adrian, Jacob Mercer, Charles de Batz de Castelmore d'Artagnan"
required
/>
Стан, що зберігається в батьківському компоненті <App>
, обов'язково повинен
бути наступного вигляду, додавати нові властивості не можна.
state = {
contacts: [],
name: ''
}
Кожен контакт повинен бути об'єктом з властивостями name
та id
. Для
генерації ідентифікаторів використовуй будь-який відповідний пакет, наприклад
nanoid. Після завершення цього кроку,
застосунок повинен виглядати приблизно так.
Розшир функціонал застосунку, дозволивши користувачам додавати номери телефонів.
Для цього додай <input type="tel">
у форму і властивість для зберігання його
значення в стані.
state = {
contacts: [],
name: '',
number: ''
}
Використовуй цю розмітку інпуту з вбудованою валідацією для номеру контакту.
<input
type="tel"
name="number"
pattern="\+?\d{1,4}?[-.\s]?\(?\d{1,3}?\)?[-.\s]?\d{1,4}[-.\s]?\d{1,4}[-.\s]?\d{1,9}"
title="Phone number must be digits and can contain spaces, dashes, parentheses and can start with +"
required
/>
Після завершення цього кроку, застосунок повинен виглядати приблизно так.
Додай поле пошуку, яке можна використовувати для фільтрації списку контактів за ім'ям.
- Поле пошуку – це інпут без форми, значення якого записується у стан (контрольований елемент).
- Логіка фільтрації повинна бути нечутливою до регістру.
state = {
contacts: [],
filter: '',
name: '',
number: ''
}
Коли ми працюємо над новим функціоналом, буває зручно жорстко закодувати деякі дані у стан. Це позбавить необхідності вручну вводити дані в інтерфейсі для тестування роботи нового функціоналу. Наприклад, можна використовувати такий початковий стан.
state = {
contacts: [
{id: 'id-1', name: 'Rosie Simpson', number: '459-12-56'},
{id: 'id-2', name: 'Hermione Kline', number: '443-89-12'},
{id: 'id-3', name: 'Eden Clements', number: '645-17-79'},
{id: 'id-4', name: 'Annie Copeland', number: '227-91-26'},
],
filter: '',
name: '',
number: ''
}
Якщо твій застосунок реалізований в одному компоненті <App>
, виконай
рефакторинг, виділивши відповідні частини в окремі компоненти. У стані
кореневого компонента <App>
залишаться тільки властивості contacts
і
filter
.
state = {
contacts: [],
filter: ''
}
Достатньо виділити чотири компоненти: форма додавання контактів, список контактів, елемент списку контактів та фільтр пошуку.
Після рефакторингу кореневий компонент програми виглядатиме так.
<div>
<h1>Phonebook</h1>
<ContactForm ... />
<h2>Contacts</h2>
<Filter ... />
<ContactList ... />
</div>
Заборони користувачеві можливість додавати контакти, імена яких вже присутні у
телефонній книзі. При спробі виконати таку дію виведи alert
із попередженням.
Розшир функціонал застосунку, дозволивши користувачеві видаляти раніше збережені контакти.