У меня есть состояние, которое я использую для добавления новых элементов JSX:
const [display, setDisplay] = useState<IDisplay>({
BookingFormDropDown: [],
}
);
Добавление элемента onClick () с помощью этой функции из render ():
const addNewService = () => {
setDisplay(
(prevState: IDisplay) => (
{
...prevState, BookingFormDropDown: [...prevState.BookingFormDropDown,
<div>
<BookingFormDropDown bookingData={bookingData}/>
</div>]
}
)
);
};
Циклы и извлечение эти элементы JSX в render () следующим образом:
display.BookingFormDropDown.map((el: JSX.Element, i: number) => {
return <div key={i}>
<div>{el}</div>
<div className="row">
<div>
<button type="button"
onClick={() => handelRemoveService(i)}>Remove
</button>
</div>
</div>
</div>
})
Почему мое приложение всегда удаляет последний элемент при любом нажатии кнопки? Насколько я понимаю, после добавления нового элемента в массив приложение должно снова l oop display.BookingFormDropDown и новые значения индекса должны быть переданы для каждой кнопки, пока map ().
const handelRemoveService = (i: number) => {
const state: JSX.Element[] = display.BookingFormDropDown;
state.splice(i, 1);
setDisplay(
(prevState: IDisplay) => ({
...prevState, BookingFormDropDown: state
})
);
};
Только сегодня решил для реализации хуков компонент на основе классов прекрасно работает с аналогичным логином c.
Полный код компонента:
import React, {FunctionComponent, useState} from 'react';
import './BookingForm.css';
import Calendar from 'react-calendar';
import Slots from "../../components/Slots/Slots";
import ClientDetails from "../../components/ClientDetails";
import BookingFormDropDown from "../../components/BookingFormDropDrown";
//import {IService, IServiceCategory, IEmployee, ISlot} from "../../interfaces/IBookingFrom";
export interface IBookingData {
Location: string[] //ILocation[]
Categories: string[] //ICategory[]
Services: string[] //IService[]
Employees: string[] //IEmployee[]
Slots: string[] //ISlot[]
date: Date[] | Date
}
interface IDisplay {
BookingFormDropDown: JSX.Element[]
BookingStage: string | null
Mode: string | null
error: string | null
}
const BookingForm: FunctionComponent = () => {
const [bookingData, setBookingData] = useState<IBookingData>({
Location: ['t', 'l', 'r'],
Categories: ['one', 'two', 'tree'],
Services: ['one1', 'two1', 'tree1'],
Employees: ['one2', 'two2', 'tree2'],
Slots: ['Mon 12:00', 'Mon 14:01', 'Mon 15:02', 'Mon 12:03',
'Mon 14:04', 'Mon 15:05', 'Mon 12:06', 'Mon 14:07', 'Mon 15:08',
'Mon 15:09', 'Mon 12:10', 'Mon 14:11', 'Mon 15:12', 'Mon 12:13',
'Mon 14:14', 'Mon 15:15'],
date: new Date()
});
const [display, setDisplay] = useState<IDisplay>({
BookingFormDropDown: [],
BookingStage: null,
Mode: null,
error: null
}
);
const handelRemoveService = (i: number) => {
const state: JSX.Element[] = display.BookingFormDropDown;
state.splice(i, 1);
setDisplay(
(prevState: IDisplay) => ({
...prevState, BookingFormDropDown: state
})
);
};
const addNewService = () => {
setDisplay(
(prevState: IDisplay) => (
{
...prevState, BookingFormDropDown: [...prevState.BookingFormDropDown,
<div>
<BookingFormDropDown bookingData={bookingData}/>
</div>]
}
)
);
};
const handleNewDate = (newDate: Date[] | Date) => setBookingData({
...bookingData, date: newDate
});
return (
<div className="container">
<div className="row">
<h1 className="col">Demo place</h1>
</div>
<BookingFormDropDown bookingData={bookingData}/>
{/* Displaying additional services from array*/
display.BookingFormDropDown.map((el: JSX.Element, i: number) => {
return <div key={i}>
<div>{el}</div>
<div className="row">
<div className="col mb-4 d-flex justify-content-end">
<button type="button" className="col-md-2 btn btn-light"
onClick={() => handelRemoveService(i)}>Remove
</button>
</div>
</div>
</div>
})
}
<div className="row">
<div className="col mb-4 d-flex justify-content-start">
<button type="button" className="col-md-2 btn btn-light"
onClick={() => addNewService()}>Add service
</button>
</div>
</div>
<div className="row">
<div className="col-md-4">
<Calendar minDetail={'year'}
onChange={handleNewDate}
value={bookingData.date}/>
</div>
<Slots list={bookingData.Slots}/>
</div>
<div className="row">
<div className="col d-flex justify-content-center p-5">
<button type="button" className=" col-md-3 btn btn-light">NEXT</button>
</div>
</div>
<ClientDetails/>
</div>
);
};
export default BookingForm;