mirror of
https://github.com/bluewave-labs/Checkmate.git
synced 2026-04-27 03:59:43 -05:00
415 lines
12 KiB
React
415 lines
12 KiB
React
import { useTheme } from "@emotion/react";
|
|
import TabPanel from "@mui/lab/TabPanel";
|
|
import {
|
|
Box,
|
|
ButtonGroup,
|
|
Divider,
|
|
IconButton,
|
|
Modal,
|
|
Stack,
|
|
TextField,
|
|
Typography,
|
|
} from "@mui/material";
|
|
import ButtonSpinner from "../../ButtonSpinner";
|
|
import Button from "../../Button";
|
|
import { useEffect, useState } from "react";
|
|
import EditSvg from "../../../assets/icons/edit.svg?react";
|
|
import Field from "../../Inputs/Field";
|
|
import { credentials } from "../../../Validation/validation";
|
|
import { networkService } from "../../../main";
|
|
import { createToast } from "../../../Utils/toastUtils";
|
|
import { useSelector } from "react-redux";
|
|
import BasicTable from "../../BasicTable";
|
|
import Remove from "../../../assets/icons/trash-bin.svg?react";
|
|
import Select from "../../Inputs/Select";
|
|
|
|
/**
|
|
* TeamPanel component manages the organization and team members,
|
|
* providing functionalities like renaming the organization, managing team members,
|
|
* and inviting new members.
|
|
*
|
|
* @returns {JSX.Element}
|
|
*/
|
|
|
|
const TeamPanel = () => {
|
|
const theme = useTheme();
|
|
|
|
const { authToken, user } = useSelector((state) => state.auth);
|
|
//TODO
|
|
const [orgStates, setOrgStates] = useState({
|
|
name: "Bluewave Labs",
|
|
isEdit: false,
|
|
});
|
|
const [toInvite, setToInvite] = useState({
|
|
email: "",
|
|
role: ["0"],
|
|
});
|
|
const [tableData, setTableData] = useState({});
|
|
const [members, setMembers] = useState([]);
|
|
const [filter, setFilter] = useState("all");
|
|
const [errors, setErrors] = useState({});
|
|
|
|
useEffect(() => {
|
|
const fetchTeam = async () => {
|
|
try {
|
|
const response = await networkService.getAllUsers(authToken);
|
|
setMembers(response.data.data);
|
|
} catch (error) {
|
|
createToast({
|
|
body: error.message || "Error fetching team members.",
|
|
});
|
|
}
|
|
};
|
|
|
|
fetchTeam();
|
|
}, [user]);
|
|
|
|
useEffect(() => {
|
|
let team = members;
|
|
if (filter !== "all")
|
|
team = members.filter((member) => member.role.includes(filter));
|
|
|
|
const data = {
|
|
cols: [
|
|
{ id: 1, name: "NAME" },
|
|
{ id: 2, name: "EMAIL" },
|
|
{ id: 3, name: "ROLE" },
|
|
// FEATURE STILL TO BE IMPLEMENTED
|
|
// { id: 4, name: "ACTION" },
|
|
],
|
|
rows: team?.map((member, idx) => {
|
|
return {
|
|
id: member._id,
|
|
data: [
|
|
{
|
|
id: idx,
|
|
data: (
|
|
<Stack>
|
|
<Typography
|
|
style={{ color: theme.palette.otherColors.blackish }}
|
|
>
|
|
{member.firstName + " " + member.lastName}
|
|
</Typography>
|
|
<Typography>
|
|
Created {new Date(member.createdAt).toLocaleDateString()}
|
|
</Typography>
|
|
</Stack>
|
|
),
|
|
},
|
|
{ id: idx + 1, data: member.email },
|
|
{
|
|
// TODO - Add select dropdown
|
|
id: idx + 2,
|
|
data: member.role.includes("admin") ? "Administrator" : "Member",
|
|
},
|
|
// FEATURE STILL TO BE IMPLEMENTED
|
|
// {
|
|
// // TODO - Add delete onClick
|
|
// id: idx + 3,
|
|
// data: (
|
|
// <IconButton
|
|
// aria-label="remove member"
|
|
// sx={{
|
|
// "&:focus": {
|
|
// outline: "none",
|
|
// },
|
|
// }}
|
|
// >
|
|
// <Remove />
|
|
// </IconButton>
|
|
// ),
|
|
// },
|
|
],
|
|
};
|
|
}),
|
|
};
|
|
|
|
setTableData(data);
|
|
}, [members, filter]);
|
|
|
|
// RENAME ORGANIZATION
|
|
const toggleEdit = () => {
|
|
setOrgStates((prev) => ({ ...prev, isEdit: !prev.isEdit }));
|
|
};
|
|
const handleRename = () => {};
|
|
|
|
// INVITE MEMBER
|
|
const [isOpen, setIsOpen] = useState(false);
|
|
|
|
const handleChange = (event) => {
|
|
const { value } = event.target;
|
|
setToInvite((prev) => ({
|
|
...prev,
|
|
email: value,
|
|
}));
|
|
|
|
const validation = credentials.validate(
|
|
{ email: value },
|
|
{ abortEarly: false }
|
|
);
|
|
|
|
setErrors((prev) => {
|
|
const updatedErrors = { ...prev };
|
|
|
|
if (validation.error) {
|
|
updatedErrors.email = validation.error.details[0].message;
|
|
} else {
|
|
delete updatedErrors.email;
|
|
}
|
|
return updatedErrors;
|
|
});
|
|
};
|
|
|
|
const handleInviteMember = async () => {
|
|
if (!toInvite.role.includes("user") || !toInvite.role.includes("admin"))
|
|
setToInvite((prev) => ({ ...prev, role: ["user"] }));
|
|
|
|
const { error } = credentials.validate(
|
|
{ email: toInvite.email },
|
|
{
|
|
abortEarly: false,
|
|
}
|
|
);
|
|
|
|
if (error) {
|
|
setErrors((prev) => ({ ...prev, email: error.details[0].message }));
|
|
} else
|
|
try {
|
|
await networkService.requestInvitationToken(
|
|
authToken,
|
|
toInvite.email,
|
|
toInvite.role
|
|
);
|
|
|
|
closeInviteModal();
|
|
createToast({
|
|
body: "Member invited. They will receive an email with details on how to create their account.",
|
|
});
|
|
} catch (error) {
|
|
createToast({
|
|
body: error.message || "Unknown error.",
|
|
});
|
|
}
|
|
};
|
|
const closeInviteModal = () => {
|
|
setIsOpen(false);
|
|
setToInvite({ email: "", role: ["0"] });
|
|
setErrors({});
|
|
};
|
|
|
|
return (
|
|
<TabPanel
|
|
value="team"
|
|
sx={{
|
|
"& h1": {
|
|
color: theme.palette.text.tertiary,
|
|
},
|
|
"& .MuiTable-root .MuiTableBody-root .MuiTableCell-root, & .MuiTable-root p + p":
|
|
{
|
|
color: theme.palette.text.accent,
|
|
},
|
|
}}
|
|
>
|
|
{/* FEATURE STILL TO BE IMPLEMENTED */}
|
|
{/* <Stack component="form">
|
|
<Box sx={{ alignSelf: "flex-start" }}>
|
|
<Typography component="h1">Organization name</Typography>
|
|
</Box>
|
|
<Stack
|
|
direction="row"
|
|
justifyContent="flex-end"
|
|
alignItems="center"
|
|
sx={{ height: "34px" }}
|
|
>
|
|
<TextField
|
|
value={orgStates.name}
|
|
onChange={(event) =>
|
|
setOrgStates((prev) => ({
|
|
...prev,
|
|
name: event.target.value,
|
|
}))
|
|
}
|
|
disabled={!orgStates.isEdit}
|
|
sx={{
|
|
color: theme.palette.otherColors.bluishGray,
|
|
"& .Mui-disabled": {
|
|
WebkitTextFillColor: "initial !important",
|
|
},
|
|
"& .Mui-disabled fieldset": {
|
|
borderColor: "transparent !important",
|
|
},
|
|
}}
|
|
inputProps={{
|
|
sx: { textAlign: "end", padding: theme.gap.small },
|
|
}}
|
|
/>
|
|
<Button
|
|
level={orgStates.isEdit ? "secondary" : "tertiary"}
|
|
label={orgStates.isEdit ? "Save" : ""}
|
|
img={!orgStates.isEdit ? <EditSvg /> : ""}
|
|
onClick={() => toggleEdit()}
|
|
sx={{
|
|
minWidth: 0,
|
|
paddingX: theme.gap.small,
|
|
ml: orgStates.isEdit ? theme.gap.small : 0,
|
|
}}
|
|
/>
|
|
</Stack>
|
|
</Stack>
|
|
<Divider aria-hidden="true" sx={{ marginY: theme.spacing(4) }} /> */}
|
|
<Stack
|
|
component="form"
|
|
noValidate
|
|
spellCheck="false"
|
|
gap={theme.spacing(12)}
|
|
>
|
|
<Typography component="h1">Team members</Typography>
|
|
<Stack direction="row" justifyContent="space-between">
|
|
<Stack
|
|
direction="row"
|
|
alignItems="flex-end"
|
|
gap={theme.spacing(6)}
|
|
sx={{ fontSize: 14 }}
|
|
>
|
|
<ButtonGroup
|
|
sx={{
|
|
"& button, & button:hover": {
|
|
borderColor: theme.palette.border.light,
|
|
},
|
|
}}
|
|
>
|
|
<Button
|
|
level="secondary"
|
|
label="All"
|
|
onClick={() => setFilter("all")}
|
|
sx={{
|
|
backgroundColor:
|
|
filter === "all" && theme.palette.background.fill,
|
|
}}
|
|
/>
|
|
<Button
|
|
level="secondary"
|
|
label="Administrator"
|
|
onClick={() => setFilter("admin")}
|
|
sx={{
|
|
backgroundColor:
|
|
filter === "admin" && theme.palette.background.fill,
|
|
}}
|
|
/>
|
|
<Button
|
|
level="secondary"
|
|
label="Member"
|
|
onClick={() => setFilter("user")}
|
|
sx={{
|
|
backgroundColor:
|
|
filter === "user" && theme.palette.background.fill,
|
|
}}
|
|
/>
|
|
</ButtonGroup>
|
|
</Stack>
|
|
<Button
|
|
level="primary"
|
|
label="Invite a team member"
|
|
sx={{ paddingX: theme.spacing(15) }}
|
|
onClick={() => setIsOpen(true)}
|
|
/>
|
|
</Stack>
|
|
<BasicTable
|
|
data={tableData}
|
|
paginated={false}
|
|
reversed={true}
|
|
table={"team"}
|
|
/>
|
|
</Stack>
|
|
<Modal
|
|
aria-labelledby="modal-invite-member"
|
|
aria-describedby="invite-member-to-team"
|
|
open={isOpen}
|
|
onClose={closeInviteModal}
|
|
disablePortal
|
|
>
|
|
<Stack
|
|
gap={theme.spacing(5)}
|
|
sx={{
|
|
position: "absolute",
|
|
top: "50%",
|
|
left: "50%",
|
|
transform: "translate(-50%, -50%)",
|
|
width: 400,
|
|
bgcolor: theme.palette.background.main,
|
|
border: 1,
|
|
borderColor: theme.palette.border.light,
|
|
borderRadius: theme.shape.borderRadius,
|
|
boxShadow: 24,
|
|
p: theme.spacing(15),
|
|
"&:focus": {
|
|
outline: "none",
|
|
},
|
|
}}
|
|
>
|
|
<Typography id="modal-invite-member" component="h1">
|
|
Invite new team member
|
|
</Typography>
|
|
<Typography
|
|
id="invite-member-to-team"
|
|
component="p"
|
|
sx={{ mb: theme.spacing(6) }}
|
|
>
|
|
When you add a new team member, they will get access to all
|
|
monitors.
|
|
</Typography>
|
|
<Field
|
|
type="email"
|
|
id="input-team-member"
|
|
placeholder="Email"
|
|
value={toInvite.email}
|
|
onChange={handleChange}
|
|
error={errors.email}
|
|
/>
|
|
<Select
|
|
id="team-member-role"
|
|
placeholder="Select role"
|
|
isHidden={true}
|
|
value={toInvite.role[0]}
|
|
onChange={(event) =>
|
|
setToInvite((prev) => ({
|
|
...prev,
|
|
role: [event.target.value],
|
|
}))
|
|
}
|
|
items={[
|
|
{ _id: "admin", name: "Admin" },
|
|
{ _id: "user", name: "User" },
|
|
]}
|
|
/>
|
|
<Stack
|
|
direction="row"
|
|
gap={theme.spacing(4)}
|
|
mt={theme.spacing(8)}
|
|
justifyContent="flex-end"
|
|
>
|
|
<Button
|
|
level="tertiary"
|
|
label="Cancel"
|
|
onClick={closeInviteModal}
|
|
/>
|
|
<ButtonSpinner
|
|
level="primary"
|
|
label="Send invite"
|
|
onClick={handleInviteMember}
|
|
isLoading={false}
|
|
disabled={Object.keys(errors).length !== 0}
|
|
/>
|
|
</Stack>
|
|
</Stack>
|
|
</Modal>
|
|
</TabPanel>
|
|
);
|
|
};
|
|
|
|
TeamPanel.propTypes = {
|
|
// No props are being passed to this component, hence no specific PropTypes are defined.
|
|
};
|
|
|
|
export default TeamPanel;
|