feat: finished queue filter on tickets list

This commit is contained in:
canove
2021-01-13 19:43:50 -03:00
parent c61c993572
commit 2c2a39ee1f
6 changed files with 124 additions and 134 deletions

View File

@@ -63,7 +63,7 @@ export const store = async (req: Request, res: Response): Promise<Response> => {
const io = getIO(); const io = getIO();
io.to(ticket.status).emit("ticket", { io.to(ticket.status).emit("ticket", {
action: "create", action: "update",
ticket ticket
}); });
@@ -100,7 +100,7 @@ export const update = async (
} }
io.to(ticket.status).to("notification").to(ticketId).emit("ticket", { io.to(ticket.status).to("notification").to(ticketId).emit("ticket", {
action: "updateStatus", action: "update",
ticket ticket
}); });

View File

@@ -34,12 +34,6 @@ const ListTicketsService = async ({
userId, userId,
withUnreadMessages withUnreadMessages
}: Request): Promise<Response> => { }: Request): Promise<Response> => {
// const user = await ShowUserService(userId);
// const userQueueIds = user.queues.map(queue => queue.id);
// console.log(userQueueIds);
let whereCondition: Filterable["where"] = { let whereCondition: Filterable["where"] = {
[Op.or]: [{ userId }, { status: "pending" }], [Op.or]: [{ userId }, { status: "pending" }],
queueId: { [Op.or]: [queueIds, null] } queueId: { [Op.or]: [queueIds, null] }
@@ -60,7 +54,7 @@ const ListTicketsService = async ({
]; ];
if (showAll === "true") { if (showAll === "true") {
whereCondition = {}; whereCondition = { queueId: { [Op.or]: [queueIds, null] } };
} }
if (status) { if (status) {
@@ -92,6 +86,7 @@ const ListTicketsService = async ({
]; ];
whereCondition = { whereCondition = {
...whereCondition,
[Op.or]: [ [Op.or]: [
{ {
"$contact.name$": where( "$contact.name$": where(

View File

@@ -90,7 +90,7 @@ const Ticket = () => {
socket.on("connect", () => socket.emit("joinChatBox", ticketId)); socket.on("connect", () => socket.emit("joinChatBox", ticketId));
socket.on("ticket", data => { socket.on("ticket", data => {
if (data.action === "updateStatus") { if (data.action === "update") {
setTicket(data.ticket); setTicket(data.ticket);
} }

View File

@@ -7,10 +7,8 @@ import Paper from "@material-ui/core/Paper";
import TicketListItem from "../TicketListItem"; import TicketListItem from "../TicketListItem";
import TicketsListSkeleton from "../TicketsListSkeleton"; import TicketsListSkeleton from "../TicketsListSkeleton";
import TicketsListQueueSelect from "../TicketsListQueueSelect";
import useTickets from "../../hooks/useTickets"; import useTickets from "../../hooks/useTickets";
import { useLocalStorage } from "../../hooks/useLocalStorage";
import { i18n } from "../../translate/i18n"; import { i18n } from "../../translate/i18n";
import { ListSubheader } from "@material-ui/core"; import { ListSubheader } from "@material-ui/core";
import { AuthContext } from "../../context/Auth/AuthContext"; import { AuthContext } from "../../context/Auth/AuthContext";
@@ -117,14 +115,14 @@ const reducer = (state, action) => {
return [...state]; return [...state];
} }
if (action.type === "UPDATE_TICKET_MESSAGES_COUNT") { if (action.type === "UPDATE_TICKET_UNREAD_MESSAGES") {
const { ticket, searchParam } = action.payload; const ticket = action.payload;
const ticketIndex = state.findIndex(t => t.id === ticket.id); const ticketIndex = state.findIndex(t => t.id === ticket.id);
if (ticketIndex !== -1) { if (ticketIndex !== -1) {
state[ticketIndex] = ticket; state[ticketIndex] = ticket;
state.unshift(state.splice(ticketIndex, 1)[0]); state.unshift(state.splice(ticketIndex, 1)[0]);
} else if (!searchParam) { } else {
state.unshift(ticket); state.unshift(ticket);
} }
@@ -155,53 +153,40 @@ const reducer = (state, action) => {
} }
}; };
const TicketsList = ({ status, searchParam, showAll }) => { const TicketsList = ({ status, searchParam, showAll, selectedQueueIds }) => {
const classes = useStyles(); const classes = useStyles();
const [pageNumber, setPageNumber] = useState(1); const [pageNumber, setPageNumber] = useState(1);
const [ticketsList, dispatch] = useReducer(reducer, []); const [ticketsList, dispatch] = useReducer(reducer, []);
const { user } = useContext(AuthContext); const { user } = useContext(AuthContext);
const [pendingSelectedQueueIds, setPendingSelectedQueueIds] = useLocalStorage(
"pendingSelectedQueues",
[]
);
const [openSelectedQueueIds, setOpenSelectedQueueIds] = useLocalStorage(
"openSelectedQueues",
[]
);
useEffect(() => { useEffect(() => {
dispatch({ type: "RESET" }); dispatch({ type: "RESET" });
setPageNumber(1); setPageNumber(1);
}, [ }, [status, searchParam, dispatch, showAll, selectedQueueIds]);
status,
searchParam,
dispatch,
showAll,
openSelectedQueueIds,
pendingSelectedQueueIds,
]);
const { tickets, hasMore, loading } = useTickets({ const { tickets, hasMore, loading } = useTickets({
pageNumber, pageNumber,
searchParam, searchParam,
status, status,
showAll, showAll,
queueIds: queueIds: JSON.stringify(selectedQueueIds),
status === "open"
? JSON.stringify(openSelectedQueueIds)
: JSON.stringify(pendingSelectedQueueIds),
}); });
useEffect(() => { useEffect(() => {
if (!status && !searchParam) return;
dispatch({ dispatch({
type: "LOAD_TICKETS", type: "LOAD_TICKETS",
payload: tickets, payload: tickets,
}); });
}, [tickets]); }, [tickets, status, searchParam]);
useEffect(() => { useEffect(() => {
const socket = openSocket(process.env.REACT_APP_BACKEND_URL); const socket = openSocket(process.env.REACT_APP_BACKEND_URL);
const shouldUpdateTicket = ticket =>
(!ticket.userId || ticket.userId === user?.id || showAll) &&
selectedQueueIds.indexOf(ticket.queueId) > -1;
socket.on("connect", () => { socket.on("connect", () => {
if (status) { if (status) {
socket.emit("joinTickets", status); socket.emit("joinTickets", status);
@@ -218,10 +203,7 @@ const TicketsList = ({ status, searchParam, showAll }) => {
}); });
} }
if ( if (data.action === "update" && shouldUpdateTicket(data.ticket)) {
(data.action === "updateStatus" || data.action === "create") &&
(!data.ticket.userId || data.ticket.userId === user?.id || showAll)
) {
dispatch({ dispatch({
type: "UPDATE_TICKET", type: "UPDATE_TICKET",
payload: data.ticket, payload: data.ticket,
@@ -234,16 +216,10 @@ const TicketsList = ({ status, searchParam, showAll }) => {
}); });
socket.on("appMessage", data => { socket.on("appMessage", data => {
if ( if (data.action === "create" && shouldUpdateTicket(data.ticket)) {
data.action === "create" &&
(!data.ticket.userId || data.ticket.userId === user?.id || showAll)
) {
dispatch({ dispatch({
type: "UPDATE_TICKET_MESSAGES_COUNT", type: "UPDATE_TICKET_UNREAD_MESSAGES",
payload: { payload: data.ticket,
ticket: data.ticket,
searchParam,
},
}); });
} }
}); });
@@ -260,7 +236,7 @@ const TicketsList = ({ status, searchParam, showAll }) => {
return () => { return () => {
socket.disconnect(); socket.disconnect();
}; };
}, [status, showAll, user, searchParam]); }, [status, showAll, user, selectedQueueIds]);
const loadMore = () => { const loadMore = () => {
setPageNumber(prevState => prevState + 1); setPageNumber(prevState => prevState + 1);
@@ -276,15 +252,6 @@ const TicketsList = ({ status, searchParam, showAll }) => {
} }
}; };
const handleSelectedQueues = values => {
if (status === "open") {
setOpenSelectedQueueIds(values);
}
if (status === "pending") {
setPendingSelectedQueueIds(values);
}
};
return ( return (
<div className={classes.ticketsListWrapper}> <div className={classes.ticketsListWrapper}>
<Paper <Paper
@@ -303,11 +270,6 @@ const TicketsList = ({ status, searchParam, showAll }) => {
{ticketsList.length} {ticketsList.length}
</span> </span>
</div> </div>
<TicketsListQueueSelect
selectedQueueIds={openSelectedQueueIds}
userQueues={user?.queues}
onChange={handleSelectedQueues}
/>
</ListSubheader> </ListSubheader>
)} )}
{status === "pending" && ( {status === "pending" && (
@@ -318,11 +280,6 @@ const TicketsList = ({ status, searchParam, showAll }) => {
{ticketsList.length} {ticketsList.length}
</span> </span>
</div> </div>
<TicketsListQueueSelect
selectedQueueIds={pendingSelectedQueueIds}
userQueues={user?.queues}
onChange={handleSelectedQueues}
/>
</ListSubheader> </ListSubheader>
)} )}
{ticketsList.length === 0 && !loading ? ( {ticketsList.length === 0 && !loading ? (

View File

@@ -1,4 +1,4 @@
import React, { useContext, useState } from "react"; import React, { useContext, useEffect, useRef, useState } from "react";
import { makeStyles } from "@material-ui/core/styles"; import { makeStyles } from "@material-ui/core/styles";
import Paper from "@material-ui/core/Paper"; import Paper from "@material-ui/core/Paper";
@@ -8,8 +8,7 @@ import Tabs from "@material-ui/core/Tabs";
import Tab from "@material-ui/core/Tab"; import Tab from "@material-ui/core/Tab";
import MoveToInboxIcon from "@material-ui/icons/MoveToInbox"; import MoveToInboxIcon from "@material-ui/icons/MoveToInbox";
import CheckBoxIcon from "@material-ui/icons/CheckBox"; import CheckBoxIcon from "@material-ui/icons/CheckBox";
import IconButton from "@material-ui/core/IconButton";
import AddIcon from "@material-ui/icons/Add";
import FormControlLabel from "@material-ui/core/FormControlLabel"; import FormControlLabel from "@material-ui/core/FormControlLabel";
import Switch from "@material-ui/core/Switch"; import Switch from "@material-ui/core/Switch";
@@ -20,6 +19,9 @@ import TabPanel from "../TabPanel";
import { i18n } from "../../translate/i18n"; import { i18n } from "../../translate/i18n";
import { AuthContext } from "../../context/Auth/AuthContext"; import { AuthContext } from "../../context/Auth/AuthContext";
import Can from "../Can"; import Can from "../Can";
import TicketsQueueSelect from "../TicketsQueueSelect";
import { Button } from "@material-ui/core";
import { useLocalStorage } from "../../hooks/useLocalStorage";
const useStyles = makeStyles(theme => ({ const useStyles = makeStyles(theme => ({
ticketsWrapper: { ticketsWrapper: {
@@ -48,17 +50,12 @@ const useStyles = makeStyles(theme => ({
width: 120, width: 120,
}, },
ticketsListActions: { ticketOptionsBox: {
flex: "none",
marginLeft: "auto",
},
searchBox: {
position: "relative",
display: "flex", display: "flex",
justifyContent: "space-between",
alignItems: "center", alignItems: "center",
background: "#fafafa", background: "#fafafa",
padding: "10px 13px", padding: theme.spacing(1),
}, },
serachInputWrapper: { serachInputWrapper: {
@@ -67,6 +64,7 @@ const useStyles = makeStyles(theme => ({
display: "flex", display: "flex",
borderRadius: 40, borderRadius: 40,
padding: 4, padding: 4,
marginRight: theme.spacing(1),
}, },
searchIcon: { searchIcon: {
@@ -91,15 +89,34 @@ const TicketsManager = () => {
const [newTicketModalOpen, setNewTicketModalOpen] = useState(false); const [newTicketModalOpen, setNewTicketModalOpen] = useState(false);
const [showAllTickets, setShowAllTickets] = useState(false); const [showAllTickets, setShowAllTickets] = useState(false);
const { user } = useContext(AuthContext); const { user } = useContext(AuthContext);
const searchInputRef = useRef();
const [selectedQueueIds, setSelectedQueueIds] = useLocalStorage(
"selectedQueueIds",
[]
);
const handleSearchContact = e => { useEffect(() => {
if (e.target.value === "") { if (tab === "search") {
setSearchParam(e.target.value.toLowerCase()); searchInputRef.current.focus();
}
}, [tab]);
let searchTimeout;
const handleSearch = e => {
const searchedTerm = e.target.value.toLowerCase();
clearTimeout(searchTimeout);
if (searchedTerm === "") {
setSearchParam(searchedTerm);
setTab("open"); setTab("open");
return; return;
} }
setSearchParam(e.target.value.toLowerCase());
setTab("search"); searchTimeout = setTimeout(() => {
setSearchParam(searchedTerm);
}, 500);
}; };
const handleChangeTab = (e, newValue) => { const handleChangeTab = (e, newValue) => {
@@ -141,57 +158,78 @@ const TicketsManager = () => {
/> />
</Tabs> </Tabs>
</Paper> </Paper>
<Paper square elevation={0} className={classes.searchBox}> <Paper square elevation={0} className={classes.ticketOptionsBox}>
<div className={classes.serachInputWrapper}> {tab === "search" ? (
<SearchIcon className={classes.searchIcon} /> <div className={classes.serachInputWrapper}>
<InputBase <SearchIcon className={classes.searchIcon} />
className={classes.searchInput} <InputBase
placeholder={i18n.t("tickets.search.placeholder")} className={classes.searchInput}
type="search" inputRef={searchInputRef}
onChange={handleSearchContact} placeholder={i18n.t("tickets.search.placeholder")}
/> type="search"
</div> onChange={handleSearch}
<div className={classes.ticketsListActions}> />
<Can </div>
role={user.profile} ) : (
perform="tickets-manager:showall" <>
yes={() => ( <Button
<FormControlLabel variant="outlined"
label={i18n.t("tickets.buttons.showAll")} color="primary"
labelPlacement="start" onClick={() => setNewTicketModalOpen(true)}
control={ >
<Switch Novo
size="small" </Button>
checked={showAllTickets} <Can
onChange={() => setShowAllTickets(prevState => !prevState)} role={user.profile}
name="showAllTickets" perform="tickets-manager:showall"
color="primary" yes={() => (
/> <FormControlLabel
} label={i18n.t("tickets.buttons.showAll")}
/> labelPlacement="start"
)} control={
/> <Switch
size="small"
<IconButton checked={showAllTickets}
aria-label="add ticket" onChange={() =>
size="small" setShowAllTickets(prevState => !prevState)
color="primary" }
onClick={e => setNewTicketModalOpen(true)} name="showAllTickets"
style={{ marginLeft: 20 }} color="primary"
> />
<AddIcon /> }
</IconButton> />
</div> )}
/>
</>
)}
<TicketsQueueSelect
style={{ marginLeft: 6 }}
selectedQueueIds={selectedQueueIds}
userQueues={user?.queues}
onChange={values => setSelectedQueueIds(values)}
/>
</Paper> </Paper>
<TabPanel value={tab} name="open" className={classes.ticketsWrapper}> <TabPanel value={tab} name="open" className={classes.ticketsWrapper}>
<TicketsList status="open" showAll={showAllTickets} /> <TicketsList
<TicketsList status="pending" /> status="open"
showAll={showAllTickets}
selectedQueueIds={selectedQueueIds}
/>
<TicketsList status="pending" selectedQueueIds={selectedQueueIds} />
</TabPanel> </TabPanel>
<TabPanel value={tab} name="closed" className={classes.ticketsWrapper}> <TabPanel value={tab} name="closed" className={classes.ticketsWrapper}>
<TicketsList status="closed" showAll={showAllTickets} /> <TicketsList
status="closed"
showAll={true}
selectedQueueIds={selectedQueueIds}
/>
</TabPanel> </TabPanel>
<TabPanel value={tab} name="search" className={classes.ticketsWrapper}> <TabPanel value={tab} name="search" className={classes.ticketsWrapper}>
<TicketsList searchParam={searchParam} showAll={showAllTickets} /> <TicketsList
searchParam={searchParam}
showAll={true}
selectedQueueIds={selectedQueueIds}
/>
</TabPanel> </TabPanel>
</Paper> </Paper>
); );

View File

@@ -5,7 +5,7 @@ import FormControl from "@material-ui/core/FormControl";
import Select from "@material-ui/core/Select"; import Select from "@material-ui/core/Select";
import { Checkbox, ListItemText } from "@material-ui/core"; import { Checkbox, ListItemText } from "@material-ui/core";
const TicketsListQueueSelect = ({ const TicketsQueueSelect = ({
userQueues, userQueues,
selectedQueueIds = [], selectedQueueIds = [],
onChange, onChange,
@@ -15,7 +15,7 @@ const TicketsListQueueSelect = ({
}; };
return ( return (
<div style={{ width: 120, marginTop: -5 }}> <div style={{ width: 120, marginTop: -4 }}>
<FormControl fullWidth margin="dense"> <FormControl fullWidth margin="dense">
<Select <Select
multiple multiple
@@ -56,4 +56,4 @@ const TicketsListQueueSelect = ({
); );
}; };
export default TicketsListQueueSelect; export default TicketsQueueSelect;