Lift Schedule child components state to parent
This commit is contained in:
parent
38858668eb
commit
950e2e3165
@ -1,79 +1,9 @@
|
|||||||
import React, { useEffect, useState } from 'react';
|
import React from 'react';
|
||||||
import FullCalendar from '@fullcalendar/react';
|
import FullCalendar from '@fullcalendar/react';
|
||||||
import dayGridPlugin from '@fullcalendar/daygrid';
|
import dayGridPlugin from '@fullcalendar/daygrid';
|
||||||
import interaction from '@fullcalendar/interaction';
|
import interaction from '@fullcalendar/interaction';
|
||||||
import { DateTime } from 'luxon';
|
|
||||||
|
|
||||||
export default function DaySelector() {
|
|
||||||
const [eventsList, setEventsList] = useState([]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
console.log('Latest events list: ', eventsList);
|
|
||||||
}, [eventsList]);
|
|
||||||
|
|
||||||
const handleSelect = (info) => {
|
|
||||||
let updatedEvents = [];
|
|
||||||
let datesList = new Set();
|
|
||||||
|
|
||||||
/// Add each date selected to datesList
|
|
||||||
let start = DateTime.fromISO(info.startStr);
|
|
||||||
let end = DateTime.fromISO(info.endStr);
|
|
||||||
|
|
||||||
// Check if selection contains only one day
|
|
||||||
let singleDay = end.toISODate() === start.plus({ days: 1 }).toISODate();
|
|
||||||
|
|
||||||
if (singleDay) {
|
|
||||||
// When a single date is selected
|
|
||||||
let selectedDate = info.startStr;
|
|
||||||
let selectedEvent = { start: selectedDate, display: 'background' };
|
|
||||||
|
|
||||||
// If selectedEvent exists in the list, find and set its index
|
|
||||||
let selectedEventIndex;
|
|
||||||
for (let index = 0; index < eventsList.length; index++) {
|
|
||||||
const eventDate = eventsList[index].start;
|
|
||||||
if (selectedDate === eventDate) {
|
|
||||||
// If it's the case, set current index to selectedEventIndex
|
|
||||||
selectedEventIndex = index;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (selectedEventIndex !== undefined) {
|
|
||||||
// If selectedEventIndex exists, remove the corresponding event from the list
|
|
||||||
eventsList.splice(selectedEventIndex, 1);
|
|
||||||
updatedEvents = [...eventsList];
|
|
||||||
} else {
|
|
||||||
// Add selected event to the list
|
|
||||||
updatedEvents = [...eventsList, selectedEvent];
|
|
||||||
}
|
|
||||||
// setEventsList(updatedEvents);
|
|
||||||
} else {
|
|
||||||
// When a range of dates is selected
|
|
||||||
let currentDate = start;
|
|
||||||
while (!(+end === +currentDate)) {
|
|
||||||
// Add currentDate to the datesList
|
|
||||||
datesList.add(currentDate.toISODate());
|
|
||||||
let newDate = currentDate.plus({ days: 1 });
|
|
||||||
currentDate = newDate;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Add each date from the eventsList to datesList
|
|
||||||
eventsList.forEach((event) => {
|
|
||||||
datesList.add(event.start);
|
|
||||||
});
|
|
||||||
|
|
||||||
// Create events from datesList and add them to updatedEvents
|
|
||||||
datesList.forEach((date) => {
|
|
||||||
updatedEvents.push({ start: date, display: 'background' });
|
|
||||||
});
|
|
||||||
}
|
|
||||||
setEventsList(updatedEvents);
|
|
||||||
};
|
|
||||||
|
|
||||||
const handleClear = () => {
|
|
||||||
setEventsList([])
|
|
||||||
}
|
|
||||||
|
|
||||||
|
export default function DaySelector({ eventsList, handleSelect, handleClear }) {
|
||||||
return (
|
return (
|
||||||
<FullCalendar
|
<FullCalendar
|
||||||
initialView='dayGridMonth'
|
initialView='dayGridMonth'
|
||||||
@ -83,8 +13,10 @@ export default function DaySelector() {
|
|||||||
select={(info) => handleSelect(info)}
|
select={(info) => handleSelect(info)}
|
||||||
defaultAllDay={true}
|
defaultAllDay={true}
|
||||||
events={eventsList}
|
events={eventsList}
|
||||||
customButtons={{resetSelection:{text: 'clear selection', click: handleClear}}}
|
customButtons={{
|
||||||
headerToolbar={{right: 'resetSelection today prev,next'}}
|
resetSelection: { text: 'clear selection', click: handleClear },
|
||||||
|
}}
|
||||||
|
headerToolbar={{ right: 'resetSelection today prev,next' }}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,102 +1,26 @@
|
|||||||
import React, { useState, useEffect } from 'react';
|
import React from 'react';
|
||||||
import { DateTime } from 'luxon';
|
|
||||||
|
|
||||||
import { Divider, Icon, IconButton } from 'rsuite';
|
import { Divider, Icon, IconButton } from 'rsuite';
|
||||||
|
|
||||||
export default function SelectedDates() {
|
export default function SelectedDates({ datesList, eventsList, handleDelete }) {
|
||||||
const [eventsList, setEventsList] = useState([
|
|
||||||
{
|
|
||||||
start: '2020-04-29',
|
|
||||||
display: 'background',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
start: '2020-04-30',
|
|
||||||
display: 'background',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
start: '2020-05-07',
|
|
||||||
display: 'background',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
start: '2020-05-06',
|
|
||||||
display: 'background',
|
|
||||||
},
|
|
||||||
]);
|
|
||||||
const [datesList, setDatesList] = useState(eventsToDates(eventsList));
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
// Update selected dates
|
|
||||||
let updatedDates = eventsToDates(eventsList);
|
|
||||||
setDatesList(updatedDates);
|
|
||||||
}, [eventsList]);
|
|
||||||
|
|
||||||
const handleDelete = (date) => {
|
|
||||||
let currentEvent = {
|
|
||||||
start: date.toFormat('yyyy-MM-dd'),
|
|
||||||
display: 'background',
|
|
||||||
};
|
|
||||||
|
|
||||||
// Find the event corresponding to the date
|
|
||||||
let selectedEventIndex;
|
|
||||||
for (let index = 0; index < eventsList.length; index++) {
|
|
||||||
const eventDate = eventsList[index].start;
|
|
||||||
if (currentEvent.start === eventDate) {
|
|
||||||
// When it's the case, set current index to selectedEventIndex
|
|
||||||
selectedEventIndex = index;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// Create updated eventsList
|
|
||||||
eventsList.splice(selectedEventIndex, 1);
|
|
||||||
let updatedEvents = [...eventsList];
|
|
||||||
// Update the eventsList
|
|
||||||
setEventsList(updatedEvents);
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<div>
|
||||||
{datesList.length > 0 && (
|
<h3>Dates selected</h3>
|
||||||
<div>
|
<ul>
|
||||||
<h3>Dates selected</h3>
|
{datesList.map((date) => (
|
||||||
<ul>
|
<li key={date}>
|
||||||
{datesList.map((date) => (
|
{date.setLocale('en-gb').toLocaleString()}
|
||||||
<li key={date}>
|
{` (${date.weekdayShort})`}
|
||||||
{date.setLocale('en-gb').toLocaleString()}
|
<Divider vertical />
|
||||||
{` (${date.weekdayShort})`}
|
<IconButton
|
||||||
<Divider vertical />
|
icon={<Icon icon='close' />}
|
||||||
<IconButton
|
appearance='subtle'
|
||||||
icon={<Icon icon='close' />}
|
circle
|
||||||
appearance='subtle'
|
onClick={() => handleDelete(date)}
|
||||||
circle
|
></IconButton>
|
||||||
onClick={() => handleDelete(date)}
|
</li>
|
||||||
></IconButton>
|
))}
|
||||||
</li>
|
</ul>
|
||||||
))}
|
</div>
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Convert events to Luxon Datetime objects
|
|
||||||
const eventsToDates = (events) => {
|
|
||||||
let dates = [];
|
|
||||||
events.forEach((event) => {
|
|
||||||
dates.push(DateTime.fromFormat(event.start, 'yyyy-MM-dd'));
|
|
||||||
});
|
|
||||||
return dates;
|
|
||||||
};
|
|
||||||
|
|
||||||
// Convert Luxon Datetime objects to events
|
|
||||||
const datesToEvents = (dates) => {
|
|
||||||
let events = [];
|
|
||||||
dates.forEach((date) => {
|
|
||||||
let event = {
|
|
||||||
start: date.toFormat('yyyy-MM-dd'),
|
|
||||||
display: 'background',
|
|
||||||
};
|
|
||||||
events.push(event);
|
|
||||||
});
|
|
||||||
return events;
|
|
||||||
};
|
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
import React from 'react';
|
import React, { useState, useEffect } from 'react';
|
||||||
import { Container, Form, FormControl, FormGroup, Input, Button } from 'rsuite';
|
import { Container, Form, FormControl, FormGroup, Input, Button } from 'rsuite';
|
||||||
|
import { DateTime } from 'luxon';
|
||||||
|
|
||||||
import TimezonePicker from '../components/General/TimezonePicker';
|
import TimezonePicker from '../components/General/TimezonePicker';
|
||||||
import DaySelector from '../components/Schedule/DaySelector';
|
import DaySelector from '../components/Schedule/DaySelector';
|
||||||
@ -7,6 +8,101 @@ import DurationSelector from '../components/Schedule/DurationSelector';
|
|||||||
import SelectedDates from '../components/Schedule/SelectedDates';
|
import SelectedDates from '../components/Schedule/SelectedDates';
|
||||||
|
|
||||||
export default function Schedule() {
|
export default function Schedule() {
|
||||||
|
const [eventsList, setEventsList] = useState([]);
|
||||||
|
const [datesList, setDatesList] = useState(eventsToDates(eventsList));
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
// Update selected dates
|
||||||
|
let updatedDates = eventsToDates(eventsList);
|
||||||
|
setDatesList(updatedDates);
|
||||||
|
}, [eventsList]);
|
||||||
|
|
||||||
|
const handleSelect = (info) => {
|
||||||
|
let updatedEvents = [];
|
||||||
|
let datesList = new Set();
|
||||||
|
|
||||||
|
/// Add each date selected to datesList
|
||||||
|
let start = DateTime.fromISO(info.startStr);
|
||||||
|
let end = DateTime.fromISO(info.endStr);
|
||||||
|
|
||||||
|
// Check if selection contains only one day
|
||||||
|
let singleDay = end.toISODate() === start.plus({ days: 1 }).toISODate();
|
||||||
|
|
||||||
|
if (singleDay) {
|
||||||
|
// When a single date is selected
|
||||||
|
let selectedDate = info.startStr;
|
||||||
|
let selectedEvent = { start: selectedDate, display: 'background' };
|
||||||
|
|
||||||
|
// If selectedEvent exists in the list, find and set its index
|
||||||
|
let selectedEventIndex;
|
||||||
|
for (let index = 0; index < eventsList.length; index++) {
|
||||||
|
const eventDate = eventsList[index].start;
|
||||||
|
if (selectedDate === eventDate) {
|
||||||
|
// If it's the case, set current index to selectedEventIndex
|
||||||
|
selectedEventIndex = index;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (selectedEventIndex !== undefined) {
|
||||||
|
// If selectedEventIndex exists, remove the corresponding event from the list
|
||||||
|
eventsList.splice(selectedEventIndex, 1);
|
||||||
|
updatedEvents = [...eventsList];
|
||||||
|
} else {
|
||||||
|
// Add selected event to the list
|
||||||
|
updatedEvents = [...eventsList, selectedEvent];
|
||||||
|
}
|
||||||
|
// setEventsList(updatedEvents);
|
||||||
|
} else {
|
||||||
|
// When a range of dates is selected
|
||||||
|
let currentDate = start;
|
||||||
|
while (!(+end === +currentDate)) {
|
||||||
|
// Add currentDate to the datesList
|
||||||
|
datesList.add(currentDate.toISODate());
|
||||||
|
let newDate = currentDate.plus({ days: 1 });
|
||||||
|
currentDate = newDate;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add each date from the eventsList to datesList
|
||||||
|
eventsList.forEach((event) => {
|
||||||
|
datesList.add(event.start);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Create events from datesList and add them to updatedEvents
|
||||||
|
datesList.forEach((date) => {
|
||||||
|
updatedEvents.push({ start: date, display: 'background' });
|
||||||
|
});
|
||||||
|
}
|
||||||
|
setEventsList(updatedEvents);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleClear = () => {
|
||||||
|
setEventsList([]);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleDelete = (date) => {
|
||||||
|
let currentEvent = {
|
||||||
|
start: date.toFormat('yyyy-MM-dd'),
|
||||||
|
display: 'background',
|
||||||
|
};
|
||||||
|
|
||||||
|
// Find the event corresponding to the date
|
||||||
|
let selectedEventIndex;
|
||||||
|
for (let index = 0; index < eventsList.length; index++) {
|
||||||
|
const eventDate = eventsList[index].start;
|
||||||
|
if (currentEvent.start === eventDate) {
|
||||||
|
// When it's the case, set current index to selectedEventIndex
|
||||||
|
selectedEventIndex = index;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Create updated eventsList
|
||||||
|
eventsList.splice(selectedEventIndex, 1);
|
||||||
|
let updatedEvents = [...eventsList];
|
||||||
|
// Update the eventsList
|
||||||
|
setEventsList(updatedEvents);
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Container>
|
<Container>
|
||||||
<h3>Schedule a meeting</h3>
|
<h3>Schedule a meeting</h3>
|
||||||
@ -30,15 +126,45 @@ export default function Schedule() {
|
|||||||
<DurationSelector />
|
<DurationSelector />
|
||||||
</FormGroup>
|
</FormGroup>
|
||||||
<FormGroup>
|
<FormGroup>
|
||||||
<DaySelector />
|
<DaySelector
|
||||||
</FormGroup>
|
eventsList={eventsList}
|
||||||
<FormGroup>
|
handleSelect={handleSelect}
|
||||||
<SelectedDates />
|
handleClear={handleClear}
|
||||||
</FormGroup>
|
/>
|
||||||
<FormGroup>
|
|
||||||
<Button>Confirm dates</Button>
|
|
||||||
</FormGroup>
|
</FormGroup>
|
||||||
|
{datesList.length > 0 && (
|
||||||
|
<FormGroup>
|
||||||
|
<SelectedDates
|
||||||
|
datesList={datesList}
|
||||||
|
eventsList={eventsList}
|
||||||
|
handleDelete={handleDelete}
|
||||||
|
/>
|
||||||
|
<Button>Confirm dates</Button>
|
||||||
|
</FormGroup>
|
||||||
|
)}
|
||||||
</Form>
|
</Form>
|
||||||
</Container>
|
</Container>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Convert events to Luxon Datetime objects
|
||||||
|
const eventsToDates = (events) => {
|
||||||
|
let dates = [];
|
||||||
|
events.forEach((event) => {
|
||||||
|
dates.push(DateTime.fromFormat(event.start, 'yyyy-MM-dd'));
|
||||||
|
});
|
||||||
|
return dates;
|
||||||
|
};
|
||||||
|
|
||||||
|
// Convert Luxon Datetime objects to events
|
||||||
|
const datesToEvents = (dates) => {
|
||||||
|
let events = [];
|
||||||
|
dates.forEach((date) => {
|
||||||
|
let event = {
|
||||||
|
start: date.toFormat('yyyy-MM-dd'),
|
||||||
|
display: 'background',
|
||||||
|
};
|
||||||
|
events.push(event);
|
||||||
|
});
|
||||||
|
return events;
|
||||||
|
};
|
||||||
|
Loading…
x
Reference in New Issue
Block a user