| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294 |
- import React, { useState, useEffect, useReducer, useContext } from "react";
- import { toast } from "react-toastify";
- import { makeStyles } from "@material-ui/core/styles";
- import Paper from "@material-ui/core/Paper";
- import Button from "@material-ui/core/Button";
- import Table from "@material-ui/core/Table";
- import TableBody from "@material-ui/core/TableBody";
- import TableCell from "@material-ui/core/TableCell";
- import TableHead from "@material-ui/core/TableHead";
- import TableRow from "@material-ui/core/TableRow";
- import IconButton from "@material-ui/core/IconButton";
- import SearchIcon from "@material-ui/icons/Search";
- import TextField from "@material-ui/core/TextField";
- import InputAdornment from "@material-ui/core/InputAdornment";
- import DeleteOutlineIcon from "@material-ui/icons/DeleteOutline";
- import EditIcon from "@material-ui/icons/Edit";
- import MainContainer from "../../components/MainContainer";
- import MainHeader from "../../components/MainHeader";
- import MainHeaderButtonsWrapper from "../../components/MainHeaderButtonsWrapper";
- import Title from "../../components/Title";
- import api from "../../services/api";
- import { i18n } from "../../translate/i18n";
- import TableRowSkeleton from "../../components/TableRowSkeleton";
- import UserModal from "../../components/UserModal";
- import ConfirmationModal from "../../components/ConfirmationModal";
- import toastError from "../../errors/toastError";
- import { SocketContext } from "../../context/Socket/SocketContext";
- const reducer = (state, action) => {
- if (action.type === "LOAD_USERS") {
- const users = action.payload;
- const newUsers = [];
- users.forEach((user) => {
- const userIndex = state.findIndex((u) => u.id === user.id);
- if (userIndex !== -1) {
- state[userIndex] = user;
- } else {
- newUsers.push(user);
- }
- });
- return [...state, ...newUsers];
- }
- if (action.type === "UPDATE_USERS") {
- const user = action.payload;
- const userIndex = state.findIndex((u) => u.id === user.id);
- if (userIndex !== -1) {
- state[userIndex] = user;
- return [...state];
- } else {
- return [user, ...state];
- }
- }
- if (action.type === "DELETE_USER") {
- const userId = action.payload;
- const userIndex = state.findIndex((u) => u.id === userId);
- if (userIndex !== -1) {
- state.splice(userIndex, 1);
- }
- return [...state];
- }
- if (action.type === "RESET") {
- return [];
- }
- };
- const useStyles = makeStyles((theme) => ({
- mainPaper: {
- flex: 1,
- padding: theme.spacing(1),
- overflowY: "scroll",
- ...theme.scrollbarStyles,
- },
- }));
- const Users = () => {
- const classes = useStyles();
- const [loading, setLoading] = useState(false);
- const [pageNumber, setPageNumber] = useState(1);
- const [hasMore, setHasMore] = useState(false);
- const [selectedUser, setSelectedUser] = useState(null);
- const [deletingUser, setDeletingUser] = useState(null);
- const [userModalOpen, setUserModalOpen] = useState(false);
- const [confirmModalOpen, setConfirmModalOpen] = useState(false);
- const [searchParam, setSearchParam] = useState("");
- const [users, dispatch] = useReducer(reducer, []);
- const socketManager = useContext(SocketContext);
- useEffect(() => {
- dispatch({ type: "RESET" });
- setPageNumber(1);
- }, [searchParam]);
- useEffect(() => {
- setLoading(true);
- const delayDebounceFn = setTimeout(() => {
- const fetchUsers = async () => {
- try {
- const { data } = await api.get("/users/", {
- params: { searchParam, pageNumber },
- });
- dispatch({ type: "LOAD_USERS", payload: data.users });
- setHasMore(data.hasMore);
- setLoading(false);
- } catch (err) {
- toastError(err);
- }
- };
- fetchUsers();
- }, 500);
- return () => clearTimeout(delayDebounceFn);
- }, [searchParam, pageNumber]);
- useEffect(() => {
- const companyId = localStorage.getItem("companyId");
- const socket = socketManager.getSocket(companyId);
- socket.on(`company-${companyId}-user`, (data) => {
- if (data.action === "update" || data.action === "create") {
- dispatch({ type: "UPDATE_USERS", payload: data.user });
- }
- if (data.action === "delete") {
- dispatch({ type: "DELETE_USER", payload: +data.userId });
- }
- });
- return () => {
- socket.disconnect();
- };
- }, [socketManager]);
- const handleOpenUserModal = () => {
- setSelectedUser(null);
- setUserModalOpen(true);
- };
- const handleCloseUserModal = () => {
- setSelectedUser(null);
- setUserModalOpen(false);
- };
- const handleSearch = (event) => {
- setSearchParam(event.target.value.toLowerCase());
- };
- const handleEditUser = (user) => {
- setSelectedUser(user);
- setUserModalOpen(true);
- };
- const handleDeleteUser = async (userId) => {
- try {
- await api.delete(`/users/${userId}`);
- toast.success(i18n.t("users.toasts.deleted"));
- } catch (err) {
- toastError(err);
- }
- setDeletingUser(null);
- setSearchParam("");
- setPageNumber(1);
- };
- 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();
- }
- };
- return (
- <MainContainer>
- <ConfirmationModal
- title={
- deletingUser &&
- `${i18n.t("users.confirmationModal.deleteTitle")} ${
- deletingUser.name
- }?`
- }
- open={confirmModalOpen}
- onClose={setConfirmModalOpen}
- onConfirm={() => handleDeleteUser(deletingUser.id)}
- >
- {i18n.t("users.confirmationModal.deleteMessage")}
- </ConfirmationModal>
- <UserModal
- open={userModalOpen}
- onClose={handleCloseUserModal}
- aria-labelledby="form-dialog-title"
- userId={selectedUser && selectedUser.id}
- />
- <MainHeader>
- <Title>{i18n.t("users.title")}</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={handleOpenUserModal}
- >
- {i18n.t("users.buttons.add")}
- </Button>
- </MainHeaderButtonsWrapper>
- </MainHeader>
- <Paper
- className={classes.mainPaper}
- variant="outlined"
- onScroll={handleScroll}
- >
- <Table size="small">
- <TableHead>
- <TableRow>
- <TableCell align="center">
- {i18n.t("users.table.id")}
- </TableCell>
- <TableCell align="center">{i18n.t("users.table.name")}</TableCell>
- <TableCell align="center">
- {i18n.t("users.table.email")}
- </TableCell>
- <TableCell align="center">
- {i18n.t("users.table.profile")}
- </TableCell>
- <TableCell align="center">
- {i18n.t("users.table.actions")}
- </TableCell>
- </TableRow>
- </TableHead>
- <TableBody>
- <>
- {users.map((user) => (
- <TableRow key={user.id}>
- <TableCell align="center">{user.id}</TableCell>
- <TableCell align="center">{user.name}</TableCell>
- <TableCell align="center">{user.email}</TableCell>
- <TableCell align="center">{user.profile}</TableCell>
- <TableCell align="center">
- <IconButton
- size="small"
- onClick={() => handleEditUser(user)}
- >
- <EditIcon />
- </IconButton>
- <IconButton
- size="small"
- onClick={(e) => {
- setConfirmModalOpen(true);
- setDeletingUser(user);
- }}
- >
- <DeleteOutlineIcon />
- </IconButton>
- </TableCell>
- </TableRow>
- ))}
- {loading && <TableRowSkeleton columns={4} />}
- </>
- </TableBody>
- </Table>
- </Paper>
- </MainContainer>
- );
- };
- export default Users;
|