| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325 |
- import React, { useState, useEffect, useReducer, useCallback, useContext } from "react";
- import { toast } from "react-toastify";
- import { useHistory } from "react-router-dom";
- import { makeStyles } from "@material-ui/core/styles";
- import Paper from "@material-ui/core/Paper";
- import Button from "@material-ui/core/Button";
- import TextField from "@material-ui/core/TextField";
- import InputAdornment from "@material-ui/core/InputAdornment";
- import MainContainer from "../../components/MainContainer";
- import MainHeader from "../../components/MainHeader";
- import Title from "../../components/Title";
- import api from "../../services/api";
- import { i18n } from "../../translate/i18n";
- import MainHeaderButtonsWrapper from "../../components/MainHeaderButtonsWrapper";
- import ScheduleModal from "../../components/ScheduleModal";
- import ConfirmationModal from "../../components/ConfirmationModal";
- import toastError from "../../errors/toastError";
- import moment from "moment";
- import { SocketContext } from "../../context/Socket/SocketContext";
- import { AuthContext } from "../../context/Auth/AuthContext";
- import usePlans from "../../hooks/usePlans";
- import { Calendar, momentLocalizer } from "react-big-calendar";
- import "moment/locale/pt-br";
- import "react-big-calendar/lib/css/react-big-calendar.css";
- import SearchIcon from "@material-ui/icons/Search";
- import DeleteOutlineIcon from "@material-ui/icons/DeleteOutline";
- import EditIcon from "@material-ui/icons/Edit";
- import "./Schedules.css"; // Importe o arquivo CSS
- import { createMomentLocalizer } from "../../translate/calendar-locale";
- // Defina a função getUrlParam antes de usá-la
- function getUrlParam(paramName) {
- const searchParams = new URLSearchParams(window.location.search);
- return searchParams.get(paramName);
- }
- const eventTitleStyle = {
- fontSize: "14px", // Defina um tamanho de fonte menor
- overflow: "hidden", // Oculte qualquer conteúdo excedente
- whiteSpace: "nowrap", // Evite a quebra de linha do texto
- textOverflow: "ellipsis", // Exiba "..." se o texto for muito longo
- };
- var defaultMessages = {
- date: i18n.t("schedules.messages.date"),
- time: i18n.t("schedules.messages.time"),
- event: i18n.t("schedules.messages.event"),
- allDay: i18n.t("schedules.messages.allDay"),
- week: i18n.t("schedules.messages.week"),
- work_week: i18n.t("schedules.messages.work_week"),
- day: i18n.t("schedules.messages.day"),
- month: i18n.t("schedules.messages.month"),
- previous: i18n.t("schedules.messages.previous"),
- next: i18n.t("schedules.messages.next"),
- yesterday: i18n.t("schedules.messages.yesterday"),
- tomorrow: i18n.t("schedules.messages.tomorrow"),
- today: i18n.t("schedules.messages.today"),
- agenda: i18n.t("schedules.messages.agenda"),
- noEventsInRange: i18n.t("schedules.messages.noEventsInRange"),
- showMore: function showMore(total) {
- return "+" + total + " " + i18n.t("schedules.messages.showMore");
- }
- };
- const reducer = (state, action) => {
- if (action.type === "LOAD_SCHEDULES") {
- return [...state, ...action.payload];
- }
- if (action.type === "UPDATE_SCHEDULES") {
- const schedule = action.payload;
- const scheduleIndex = state.findIndex((s) => s.id === schedule.id);
- if (scheduleIndex !== -1) {
- state[scheduleIndex] = schedule;
- return [...state];
- } else {
- return [schedule, ...state];
- }
- }
- if (action.type === "DELETE_SCHEDULE") {
- const scheduleId = action.payload;
- return state.filter((s) => s.id !== scheduleId);
- }
- if (action.type === "RESET") {
- return [];
- }
- return state;
- };
- const useStyles = makeStyles((theme) => ({
- mainPaper: {
- flex: 1,
- padding: theme.spacing(1),
- overflowY: "scroll",
- ...theme.scrollbarStyles,
- },
- }));
- const Schedules = () => {
- const localizer = createMomentLocalizer();
- const classes = useStyles();
- const history = useHistory();
- const { user } = useContext(AuthContext);
- const [loading, setLoading] = useState(false);
- const [pageNumber, setPageNumber] = useState(1);
- const [hasMore, setHasMore] = useState(false);
- const [selectedSchedule, setSelectedSchedule] = useState(null);
- const [deletingSchedule, setDeletingSchedule] = useState(null);
- const [confirmModalOpen, setConfirmModalOpen] = useState(false);
- const [searchParam, setSearchParam] = useState("");
- const [schedules, dispatch] = useReducer(reducer, []);
- const [scheduleModalOpen, setScheduleModalOpen] = useState(false);
- const [contactId, setContactId] = useState(+getUrlParam("contactId"));
- const fetchSchedules = useCallback(async () => {
- try {
- const { data } = await api.get("/schedules/", {
- params: { searchParam, pageNumber },
- });
- dispatch({ type: "LOAD_SCHEDULES", payload: data.schedules });
- setHasMore(data.hasMore);
- setLoading(false);
- } catch (err) {
- toastError(err);
- }
- }, [searchParam, pageNumber]);
- const handleOpenScheduleModalFromContactId = useCallback(() => {
- if (contactId) {
- handleOpenScheduleModal();
- }
- }, [contactId]);
- const socketManager = useContext(SocketContext);
- useEffect(() => {
- dispatch({ type: "RESET" });
- setPageNumber(1);
- }, [searchParam]);
- useEffect(() => {
- setLoading(true);
- const delayDebounceFn = setTimeout(() => {
- fetchSchedules();
- }, 500);
- return () => clearTimeout(delayDebounceFn);
- }, [
- searchParam,
- pageNumber,
- contactId,
- fetchSchedules,
- handleOpenScheduleModalFromContactId,
- ]);
- useEffect(() => {
- handleOpenScheduleModalFromContactId();
- const socket = socketManager.getSocket(user.companyId);
- socket.on(`company${user.companyId}-schedule`, (data) => {
- if (data.action === "update" || data.action === "create") {
- dispatch({ type: "UPDATE_SCHEDULES", payload: data.schedule });
- }
- if (data.action === "delete") {
- dispatch({ type: "DELETE_SCHEDULE", payload: +data.scheduleId });
- }
- });
- return () => {
- socket.disconnect();
- };
- }, [handleOpenScheduleModalFromContactId, socketManager, user]);
- const cleanContact = () => {
- setContactId("");
- };
- const handleOpenScheduleModal = () => {
- setSelectedSchedule(null);
- setScheduleModalOpen(true);
- };
- const handleCloseScheduleModal = () => {
- setSelectedSchedule(null);
- setScheduleModalOpen(false);
- };
- const handleSearch = (event) => {
- setSearchParam(event.target.value.toLowerCase());
- };
- const handleEditSchedule = (schedule) => {
- setSelectedSchedule(schedule);
- setScheduleModalOpen(true);
- };
- const handleDeleteSchedule = async (scheduleId) => {
- try {
- await api.delete(`/schedules/${scheduleId}`);
- toast.success(i18n.t("schedules.toasts.deleted"));
- } catch (err) {
- toastError(err);
- }
- setDeletingSchedule(null);
- setSearchParam("");
- setPageNumber(1);
- dispatch({ type: "RESET" });
- setPageNumber(1);
- await fetchSchedules();
- };
- const loadMore = () => {
- setPageNumber((prevState) => prevState + 1);
- };
- const handleScroll = (e) => {
- if (!hasMore || loading) return;
- const { scrollTop, scrollHeight, clientHeight } = e.currentTarget;
- if (scrollHeight - (scrollTop + 100) < clientHeight) {
- loadMore();
- }
- };
- const truncate = (str, len) => {
- if (str.length > len) {
- return str.substring(0, len) + "...";
- }
- return str;
- };
- return (
- <MainContainer>
- <ConfirmationModal
- title={
- deletingSchedule &&
- `${i18n.t("schedules.confirmationModal.deleteTitle")}`
- }
- open={confirmModalOpen}
- onClose={() => setConfirmModalOpen(false)}
- onConfirm={() => handleDeleteSchedule(deletingSchedule.id)}
- >
- {i18n.t("schedules.confirmationModal.deleteMessage")}
- </ConfirmationModal>
- <ScheduleModal
- open={scheduleModalOpen}
- onClose={handleCloseScheduleModal}
- reload={fetchSchedules}
- aria-labelledby="form-dialog-title"
- scheduleId={selectedSchedule && selectedSchedule.id}
- contactId={contactId}
- cleanContact={cleanContact}
- />
- <MainHeader>
- <Title>{i18n.t("schedules.title")} ({schedules.length})</Title>
- <MainHeaderButtonsWrapper>
- <TextField
- placeholder={i18n.t("contacts.searchPlaceholder")}
- type="search"
- value={searchParam}
- onChange={handleSearch}
- InputProps={{
- startAdornment: (
- <InputAdornment position="start">
- <SearchIcon style={{ color: "gray" }} />
- </InputAdornment>
- ),
- }}
- />
- <Button
- variant="contained"
- color="primary"
- onClick={handleOpenScheduleModal}
- >
- {i18n.t("schedules.buttons.add")}
- </Button>
- </MainHeaderButtonsWrapper>
- </MainHeader>
- <Paper className={classes.mainPaper} variant="outlined" onScroll={handleScroll}>
- <Calendar
- messages={defaultMessages}
- formats={{
- agendaDateFormat: "DD/MM ddd",
- weekdayFormat: "dddd"
- }}
- localizer={localizer}
- events={schedules.map((schedule) => ({
- title: (
- <div className="event-container">
- <div style={eventTitleStyle}>{schedule.contact.name}</div>
- <DeleteOutlineIcon
- onClick={() => handleDeleteSchedule(schedule.id)}
- className="delete-icon"
- />
- <EditIcon
- onClick={() => {
- handleEditSchedule(schedule);
- setScheduleModalOpen(true);
- }}
- className="edit-icon"
- />
- </div>
- ),
- start: new Date(schedule.sendAt),
- end: new Date(schedule.sendAt),
- }))}
- startAccessor="start"
- endAccessor="end"
- style={{ height: 500 }}
- />
- </Paper>
- </MainContainer>
- );
- };
- export default Schedules;
|