2022-01-06 01:26:15 +00:00
|
|
|
import React, { useContext, useEffect, useRef, useState } from "react";
|
|
|
|
|
|
|
|
import { makeStyles } from "@material-ui/core/styles";
|
2023-07-12 14:54:29 +00:00
|
|
|
import { IconButton } from "@mui/material";
|
2022-01-06 01:26:15 +00:00
|
|
|
import Paper from "@material-ui/core/Paper";
|
|
|
|
import InputBase from "@material-ui/core/InputBase";
|
|
|
|
import Tabs from "@material-ui/core/Tabs";
|
|
|
|
import Tab from "@material-ui/core/Tab";
|
|
|
|
import Badge from "@material-ui/core/Badge";
|
2023-07-12 14:54:29 +00:00
|
|
|
|
|
|
|
import Tooltip from "@material-ui/core/Tooltip";
|
|
|
|
|
|
|
|
|
|
|
|
import SearchIcon from "@material-ui/icons/Search";
|
2022-01-06 01:26:15 +00:00
|
|
|
import MoveToInboxIcon from "@material-ui/icons/MoveToInbox";
|
|
|
|
import CheckBoxIcon from "@material-ui/icons/CheckBox";
|
2023-07-12 14:54:29 +00:00
|
|
|
import MenuIcon from "@material-ui/icons/Menu";
|
|
|
|
import FindInPageIcon from '@material-ui/icons/FindInPage';
|
2022-01-06 01:26:15 +00:00
|
|
|
|
|
|
|
import FormControlLabel from "@material-ui/core/FormControlLabel";
|
|
|
|
import Switch from "@material-ui/core/Switch";
|
|
|
|
|
|
|
|
import NewTicketModal from "../NewTicketModal";
|
|
|
|
import TicketsList from "../TicketsList";
|
|
|
|
import TabPanel from "../TabPanel";
|
|
|
|
|
|
|
|
import { i18n } from "../../translate/i18n";
|
|
|
|
import { AuthContext } from "../../context/Auth/AuthContext";
|
|
|
|
import { Can } from "../Can";
|
|
|
|
import TicketsQueueSelect from "../TicketsQueueSelect";
|
|
|
|
import { Button } from "@material-ui/core";
|
|
|
|
|
2022-10-25 14:16:36 +00:00
|
|
|
import { TabTicketContext } from "../../context/TabTicketHeaderOption/TabTicketHeaderOption";
|
|
|
|
|
|
|
|
import { SearchTicketContext } from "../../context/SearchTicket/SearchTicket";
|
|
|
|
|
2022-01-06 01:26:15 +00:00
|
|
|
const useStyles = makeStyles((theme) => ({
|
|
|
|
ticketsWrapper: {
|
|
|
|
position: "relative",
|
|
|
|
display: "flex",
|
|
|
|
height: "100%",
|
|
|
|
flexDirection: "column",
|
|
|
|
overflow: "hidden",
|
|
|
|
borderTopRightRadius: 0,
|
|
|
|
borderBottomRightRadius: 0,
|
|
|
|
},
|
|
|
|
|
|
|
|
tabsHeader: {
|
|
|
|
flex: "none",
|
|
|
|
backgroundColor: "#eee",
|
|
|
|
},
|
|
|
|
|
|
|
|
settingsIcon: {
|
|
|
|
alignSelf: "center",
|
|
|
|
marginLeft: "auto",
|
|
|
|
padding: 8,
|
|
|
|
},
|
|
|
|
|
|
|
|
tab: {
|
|
|
|
minWidth: 120,
|
|
|
|
width: 120,
|
|
|
|
},
|
|
|
|
|
|
|
|
ticketOptionsBox: {
|
|
|
|
display: "flex",
|
|
|
|
justifyContent: "space-between",
|
|
|
|
alignItems: "center",
|
|
|
|
background: "#fafafa",
|
|
|
|
padding: theme.spacing(1),
|
|
|
|
},
|
|
|
|
|
|
|
|
serachInputWrapper: {
|
2023-07-12 14:54:29 +00:00
|
|
|
flex: 1,
|
|
|
|
display: "flex",
|
|
|
|
flexDirection: "column",
|
|
|
|
gap: "10px",
|
|
|
|
borderRadius: 40,
|
|
|
|
padding: 4,
|
|
|
|
marginRight: theme.spacing(1),
|
|
|
|
},
|
|
|
|
|
|
|
|
searchInputHeader: {
|
|
|
|
flex: 1,
|
|
|
|
background: "#fff",
|
|
|
|
display: "flex",
|
|
|
|
borderRadius: 40,
|
|
|
|
padding: 4,
|
|
|
|
marginRight: theme.spacing(1),
|
|
|
|
},
|
|
|
|
|
|
|
|
searchContentInput: {
|
2022-01-06 01:26:15 +00:00
|
|
|
flex: 1,
|
|
|
|
background: "#fff",
|
|
|
|
display: "flex",
|
|
|
|
borderRadius: 40,
|
|
|
|
padding: 4,
|
|
|
|
marginRight: theme.spacing(1),
|
|
|
|
},
|
|
|
|
|
|
|
|
searchIcon: {
|
|
|
|
color: "grey",
|
|
|
|
marginLeft: 6,
|
|
|
|
marginRight: 6,
|
|
|
|
alignSelf: "center",
|
|
|
|
},
|
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
menuSearch: {
|
|
|
|
color: "grey",
|
|
|
|
alignSelf: "center",
|
|
|
|
},
|
|
|
|
|
2022-01-06 01:26:15 +00:00
|
|
|
searchInput: {
|
|
|
|
flex: 1,
|
|
|
|
border: "none",
|
|
|
|
borderRadius: 30,
|
|
|
|
},
|
|
|
|
|
|
|
|
badge: {
|
|
|
|
right: "-10px",
|
|
|
|
},
|
|
|
|
show: {
|
|
|
|
display: "block",
|
|
|
|
},
|
|
|
|
hide: {
|
|
|
|
display: "none !important",
|
|
|
|
},
|
|
|
|
}));
|
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
const DEFAULT_SEARCH_PARAM = { searchParam: "", searchParamContent: "" }
|
|
|
|
|
2022-01-06 01:26:15 +00:00
|
|
|
const TicketsManager = () => {
|
2022-10-25 14:16:36 +00:00
|
|
|
|
|
|
|
const { tabOption, setTabOption } = useContext(TabTicketContext);
|
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
const { setSearchTicket } = useContext(SearchTicketContext)
|
2022-10-25 14:16:36 +00:00
|
|
|
|
2022-01-06 01:26:15 +00:00
|
|
|
const classes = useStyles();
|
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
const [searchParam, setSearchParam] = useState(DEFAULT_SEARCH_PARAM);
|
2022-01-06 01:26:15 +00:00
|
|
|
const [tab, setTab] = useState("open");
|
|
|
|
const [tabOpen, setTabOpen] = useState("open");
|
|
|
|
const [newTicketModalOpen, setNewTicketModalOpen] = useState(false);
|
|
|
|
const [showAllTickets, setShowAllTickets] = useState(false);
|
|
|
|
const { user } = useContext(AuthContext);
|
|
|
|
|
|
|
|
const [openCount, setOpenCount] = useState(0);
|
|
|
|
const [pendingCount, setPendingCount] = useState(0);
|
|
|
|
|
|
|
|
const userQueueIds = user.queues.map((q) => q.id);
|
|
|
|
const [selectedQueueIds, setSelectedQueueIds] = useState(userQueueIds || []);
|
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
const [showContentSearch, setShowContentSearch] = useState(false)
|
|
|
|
const searchInputRef = useRef();
|
|
|
|
const searchContentInputRef = useRef();
|
2022-10-25 14:16:36 +00:00
|
|
|
const [inputSearch, setInputSearch] = useState('');
|
2023-07-12 14:54:29 +00:00
|
|
|
const [inputContentSearch, setInputContentSearch] = useState("")
|
|
|
|
|
|
|
|
const [openTooltipSearch, setOpenTooltipSearch] = useState(false)
|
|
|
|
|
|
|
|
let searchTimeout;
|
|
|
|
let searchContentTimeout;
|
2022-10-25 14:16:36 +00:00
|
|
|
|
2022-01-06 01:26:15 +00:00
|
|
|
useEffect(() => {
|
2024-02-05 15:29:49 +00:00
|
|
|
if (user.profile.toUpperCase() === "ADMIN" ||
|
|
|
|
user.profile.toUpperCase() === "SUPERVISOR" ||
|
|
|
|
user.profile.toUpperCase() === "MASTER") {
|
2022-01-06 01:26:15 +00:00
|
|
|
setShowAllTickets(true);
|
|
|
|
}
|
|
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (tab === "search") {
|
|
|
|
searchInputRef.current.focus();
|
|
|
|
}
|
2022-10-25 14:16:36 +00:00
|
|
|
|
|
|
|
setTabOption(tab)
|
|
|
|
|
|
|
|
}, [tab, setTabOption]);
|
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
useEffect(() => {
|
|
|
|
|
|
|
|
// clearTimeout(searchContentTimeout);
|
|
|
|
|
|
|
|
// setSearchParam(prev => ({ ...prev, searchParamContent: "" }))
|
|
|
|
|
|
|
|
if (!inputContentSearch) return
|
|
|
|
|
|
|
|
if (!searchContentTimeout) return
|
|
|
|
|
|
|
|
// searchContentTimeout = setTimeout(() => {
|
|
|
|
|
|
|
|
// setSearchParam(prev => ({ ...prev, searchParamContent: inputContentSearch }));
|
|
|
|
|
|
|
|
// }, 500);
|
|
|
|
|
|
|
|
clearTimeout(searchContentTimeout);
|
|
|
|
|
|
|
|
setSearchParam(prev => ({ ...prev, searchParamContent: "" }))
|
|
|
|
|
|
|
|
|
|
|
|
}, [inputContentSearch, searchContentTimeout]);
|
|
|
|
|
2022-10-25 14:16:36 +00:00
|
|
|
useEffect(() => {
|
|
|
|
|
|
|
|
|
|
|
|
if (tabOption === 'open') {
|
|
|
|
|
|
|
|
setTabOption('')
|
2023-07-12 14:54:29 +00:00
|
|
|
setSearchParam(DEFAULT_SEARCH_PARAM);
|
2022-10-25 14:16:36 +00:00
|
|
|
setInputSearch('');
|
2023-07-12 14:54:29 +00:00
|
|
|
setInputContentSearch('')
|
2022-10-25 14:16:36 +00:00
|
|
|
setTab("open");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
}, [tabOption, setTabOption])
|
2022-01-06 01:26:15 +00:00
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
|
2022-10-25 14:16:36 +00:00
|
|
|
const removeExtraSpace = (str) => {
|
|
|
|
|
|
|
|
str = str.replace(/^\s+/g, '')
|
|
|
|
|
|
|
|
return str.replace(/\s+/g, ' ')
|
|
|
|
}
|
|
|
|
|
2022-01-06 01:26:15 +00:00
|
|
|
const handleSearch = (e) => {
|
2022-10-25 14:16:36 +00:00
|
|
|
let searchedTerm = e.target.value.toLowerCase()
|
|
|
|
|
|
|
|
setInputSearch(removeExtraSpace(searchedTerm))
|
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
setSearchTicket(searchParam.searchParam)
|
2022-01-06 01:26:15 +00:00
|
|
|
|
|
|
|
clearTimeout(searchTimeout);
|
|
|
|
|
|
|
|
if (searchedTerm === "") {
|
2023-07-12 14:54:29 +00:00
|
|
|
setSearchParam(prev => ({ ...prev, searchParam: searchedTerm }))
|
2022-10-25 14:16:36 +00:00
|
|
|
setInputSearch(searchedTerm)
|
2023-07-12 14:54:29 +00:00
|
|
|
setShowContentSearch(false)
|
2022-01-06 01:26:15 +00:00
|
|
|
setTab("open");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
if (searchedTerm.length < 4) {
|
|
|
|
setSearchParam(prev => ({ ...prev, searchParamContent: "" }))
|
|
|
|
setInputContentSearch('')
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-06 01:26:15 +00:00
|
|
|
searchTimeout = setTimeout(() => {
|
2022-10-25 14:16:36 +00:00
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
setSearchParam(prev => ({ ...prev, searchParam: searchedTerm }));
|
2022-10-25 14:16:36 +00:00
|
|
|
|
|
|
|
}, 500);
|
2022-01-06 01:26:15 +00:00
|
|
|
};
|
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
const handleContentSearch = e => {
|
|
|
|
|
|
|
|
let searchedContentText = removeExtraSpace(e.target.value.toLowerCase())
|
|
|
|
|
|
|
|
setInputContentSearch(searchedContentText)
|
|
|
|
|
|
|
|
searchContentTimeout = setTimeout(() => {
|
|
|
|
|
|
|
|
setSearchParam(prev => ({ ...prev, searchParamContent: searchedContentText }));
|
|
|
|
|
|
|
|
}, 500);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleOpenTooltipSearch = () => {
|
|
|
|
if (searchParam.searchParam.length < 4) {
|
|
|
|
setOpenTooltipSearch(true)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const handleCloseTooltipSearch = () => {
|
|
|
|
setOpenTooltipSearch(false)
|
|
|
|
if (searchParam.searchParam.length < 4) {
|
|
|
|
searchInputRef.current.focus()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-06 01:26:15 +00:00
|
|
|
const handleChangeTab = (e, newValue) => {
|
|
|
|
setTab(newValue);
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleChangeTabOpen = (e, newValue) => {
|
|
|
|
setTabOpen(newValue);
|
|
|
|
};
|
|
|
|
|
|
|
|
const applyPanelStyle = (status) => {
|
|
|
|
if (tabOpen !== status) {
|
|
|
|
return { width: 0, height: 0 };
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Paper elevation={0} variant="outlined" className={classes.ticketsWrapper}>
|
|
|
|
<NewTicketModal
|
|
|
|
modalOpen={newTicketModalOpen}
|
|
|
|
onClose={(e) => setNewTicketModalOpen(false)}
|
|
|
|
/>
|
|
|
|
<Paper elevation={0} square className={classes.tabsHeader}>
|
|
|
|
<Tabs
|
|
|
|
value={tab}
|
|
|
|
onChange={handleChangeTab}
|
|
|
|
variant="fullWidth"
|
|
|
|
indicatorColor="primary"
|
|
|
|
textColor="primary"
|
|
|
|
aria-label="icon label tabs example"
|
|
|
|
>
|
|
|
|
<Tab
|
|
|
|
value={"open"}
|
|
|
|
icon={<MoveToInboxIcon />}
|
|
|
|
label={i18n.t("tickets.tabs.open.title")}
|
|
|
|
classes={{ root: classes.tab }}
|
|
|
|
/>
|
|
|
|
<Tab
|
|
|
|
value={"closed"}
|
|
|
|
icon={<CheckBoxIcon />}
|
|
|
|
label={i18n.t("tickets.tabs.closed.title")}
|
|
|
|
classes={{ root: classes.tab }}
|
|
|
|
/>
|
|
|
|
<Tab
|
|
|
|
value={"search"}
|
|
|
|
icon={<SearchIcon />}
|
|
|
|
label={i18n.t("tickets.tabs.search.title")}
|
|
|
|
classes={{ root: classes.tab }}
|
|
|
|
/>
|
|
|
|
</Tabs>
|
|
|
|
</Paper>
|
|
|
|
<Paper square elevation={0} className={classes.ticketOptionsBox}>
|
|
|
|
{tab === "search" ? (
|
|
|
|
<div className={classes.serachInputWrapper}>
|
2023-07-12 14:54:29 +00:00
|
|
|
<div className={classes.searchInputHeader}>
|
|
|
|
<SearchIcon className={classes.searchIcon} />
|
|
|
|
<InputBase
|
|
|
|
className={classes.searchInput}
|
|
|
|
inputRef={searchInputRef}
|
|
|
|
placeholder={i18n.t("tickets.search.placeholder")}
|
|
|
|
type="search"
|
|
|
|
value={inputSearch}
|
|
|
|
onChange={handleSearch}
|
|
|
|
/>
|
|
|
|
{/* <IconButton onClick={() => setShowContentSearch(prev => !prev)}>
|
|
|
|
<MenuIcon className={classes.menuSearch} />
|
|
|
|
</IconButton> */}
|
|
|
|
<Tooltip
|
|
|
|
open={openTooltipSearch}
|
|
|
|
onOpen={() => handleOpenTooltipSearch()}
|
|
|
|
onClose={() => handleCloseTooltipSearch()}
|
|
|
|
title="Digite pelo menos 4 caracteres"
|
|
|
|
arrow>
|
|
|
|
<span>
|
|
|
|
<IconButton
|
|
|
|
disabled={searchParam.searchParam.length < 4}
|
|
|
|
onClick={() => setShowContentSearch(prev => !prev)}
|
|
|
|
>
|
|
|
|
<MenuIcon className={classes.menuSearch} />
|
|
|
|
</IconButton>
|
|
|
|
</span>
|
|
|
|
</Tooltip>
|
|
|
|
</div>
|
|
|
|
{
|
|
|
|
// showContentSearch ?
|
|
|
|
(showContentSearch && searchParam.searchParam.length >= 4) ?
|
|
|
|
(<div className={classes.searchContentInput}>
|
|
|
|
<FindInPageIcon className={classes.searchIcon} />
|
|
|
|
<InputBase
|
|
|
|
className={classes.searchInput}
|
|
|
|
inputRef={searchContentInputRef}
|
|
|
|
placeholder={i18n.t("Busca por conteúdo")}
|
|
|
|
type="search"
|
|
|
|
value={inputContentSearch}
|
|
|
|
onChange={(e) => handleContentSearch(e)}
|
|
|
|
/>
|
|
|
|
</div>) : null
|
|
|
|
}
|
2022-01-06 01:26:15 +00:00
|
|
|
</div>
|
|
|
|
) : (
|
|
|
|
<>
|
|
|
|
<Button
|
|
|
|
variant="outlined"
|
|
|
|
color="primary"
|
|
|
|
onClick={() => setNewTicketModalOpen(true)}
|
|
|
|
>
|
|
|
|
{i18n.t("ticketsManager.buttons.newTicket")}
|
|
|
|
</Button>
|
|
|
|
<Can
|
|
|
|
role={user.profile}
|
|
|
|
perform="tickets-manager:showall"
|
|
|
|
yes={() => (
|
|
|
|
<FormControlLabel
|
|
|
|
label={i18n.t("tickets.buttons.showAll")}
|
|
|
|
labelPlacement="start"
|
|
|
|
control={
|
|
|
|
<Switch
|
|
|
|
size="small"
|
|
|
|
checked={showAllTickets}
|
|
|
|
onChange={() =>
|
|
|
|
setShowAllTickets((prevState) => !prevState)
|
|
|
|
}
|
|
|
|
name="showAllTickets"
|
|
|
|
color="primary"
|
|
|
|
/>
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
)}
|
|
|
|
/>
|
|
|
|
</>
|
|
|
|
)}
|
|
|
|
<TicketsQueueSelect
|
|
|
|
style={{ marginLeft: 6 }}
|
|
|
|
selectedQueueIds={selectedQueueIds}
|
|
|
|
userQueues={user?.queues}
|
|
|
|
onChange={(values) => setSelectedQueueIds(values)}
|
|
|
|
/>
|
|
|
|
</Paper>
|
|
|
|
<TabPanel value={tab} name="open" className={classes.ticketsWrapper}>
|
|
|
|
<Tabs
|
|
|
|
value={tabOpen}
|
|
|
|
onChange={handleChangeTabOpen}
|
|
|
|
indicatorColor="primary"
|
|
|
|
textColor="primary"
|
|
|
|
variant="fullWidth"
|
|
|
|
>
|
|
|
|
<Tab
|
|
|
|
label={
|
|
|
|
<Badge
|
|
|
|
className={classes.badge}
|
|
|
|
badgeContent={openCount}
|
|
|
|
color="primary"
|
|
|
|
>
|
|
|
|
{i18n.t("ticketsList.assignedHeader")}
|
|
|
|
</Badge>
|
|
|
|
}
|
|
|
|
value={"open"}
|
|
|
|
/>
|
|
|
|
<Tab
|
|
|
|
label={
|
|
|
|
<Badge
|
|
|
|
className={classes.badge}
|
|
|
|
badgeContent={pendingCount}
|
|
|
|
color="secondary"
|
|
|
|
>
|
|
|
|
{i18n.t("ticketsList.pendingHeader")}
|
|
|
|
</Badge>
|
|
|
|
}
|
|
|
|
value={"pending"}
|
|
|
|
/>
|
|
|
|
</Tabs>
|
|
|
|
<Paper className={classes.ticketsWrapper}>
|
|
|
|
<TicketsList
|
2022-08-01 19:33:24 +00:00
|
|
|
status={"open"}
|
2022-01-06 01:26:15 +00:00
|
|
|
showAll={showAllTickets}
|
|
|
|
selectedQueueIds={selectedQueueIds}
|
|
|
|
updateCount={(val) => setOpenCount(val)}
|
|
|
|
style={applyPanelStyle("open")}
|
|
|
|
/>
|
|
|
|
<TicketsList
|
|
|
|
status="pending"
|
|
|
|
selectedQueueIds={selectedQueueIds}
|
|
|
|
updateCount={(val) => setPendingCount(val)}
|
|
|
|
style={applyPanelStyle("pending")}
|
2022-10-25 14:16:36 +00:00
|
|
|
/>
|
2022-01-06 01:26:15 +00:00
|
|
|
</Paper>
|
|
|
|
</TabPanel>
|
|
|
|
<TabPanel value={tab} name="closed" className={classes.ticketsWrapper}>
|
|
|
|
<TicketsList
|
|
|
|
status="closed"
|
|
|
|
showAll={true}
|
|
|
|
selectedQueueIds={selectedQueueIds}
|
|
|
|
/>
|
|
|
|
</TabPanel>
|
|
|
|
<TabPanel value={tab} name="search" className={classes.ticketsWrapper}>
|
2022-10-25 14:16:36 +00:00
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
|
|
|
|
<TicketsList
|
|
|
|
searchParam={searchParam.searchParam}
|
|
|
|
searchParamContent={searchParam.searchParamContent}
|
|
|
|
tab={tab}
|
|
|
|
showAll={true}
|
|
|
|
selectedQueueIds={selectedQueueIds}
|
|
|
|
/>
|
|
|
|
|
2022-01-06 01:26:15 +00:00
|
|
|
</TabPanel>
|
|
|
|
</Paper>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2023-07-12 14:54:29 +00:00
|
|
|
export default TicketsManager;
|