У вас є компонент React, який використовує useRef та IntersectionObserver для визначення, коли користувач переглядає кінець вмісту. Ваше завдання полягає в наступному:
Встановіть правильні типи пропсів для цього компонента. У ньому є дві властивості: children і onContentEndVisible. children - це будь-який валідний React вузол, а onContentEndVisible - це функція без аргументів, що повертає void.
Встановіть правильний тип useRef. Посилання endContentRef використовується для div, який міститься в кінці вмісту.
Встановіть правильний тип для options (клас також може бути типом для options).
import React, { useEffect, useRef } from 'react';
// Опишіть Props
export function Observer({ children, onContentEndVisible }: Props) {
// Вкажіть правильний тип для useRef зверніть увагу, в який DOM елемент ми його передаємо
const endContentRef = useRef(null);
useEffect(() => {
// Вкажіть правильний тип для options, підказка, клас також можна вказувати як тип
const options = {
rootMargin: '0px',
threshold: 1.0,
root: null,
};
const observer = new IntersectionObserver(entries => {
entries.forEach(entry => {
if (entry.intersectionRatio > 0) {
onContentEndVisible();
observer.disconnect();
}
});
}, options);
if (endContentRef.current) {
observer.observe(endContentRef.current);
}
return () => {
observer.disconnect();
};
}, [onContentEndVisible]);
return (
<div>
{children}
<div ref={endContentRef} />
</div>
);
}
Ваше завдання – додати типи для наступних елементів коду:
RequestStep: Це рядковий літерал.
State: Цей тип являє собою об'єкт з двома властивостями isRequestInProgress і RequestStep
Action: Це тип, що представляє можливі дії, які можуть бути відправлені до редюсера.
Дивіться код і опишіть для нього правильні типи.
import React, { useReducer } from 'react';
const initialState: State = {
isRequestInProgress: false,
requestStep: 'idle',
};
function requestReducer(state: State, action: Action): State {
switch (action.type) {
case 'START_REQUEST':
return { ...state, isRequestInProgress: true, requestStep: 'start' };
case 'PENDING_REQUEST':
return { ...state, isRequestInProgress: true, requestStep: 'pending' };
case 'FINISH_REQUEST':
return { ...state, isRequestInProgress: false, requestStep: 'finished' };
case 'RESET_REQUEST':
return { ...state, isRequestInProgress: false, requestStep: 'idle' };
default:
return state;
}
}
export function RequestComponent() {
const [requestState, requestDispatch] = useReducer(
requestReducer,
initialState,
);
const startRequest = () => {
requestDispatch({ type: 'START_REQUEST' });
// Імітуємо запит до сервера
setTimeout(() => {
requestDispatch({ type: 'PENDING_REQUEST' });
// Імітуємо отримання відповіді від сервера
setTimeout(() => {
requestDispatch({ type: 'FINISH_REQUEST' });
}, 2000);
}, 2000);
};
const resetRequest = () => {
requestDispatch({ type: 'RESET_REQUEST' });
};
return (
<div>
<button onClick={startRequest}>Почати запит</button>
<button onClick={resetRequest}>Скинути запит</button>
<p>Стан запиту: {requestState.requestStep}</p>
</div>
);
}
export default RequestComponent;
Ви створюєте компонент форми у React. Ви маєте поле введення, в якому ви хочете відстежити зміни. Для цього ви використовуєте обробник подій onChange. Ваше завдання – правильно типізувати подію, яка передається у цю функцію.
import React, { useState } from 'react';
export function FormComponent() {
const [value, setValue] = useState('');
const handleChange = event => {
setValue(event.target.value);
};
return <input type="text" value={value} onChange={handleChange} />;
}
Ви вирішили застосувати до меню контекст і тепер вам потрібно його типізувати.
Описати тип SelectedMenu: Це має бути об'єкт, який містить id з типом MenuIds
Описати тип MenuSelected: Цей тип є об'єктом, що містить selectedMenu
Описати тип MenuAction: Цей тип являє собою об'єкт з методом onSelectedMenu, який приймає об'єкт типу SelectedMenu як аргумент повертає void.
Описати тип PropsProvider: Опишіть правильний тип для дітей
Описати тип PropsMenu: Опишіть тип для menus, він має бути від типу Menu
import React, { createContext, useMemo, useState, useContext } from 'react';
import noop from 'lodash/noop';
type MenuIds = 'first' | 'second' | 'last';
type Menu = { id: MenuIds; title: string };
// Додати тип Menu Selected
const MenuSelectedContext = createContext<MenuSelected>({
selectedMenu: {},
});
// Додайте тип MenuAction
const MenuActionContext = createContext<MenuAction>({
onSelectedMenu: noop,
});
type PropsProvider = {
children; // Додати тип для children
};
function MenuProvider({ children }: PropsProvider) {
// Додати тип для SelectedMenu він повинен містити { id }
const [selectedMenu, setSelectedMenu] = useState<SelectedMenu>({});
const menuContextAction = useMemo(
() => ({
onSelectedMenu: setSelectedMenu,
}),
[],
);
const menuContextSelected = useMemo(
() => ({
selectedMenu,
}),
[selectedMenu],
);
return (
<MenuActionContext.Provider value={menuContextAction}>
<MenuSelectedContext.Provider value={menuContextSelected}>
{children}
</MenuSelectedContext.Provider>
</MenuActionContext.Provider>
);
}
type PropsMenu = {
menus; // Додайте вірний тип для меню
};
function MenuComponent({ menus }: PropsMenu) {
const { onSelectedMenu } = useContext(MenuActionContext);
const { selectedMenu } = useContext(MenuSelectedContext);
return (
<>
{menus.map(menu => (
<div key={menu.id} onClick={() => onSelectedMenu({ id: menu.id })}>
{menu.title}{' '}
{selectedMenu.id === menu.id ? 'Selected' : 'Not selected'}
</div>
))}
</>
);
}
export function ComponentApp() {
const menus: Menu[] = [
{
id: 'first',
title: 'first',
},
{
id: 'second',
title: 'second',
},
{
id: 'last',
title: 'last',
},
];
return (
<MenuProvider>
<MenuComponent menus={menus} />
</MenuProvider>
);
}