Compare commits
126 Commits
feature/Ex
...
approval-w
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d7a8f496c0 | ||
|
|
5e363e9951 | ||
|
|
3370f3c648 | ||
|
|
d77336374d | ||
|
|
e765dea106 | ||
|
|
75fb9490e0 | ||
|
|
3ef7998193 | ||
|
|
32cd8495d6 | ||
|
|
4e3cfec9e8 | ||
|
|
ba8cc342b1 | ||
|
|
dd8f821e35 | ||
|
|
a4ef2222e2 | ||
|
|
93d9e49358 | ||
|
|
5d0a3acbee | ||
|
|
340ff5a30a | ||
|
|
37908423eb | ||
|
|
b388ee399f | ||
|
|
4ac11df6ae | ||
|
|
14e2702aca | ||
|
|
fec3b51553 | ||
|
|
d8386bdd8e | ||
|
|
df2f83e496 | ||
|
|
e214d8b598 | ||
|
|
c14f16c97a | ||
|
|
ca2cf739ee | ||
|
|
d432fb4bc4 | ||
|
|
d5bffc9bad | ||
|
|
75b4643918 | ||
|
|
9ae6b8e894 | ||
|
|
6f6c5a4209 | ||
|
|
769b1b39d3 | ||
|
|
4bb12c7f01 | ||
|
|
a80a342ae2 | ||
|
|
e5e60fcce9 | ||
|
|
b175d8797e | ||
|
|
f06349e350 | ||
|
|
34caf9986c | ||
|
|
3a3d3d014d | ||
|
|
c49c303f20 | ||
|
|
cbe353c2c5 | ||
|
|
991adede96 | ||
|
|
f95bce6fa2 | ||
|
|
1dd6cead9e | ||
|
|
5a72ebaea1 | ||
|
|
00d2a7c2ad | ||
|
|
a6e122e82d | ||
|
|
bf2aa29b98 | ||
|
|
cf12a4ed4f | ||
|
|
8711802b97 | ||
|
|
36afde8aa4 | ||
|
|
752881df41 | ||
|
|
63604b68e2 | ||
|
|
d74aa39076 | ||
|
|
c3849518fb | ||
|
|
7fb5e1a62b | ||
|
|
4b405297f2 | ||
|
|
f0849b9b42 | ||
|
|
845a5aa9dc | ||
|
|
d48c7b0d03 | ||
|
|
6692c201e4 | ||
|
|
f4c7961caa | ||
|
|
b215885dc6 | ||
|
|
de15eb5ee1 | ||
|
|
d3385caaf8 | ||
|
|
19f2193414 | ||
|
|
d59b654ac2 | ||
|
|
29b6a02118 | ||
|
|
b77476dc9a | ||
|
|
5a685ebe80 | ||
|
|
835a9bee03 | ||
|
|
16545d2075 | ||
|
|
b684262759 | ||
|
|
ac539332e6 | ||
|
|
ed87c8b163 | ||
|
|
e33ab315ad | ||
|
|
1feef5c419 | ||
|
|
a0229cd971 | ||
|
|
662e3b0266 | ||
|
|
b9aec7261f | ||
|
|
54a9f6869a | ||
|
|
9de4cba8e8 | ||
|
|
7d750dc584 | ||
|
|
98ba0bfc04 | ||
|
|
f89b42c41c | ||
|
|
c968044160 | ||
|
|
5d727fc528 | ||
|
|
bdc5ff7797 | ||
|
|
011c6e9e30 | ||
|
|
42a8ec2f8a | ||
|
|
58aebaa66c | ||
|
|
b69b6e6c77 | ||
|
|
86af876f01 | ||
|
|
b685259dc7 | ||
|
|
16b959fb7a | ||
|
|
a40ae04aa3 | ||
|
|
db95fc5681 | ||
|
|
8db47a3962 | ||
|
|
ab81a1753d | ||
|
|
c98af863c3 | ||
|
|
73610dc273 | ||
|
|
37216e2a5a | ||
|
|
ac072b0a5a | ||
|
|
2c0153e055 | ||
|
|
2eff08bf86 | ||
|
|
f71a7182dd | ||
|
|
1f7639a30e | ||
|
|
41d09eaad8 | ||
|
|
f6b0c96b3b | ||
|
|
dcd25465fd | ||
|
|
c921d54d50 | ||
|
|
a4f60455b5 | ||
|
|
a0936cb1a4 | ||
|
|
aa76c2b54b | ||
|
|
4e81c08adb | ||
|
|
4895f00184 | ||
|
|
f727ab4792 | ||
|
|
8f8d5e5640 | ||
|
|
73e2e95449 | ||
|
|
48187fc7f2 | ||
|
|
01222b3a13 | ||
|
|
39a397d262 | ||
|
|
50d2841349 | ||
|
|
f485c782f3 | ||
|
|
c2c9b3374c | ||
|
|
66d23b4140 | ||
|
|
580e319fb9 |
4
.gitignore
vendored
4
.gitignore
vendored
@@ -40,4 +40,6 @@ next-env.d.ts
|
|||||||
.env
|
.env
|
||||||
.yarn/*
|
.yarn/*
|
||||||
.history*
|
.history*
|
||||||
__ENV.js
|
__ENV.js
|
||||||
|
|
||||||
|
settings.json
|
||||||
@@ -39,6 +39,7 @@
|
|||||||
"country-codes-list": "^1.6.11",
|
"country-codes-list": "^1.6.11",
|
||||||
"currency-symbol-map": "^5.1.0",
|
"currency-symbol-map": "^5.1.0",
|
||||||
"daisyui": "^3.1.5",
|
"daisyui": "^3.1.5",
|
||||||
|
"deep-diff": "^1.0.2",
|
||||||
"eslint": "8.33.0",
|
"eslint": "8.33.0",
|
||||||
"eslint-config-next": "13.1.6",
|
"eslint-config-next": "13.1.6",
|
||||||
"exceljs": "^4.4.0",
|
"exceljs": "^4.4.0",
|
||||||
@@ -97,6 +98,7 @@
|
|||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@simbathesailor/use-what-changed": "^2.0.0",
|
"@simbathesailor/use-what-changed": "^2.0.0",
|
||||||
"@types/blob-stream": "^0.1.33",
|
"@types/blob-stream": "^0.1.33",
|
||||||
|
"@types/deep-diff": "^1.0.5",
|
||||||
"@types/formidable": "^3.4.0",
|
"@types/formidable": "^3.4.0",
|
||||||
"@types/howler": "^2.2.11",
|
"@types/howler": "^2.2.11",
|
||||||
"@types/lodash": "^4.14.191",
|
"@types/lodash": "^4.14.191",
|
||||||
|
|||||||
51
scripts/updatePrivateFieldExams.js
Normal file
51
scripts/updatePrivateFieldExams.js
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
import dotenv from "dotenv";
|
||||||
|
dotenv.config();
|
||||||
|
import { MongoClient } from "mongodb";
|
||||||
|
const uri = process.env.MONGODB_URI || "";
|
||||||
|
const options = {
|
||||||
|
maxPoolSize: 10,
|
||||||
|
};
|
||||||
|
const dbName = process.env.MONGODB_DB; // change this to prod db when needed
|
||||||
|
async function migrateData() {
|
||||||
|
const MODULE_ARRAY = ["reading", "listening", "writing", "speaking", "level"];
|
||||||
|
const client = new MongoClient(uri, options);
|
||||||
|
|
||||||
|
try {
|
||||||
|
await client.connect();
|
||||||
|
console.log("Connected to MongoDB");
|
||||||
|
if (!process.env.MONGODB_DB) {
|
||||||
|
throw new Error("Missing env var: MONGODB_DB");
|
||||||
|
}
|
||||||
|
const db = client.db(dbName);
|
||||||
|
for (const string of MODULE_ARRAY) {
|
||||||
|
const collection = db.collection(string);
|
||||||
|
const result = await collection.updateMany(
|
||||||
|
{ private: { $exists: false } },
|
||||||
|
{ $set: { access: "public" } }
|
||||||
|
);
|
||||||
|
const result2 = await collection.updateMany(
|
||||||
|
{ private: true },
|
||||||
|
{ $set: { access: "private" }, $unset: { private: "" } }
|
||||||
|
);
|
||||||
|
const result1 = await collection.updateMany(
|
||||||
|
{ private: { $exists: true } },
|
||||||
|
{ $set: { access: "public" } }
|
||||||
|
);
|
||||||
|
console.log(
|
||||||
|
`Updated ${
|
||||||
|
result.modifiedCount + result1.modifiedCount
|
||||||
|
} documents to "access: public" in ${string}`
|
||||||
|
);
|
||||||
|
console.log(
|
||||||
|
`Updated ${result2.modifiedCount} documents to "access: private" and removed private var in ${string}`
|
||||||
|
);
|
||||||
|
}
|
||||||
|
console.log("Migration completed successfully!");
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Migration failed:", error);
|
||||||
|
} finally {
|
||||||
|
await client.close();
|
||||||
|
console.log("MongoDB connection closed.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
//migrateData(); // uncomment to run the migration
|
||||||
32
src/components/ApprovalWorkflows/RequestedBy.tsx
Normal file
32
src/components/ApprovalWorkflows/RequestedBy.tsx
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
import Image from "next/image";
|
||||||
|
import React from "react";
|
||||||
|
import { FaRegUser } from "react-icons/fa";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
prefix: string;
|
||||||
|
name: string;
|
||||||
|
profileImage: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function RequestedBy({ prefix, name, profileImage }: Props) {
|
||||||
|
return (
|
||||||
|
<div className="flex items-center space-x-3">
|
||||||
|
<div className="flex items-center justify-center w-12 h-12 bg-gray-100 rounded-lg border border-gray-300">
|
||||||
|
<FaRegUser className="text-mti-purple-dark size-5"/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p className="text-sm font-medium text-gray-800">Requested by</p>
|
||||||
|
<div className="flex items-center space-x-2">
|
||||||
|
<p className="text-xs font-medium text-gray-800">{prefix} {name}</p>
|
||||||
|
<img
|
||||||
|
src={profileImage ? profileImage : "/defaultAvatar.png"}
|
||||||
|
alt={name}
|
||||||
|
width={24}
|
||||||
|
height={24}
|
||||||
|
className="w-6 h-6 rounded-full border-[1px] border-gray-400 border-opacity-50"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
41
src/components/ApprovalWorkflows/StartedOn.tsx
Normal file
41
src/components/ApprovalWorkflows/StartedOn.tsx
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
import React from "react";
|
||||||
|
import { PiCalendarDots } from "react-icons/pi";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
date: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function StartedOn({ date }: Props) {
|
||||||
|
const formattedDate = new Date(date);
|
||||||
|
|
||||||
|
const yearMonthDay = formattedDate.toISOString().split("T")[0];
|
||||||
|
|
||||||
|
const fullDateTime = formattedDate.toLocaleString("en-US", {
|
||||||
|
year: "numeric",
|
||||||
|
month: "long",
|
||||||
|
day: "numeric",
|
||||||
|
hour: "2-digit",
|
||||||
|
minute: "2-digit",
|
||||||
|
second: "2-digit",
|
||||||
|
hour12: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex items-center space-x-3">
|
||||||
|
<div className="flex items-center justify-center w-12 h-12 bg-gray-100 rounded-lg border border-gray-300">
|
||||||
|
<PiCalendarDots className="text-mti-purple-dark size-7" />
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p className="pb-1 text-sm font-medium text-gray-800">Started on</p>
|
||||||
|
<div className="flex items-center">
|
||||||
|
<p
|
||||||
|
className="text-xs font-medium text-gray-800"
|
||||||
|
title={fullDateTime}
|
||||||
|
>
|
||||||
|
{yearMonthDay}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
23
src/components/ApprovalWorkflows/Status.tsx
Normal file
23
src/components/ApprovalWorkflows/Status.tsx
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
import { ApprovalWorkflowStatus, ApprovalWorkflowStatusLabel } from "@/interfaces/approval.workflow";
|
||||||
|
import React from "react";
|
||||||
|
import { RiProgress5Line } from "react-icons/ri";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
status: ApprovalWorkflowStatus;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function Status({ status }: Props) {
|
||||||
|
return (
|
||||||
|
<div className="flex items-center space-x-3">
|
||||||
|
<div className="flex items-center justify-center w-12 h-12 bg-gray-100 rounded-lg border border-gray-300">
|
||||||
|
<RiProgress5Line className="text-mti-purple-dark size-7"/>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<p className="pb-1 text-sm font-medium text-gray-800">Status</p>
|
||||||
|
<div className="flex items-center">
|
||||||
|
<p className="text-xs font-medium text-gray-800">{ApprovalWorkflowStatusLabel[status]}</p>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
14
src/components/ApprovalWorkflows/Tip.tsx
Normal file
14
src/components/ApprovalWorkflows/Tip.tsx
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
import { MdTipsAndUpdates } from "react-icons/md";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
text: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function Tip({ text }: Props) {
|
||||||
|
return (
|
||||||
|
<div className="flex flex-row gap-3 text-gray-500 font-medium">
|
||||||
|
<MdTipsAndUpdates size={25} />
|
||||||
|
<p>{text}</p>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
24
src/components/ApprovalWorkflows/UserWithProfilePic.tsx
Normal file
24
src/components/ApprovalWorkflows/UserWithProfilePic.tsx
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
import Image from "next/image";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
prefix: string;
|
||||||
|
name: string;
|
||||||
|
profileImage: string;
|
||||||
|
textSize?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function UserWithProfilePic({ prefix, name, profileImage, textSize }: Props) {
|
||||||
|
const textClassName = `${textSize ? textSize : "text-xs"} font-medium`
|
||||||
|
return (
|
||||||
|
<div className="flex items-center space-x-2">
|
||||||
|
<p className={textClassName}>{prefix} {name}</p>
|
||||||
|
<img
|
||||||
|
src={profileImage ? profileImage : "/defaultAvatar.png"}
|
||||||
|
alt={name}
|
||||||
|
width={24}
|
||||||
|
height={24}
|
||||||
|
className="rounded-full h-auto border-[1px] border-gray-400 border-opacity-50"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
@@ -0,0 +1,136 @@
|
|||||||
|
import { EditableWorkflowStep } from "@/interfaces/approval.workflow";
|
||||||
|
import Option from "@/interfaces/option";
|
||||||
|
import { CorporateUser, DeveloperUser, MasterCorporateUser, TeacherUser } from "@/interfaces/user";
|
||||||
|
import Image from "next/image";
|
||||||
|
import { useEffect, useMemo, useState } from "react";
|
||||||
|
import { AiOutlineUserAdd } from "react-icons/ai";
|
||||||
|
import { BsTrash } from "react-icons/bs";
|
||||||
|
import { LuGripHorizontal } from "react-icons/lu";
|
||||||
|
import WorkflowStepNumber from "./WorkflowStepNumber";
|
||||||
|
import WorkflowStepSelects from "./WorkflowStepSelects";
|
||||||
|
|
||||||
|
interface Props extends Pick<EditableWorkflowStep, 'stepNumber' | 'assignees' | 'finalStep' | 'onDelete'> {
|
||||||
|
entityApprovers: (TeacherUser | CorporateUser | MasterCorporateUser | DeveloperUser)[];
|
||||||
|
onSelectChange: (numberOfSelects: number, index: number, value: Option | null) => void;
|
||||||
|
isCompleted: boolean,
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function WorkflowEditableStepComponent({
|
||||||
|
stepNumber,
|
||||||
|
assignees = [null],
|
||||||
|
finalStep,
|
||||||
|
onDelete,
|
||||||
|
onSelectChange,
|
||||||
|
entityApprovers,
|
||||||
|
isCompleted,
|
||||||
|
}: Props) {
|
||||||
|
|
||||||
|
const [selects, setSelects] = useState<(Option | null | undefined)[]>([null]);
|
||||||
|
const [isAdding, setIsAdding] = useState(false);
|
||||||
|
|
||||||
|
const approverOptions: Option[] = useMemo(() =>
|
||||||
|
entityApprovers
|
||||||
|
.map((approver) => ({
|
||||||
|
value: approver.id,
|
||||||
|
label: approver.name,
|
||||||
|
icon: () => <img src={approver.profilePicture} alt={approver.name} />
|
||||||
|
}))
|
||||||
|
.sort((a, b) => a.label.localeCompare(b.label)),
|
||||||
|
[entityApprovers]
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (assignees && assignees.length > 0) {
|
||||||
|
const initialSelects = assignees.map((assignee) =>
|
||||||
|
typeof assignee === 'string' ? approverOptions.find(option => option.value === assignee) || null : null
|
||||||
|
);
|
||||||
|
|
||||||
|
setSelects((prevSelects) => {
|
||||||
|
// This is needed to avoid unnecessary re-renders which can cause warning of a child component being re-rendered while parent is in the midle of also re-rendering.
|
||||||
|
const areEqual = initialSelects.length === prevSelects.length && initialSelects.every((option, idx) => option?.value === prevSelects[idx]?.value);
|
||||||
|
|
||||||
|
if (!areEqual) {
|
||||||
|
return initialSelects;
|
||||||
|
}
|
||||||
|
return prevSelects;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}, [assignees, approverOptions]);
|
||||||
|
|
||||||
|
const selectedValues = useMemo(() =>
|
||||||
|
selects.filter((opt): opt is Option => !!opt).map(opt => opt.value),
|
||||||
|
[selects]
|
||||||
|
);
|
||||||
|
|
||||||
|
const availableApproverOptions = useMemo(() =>
|
||||||
|
approverOptions.filter(opt => !selectedValues.includes(opt.value)),
|
||||||
|
[approverOptions, selectedValues]
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleAddSelectComponent = () => {
|
||||||
|
setIsAdding(true); // I hate to use flags... but it was the only way i was able to prevent onSelectChange to cause parent component from re-rendering in the midle of EditableWorkflowStep rerender.
|
||||||
|
setSelects(prev => [...prev, null]);
|
||||||
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (isAdding) {
|
||||||
|
onSelectChange(selects.length, selects.length - 1, null);
|
||||||
|
setIsAdding(false);
|
||||||
|
}
|
||||||
|
}, [selects.length, isAdding, onSelectChange]);
|
||||||
|
|
||||||
|
const handleSelectChangeAt = (numberOfSelects: number, index: number, option: Option | null) => {
|
||||||
|
const updated = [...selects];
|
||||||
|
updated[index] = option;
|
||||||
|
setSelects(updated);
|
||||||
|
onSelectChange(numberOfSelects, index, option);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex w-full">
|
||||||
|
<div className="flex flex-col items-center">
|
||||||
|
<WorkflowStepNumber stepNumber={stepNumber} completed={false} selected={false} />
|
||||||
|
|
||||||
|
{/* Vertical Bar connecting steps */}
|
||||||
|
{!finalStep && (
|
||||||
|
<div className="w-1 h-full min-h-10 bg-mti-purple-dark"></div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{stepNumber !== 1 && !finalStep && !isCompleted
|
||||||
|
? <LuGripHorizontal className="ml-3 mt-2 cursor-grab active:cursor-grabbing min-w-[25px] min-h-[25px]" />
|
||||||
|
: <div className="ml-3 mt-2" style={{ width: 25, height: 25 }}></div>
|
||||||
|
}
|
||||||
|
|
||||||
|
<div className="ml-10 mb-12">
|
||||||
|
<WorkflowStepSelects
|
||||||
|
approvers={availableApproverOptions}
|
||||||
|
selects={selects}
|
||||||
|
placeholder={stepNumber === 1 ? "Form Intake By:" : "Approval By:"}
|
||||||
|
onSelectChange={handleSelectChangeAt}
|
||||||
|
isCompleted={isCompleted}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex flex-row items-start mt-1.5 ml-3">
|
||||||
|
<button
|
||||||
|
type="button"
|
||||||
|
onClick={handleAddSelectComponent}
|
||||||
|
className="cursor-pointer"
|
||||||
|
>
|
||||||
|
<AiOutlineUserAdd className="size-7 hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
|
</button>
|
||||||
|
{stepNumber !== 1 && !finalStep && (
|
||||||
|
<button
|
||||||
|
className="cursor-pointer"
|
||||||
|
onClick={onDelete}
|
||||||
|
type="button"
|
||||||
|
>
|
||||||
|
<BsTrash className="size-6 mt-0.5 ml-3 hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
);
|
||||||
|
};
|
||||||
203
src/components/ApprovalWorkflows/WorkflowForm.tsx
Normal file
203
src/components/ApprovalWorkflows/WorkflowForm.tsx
Normal file
@@ -0,0 +1,203 @@
|
|||||||
|
import { EditableApprovalWorkflow, EditableWorkflowStep } from "@/interfaces/approval.workflow";
|
||||||
|
import Option from "@/interfaces/option";
|
||||||
|
import { CorporateUser, DeveloperUser, MasterCorporateUser, TeacherUser } from "@/interfaces/user";
|
||||||
|
import { AnimatePresence, Reorder, motion } from "framer-motion";
|
||||||
|
import { FaRegCheckCircle, FaSpinner } from "react-icons/fa";
|
||||||
|
import { IoIosAddCircleOutline } from "react-icons/io";
|
||||||
|
import Button from "../Low/Button";
|
||||||
|
import Tip from "./Tip";
|
||||||
|
import WorkflowEditableStepComponent from "./WorkflowEditableStepComponent";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
workflow: EditableApprovalWorkflow;
|
||||||
|
onWorkflowChange: (workflow: EditableApprovalWorkflow) => void;
|
||||||
|
entityApprovers: (TeacherUser | CorporateUser | MasterCorporateUser | DeveloperUser)[];
|
||||||
|
entityAvailableFormIntakers?: (TeacherUser | CorporateUser | MasterCorporateUser | DeveloperUser)[];
|
||||||
|
isLoading: boolean;
|
||||||
|
isRedirecting?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function WorkflowForm({ workflow, onWorkflowChange, entityApprovers, entityAvailableFormIntakers, isLoading, isRedirecting }: Props) {
|
||||||
|
const lastStep = workflow.steps[workflow.steps.length - 1];
|
||||||
|
|
||||||
|
const renumberSteps = (steps: EditableWorkflowStep[]): EditableWorkflowStep[] => {
|
||||||
|
return steps.map((step, index) => ({
|
||||||
|
...step,
|
||||||
|
stepNumber: index + 1,
|
||||||
|
}));
|
||||||
|
};
|
||||||
|
|
||||||
|
const addStep = () => {
|
||||||
|
const newStep: EditableWorkflowStep = {
|
||||||
|
key: Date.now(),
|
||||||
|
stepType: "approval-by",
|
||||||
|
stepNumber: workflow.steps.length,
|
||||||
|
completed: false,
|
||||||
|
assignees: [null],
|
||||||
|
firstStep: false,
|
||||||
|
finalStep: false,
|
||||||
|
};
|
||||||
|
|
||||||
|
const updatedSteps = [
|
||||||
|
...workflow.steps.slice(0, -1),
|
||||||
|
newStep,
|
||||||
|
lastStep
|
||||||
|
];
|
||||||
|
onWorkflowChange({ ...workflow, steps: renumberSteps(updatedSteps) });
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleDelete = (key: number | undefined) => {
|
||||||
|
if (!key) return;
|
||||||
|
|
||||||
|
const updatedSteps = workflow.steps.filter((step) => step.key !== key);
|
||||||
|
onWorkflowChange({ ...workflow, steps: renumberSteps(updatedSteps) });
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleSelectChange = (key: number | undefined, numberOfSelects: number, index: number, selectedOption: Option | null) => {
|
||||||
|
if (!key) return;
|
||||||
|
|
||||||
|
const updatedSteps = workflow.steps.map((step) => {
|
||||||
|
if (step.key !== key) return step;
|
||||||
|
|
||||||
|
const assignees = step.assignees ?? [];
|
||||||
|
let newAssignees = [...assignees];
|
||||||
|
|
||||||
|
if (numberOfSelects === assignees.length) { // means no new select was added and instead one was changed
|
||||||
|
newAssignees[index] = selectedOption?.value;
|
||||||
|
} else if (numberOfSelects === assignees.length + 1) { // means a new select was added
|
||||||
|
newAssignees.push(selectedOption?.value || null);
|
||||||
|
}
|
||||||
|
|
||||||
|
return { ...step, assignees: newAssignees };
|
||||||
|
});
|
||||||
|
onWorkflowChange({ ...workflow, steps: updatedSteps });
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleReorder = (newOrder: EditableWorkflowStep[]) => {
|
||||||
|
let draggableIndex = 0;
|
||||||
|
const updatedSteps = workflow.steps.map((step) => {
|
||||||
|
if (!step.firstStep && !step.finalStep && !step.completed) {
|
||||||
|
return newOrder[draggableIndex++];
|
||||||
|
}
|
||||||
|
// Keep static steps as-is
|
||||||
|
return step;
|
||||||
|
});
|
||||||
|
onWorkflowChange({ ...workflow, steps: renumberSteps(updatedSteps) });
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{workflow.entityId && workflow.name &&
|
||||||
|
<div>
|
||||||
|
<div
|
||||||
|
className="flex flex-col gap-6"
|
||||||
|
>
|
||||||
|
<Tip text="Introduce here all the steps associated with this instance." />
|
||||||
|
<Button
|
||||||
|
color="purple"
|
||||||
|
variant="solid"
|
||||||
|
onClick={addStep}
|
||||||
|
type="button"
|
||||||
|
className="max-w-fit text-lg font-medium flex items-center gap-2 text-left mb-7"
|
||||||
|
>
|
||||||
|
<IoIosAddCircleOutline className="size-6" />
|
||||||
|
Add Step
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Reorder.Group
|
||||||
|
axis="y"
|
||||||
|
values={workflow.steps}
|
||||||
|
onReorder={handleReorder}
|
||||||
|
className="flex flex-col gap-0"
|
||||||
|
>
|
||||||
|
<AnimatePresence>
|
||||||
|
{workflow.steps.map((step, index) =>
|
||||||
|
step.completed || step.firstStep || step.finalStep ? (
|
||||||
|
<motion.div
|
||||||
|
key={step.key}
|
||||||
|
layout
|
||||||
|
initial={{ opacity: 0, y: -30 }}
|
||||||
|
animate={{ opacity: 1, y: 0 }}
|
||||||
|
exit={{ opacity: 0, x: 30 }}
|
||||||
|
transition={{ duration: 0.20 }}
|
||||||
|
>
|
||||||
|
<WorkflowEditableStepComponent
|
||||||
|
stepNumber={index + 1}
|
||||||
|
assignees={step.assignees}
|
||||||
|
finalStep={step.finalStep}
|
||||||
|
onDelete={() => handleDelete(step.key)}
|
||||||
|
onSelectChange={(numberOfSelects, idx, option) =>
|
||||||
|
handleSelectChange(step.key, numberOfSelects, idx, option)
|
||||||
|
}
|
||||||
|
entityApprovers={
|
||||||
|
step.stepNumber === 1 && entityAvailableFormIntakers
|
||||||
|
? entityAvailableFormIntakers
|
||||||
|
: entityApprovers
|
||||||
|
}
|
||||||
|
isCompleted={step.completed}
|
||||||
|
/>
|
||||||
|
</motion.div>
|
||||||
|
) : (
|
||||||
|
// Render non-completed steps as draggable items
|
||||||
|
<Reorder.Item
|
||||||
|
key={step.key}
|
||||||
|
value={step}
|
||||||
|
initial={{ opacity: 0, y: -30 }}
|
||||||
|
animate={{ opacity: 1, y: 0 }}
|
||||||
|
exit={{ opacity: 0, x: 30 }}
|
||||||
|
transition={{ duration: 0.20 }}
|
||||||
|
layout
|
||||||
|
drag={!step.firstStep && !step.finalStep}
|
||||||
|
dragListener={!step.firstStep && !step.finalStep}
|
||||||
|
>
|
||||||
|
<WorkflowEditableStepComponent
|
||||||
|
stepNumber={index + 1}
|
||||||
|
assignees={step.assignees}
|
||||||
|
finalStep={step.finalStep}
|
||||||
|
onDelete={() => handleDelete(step.key)}
|
||||||
|
onSelectChange={(numberOfSelects, idx, option) =>
|
||||||
|
handleSelectChange(step.key, numberOfSelects, idx, option)
|
||||||
|
}
|
||||||
|
entityApprovers={
|
||||||
|
step.stepNumber === 1 && entityAvailableFormIntakers
|
||||||
|
? entityAvailableFormIntakers
|
||||||
|
: entityApprovers
|
||||||
|
}
|
||||||
|
isCompleted={step.completed}
|
||||||
|
/>
|
||||||
|
</Reorder.Item>
|
||||||
|
)
|
||||||
|
)}
|
||||||
|
<Button
|
||||||
|
type="submit"
|
||||||
|
color="purple"
|
||||||
|
variant="solid"
|
||||||
|
disabled={isLoading}
|
||||||
|
className="max-w-fit text-lg font-medium flex items-center gap-2 text-left -mt-4"
|
||||||
|
>
|
||||||
|
{isRedirecting ? (
|
||||||
|
<>
|
||||||
|
<FaSpinner className="animate-spin size-5" />
|
||||||
|
Redirecting...
|
||||||
|
</>
|
||||||
|
) : isLoading ? (
|
||||||
|
<>
|
||||||
|
<FaSpinner className="animate-spin size-5" />
|
||||||
|
Loading...
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<FaRegCheckCircle className="size-5" />
|
||||||
|
Confirm Exam Workflow Pipeline
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
</AnimatePresence>
|
||||||
|
</Reorder.Group>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
};
|
||||||
101
src/components/ApprovalWorkflows/WorkflowStepComponent.tsx
Normal file
101
src/components/ApprovalWorkflows/WorkflowStepComponent.tsx
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
import { getUserTypeLabel, getUserTypeLabelShort, WorkflowStep } from "@/interfaces/approval.workflow";
|
||||||
|
import WorkflowStepNumber from "./WorkflowStepNumber";
|
||||||
|
import clsx from "clsx";
|
||||||
|
import { RiThumbUpLine } from "react-icons/ri";
|
||||||
|
import { FaWpforms } from "react-icons/fa6";
|
||||||
|
import { User } from "@/interfaces/user";
|
||||||
|
import UserWithProfilePic from "./UserWithProfilePic";
|
||||||
|
|
||||||
|
interface Props extends WorkflowStep {
|
||||||
|
workflowAssignees: User[],
|
||||||
|
currentStep: boolean,
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function WorkflowStepComponent({
|
||||||
|
workflowAssignees,
|
||||||
|
currentStep,
|
||||||
|
stepType,
|
||||||
|
stepNumber,
|
||||||
|
completed,
|
||||||
|
rejected = false,
|
||||||
|
completedBy,
|
||||||
|
assignees,
|
||||||
|
finalStep,
|
||||||
|
selected = false,
|
||||||
|
onClick,
|
||||||
|
}: Props) {
|
||||||
|
|
||||||
|
const completedByUser = workflowAssignees.find((assignee) => assignee.id === completedBy);
|
||||||
|
const assigneesUsers = workflowAssignees.filter(user => assignees.includes(user.id));
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
onClick={onClick}
|
||||||
|
className={clsx("flex flex-row gap-5 w-[600px] p-6 mb-5 rounded-2xl transition ease-in-out duration-300 cursor-pointer", {
|
||||||
|
"bg-mti-red-ultralight": rejected && selected,
|
||||||
|
"bg-mti-purple-ultralight": selected,
|
||||||
|
})}
|
||||||
|
>
|
||||||
|
<div className="relative flex flex-col items-center">
|
||||||
|
<WorkflowStepNumber stepNumber={stepNumber} selected={selected} completed={completed} finalStep={finalStep} rejected={rejected} />
|
||||||
|
|
||||||
|
{/* Vertical Bar connecting steps */}
|
||||||
|
{!finalStep && (
|
||||||
|
<div className="absolute w-1 bg-mti-purple-dark -bottom-20 top-11"></div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="mt-1.5">
|
||||||
|
{stepType === "approval-by" ? (
|
||||||
|
<RiThumbUpLine size={25} />
|
||||||
|
) : (
|
||||||
|
<FaWpforms size={25} />
|
||||||
|
)
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="mt-1 flex flex-col gap-0">
|
||||||
|
{completed && completedBy && rejected ? (
|
||||||
|
<div className={clsx("text-xs font-medium", { "text-mti-purple-ultradark": selected, "text-gray-800": !selected })}>
|
||||||
|
<p className="text-sm font-medium text-gray-800">{stepType === "approval-by" ? `Approval: ${getUserTypeLabel(completedByUser!.type)} Approval` : `Form Intake: ${getUserTypeLabel(completedByUser!.type)} Intake`} </p>
|
||||||
|
<UserWithProfilePic
|
||||||
|
prefix={`Rejected by: ${getUserTypeLabelShort(completedByUser!.type)}`}
|
||||||
|
name={completedByUser!.name}
|
||||||
|
profileImage={completedByUser!.profilePicture}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
) : completed && completedBy && !rejected ? (
|
||||||
|
<div className={clsx("text-xs font-medium", { "text-mti-purple-ultradark": selected, "text-gray-800": !selected })}>
|
||||||
|
<p className="text-sm font-medium text-gray-800">{stepType === "approval-by" ? `Approval: ${getUserTypeLabel(completedByUser!.type)} Approval` : `Form Intake: ${getUserTypeLabel(completedByUser!.type)} Intake`} </p>
|
||||||
|
<UserWithProfilePic
|
||||||
|
prefix={`Completed by: ${getUserTypeLabelShort(completedByUser!.type)}`}
|
||||||
|
name={completedByUser!.name}
|
||||||
|
profileImage={completedByUser!.profilePicture}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
) : !completed && currentStep ? (
|
||||||
|
<div className={clsx("text-xs font-medium", { "text-mti-purple-ultradark": selected, "text-gray-800": !selected })}>
|
||||||
|
<p className="text-sm font-medium text-gray-800">{stepType === "approval-by" ? `Approval:` : `Form Intake:`} </p>
|
||||||
|
In Progress... Assignees:
|
||||||
|
<div className="flex flex-row flex-wrap gap-3 items-center">
|
||||||
|
{assigneesUsers.map(user => (
|
||||||
|
<span key={user.id}>
|
||||||
|
<UserWithProfilePic
|
||||||
|
prefix={getUserTypeLabelShort(user.type)}
|
||||||
|
name={user.name}
|
||||||
|
profileImage={user.profilePicture}
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className={clsx("text-xs font-medium", { "text-mti-purple-ultradark": selected, "text-gray-800": !selected })}>
|
||||||
|
<p className="text-sm font-medium text-gray-800">{stepType === "approval-by" ? `Approval:` : `Form Intake:`} </p>
|
||||||
|
Waiting for previous steps...
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
31
src/components/ApprovalWorkflows/WorkflowStepNumber.tsx
Normal file
31
src/components/ApprovalWorkflows/WorkflowStepNumber.tsx
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
import { WorkflowStep } from "@/interfaces/approval.workflow";
|
||||||
|
import clsx from "clsx";
|
||||||
|
import { IoCheckmarkDoneSharp, IoCheckmarkSharp } from "react-icons/io5";
|
||||||
|
import { RxCross2 } from "react-icons/rx";
|
||||||
|
|
||||||
|
type Props = Pick<WorkflowStep, 'stepNumber' | 'completed' | 'finalStep' | 'selected' | 'rejected'>
|
||||||
|
|
||||||
|
export default function WorkflowStepNumber({ stepNumber, selected = false, completed, rejected, finalStep }: Props) {
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
'flex items-center justify-center min-w-11 min-h-11 rounded-full',
|
||||||
|
{
|
||||||
|
'bg-mti-red-dark text-mti-red-ultralight': rejected,
|
||||||
|
'bg-mti-purple-dark text-mti-purple-ultralight': selected,
|
||||||
|
'bg-mti-purple-ultralight text-gray-500': !selected,
|
||||||
|
}
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{rejected ? (
|
||||||
|
<RxCross2 className="text-xl font-bold" size={25}/>
|
||||||
|
) : completed && finalStep ? (
|
||||||
|
<IoCheckmarkDoneSharp className="text-xl font-bold" size={25} />
|
||||||
|
) : completed && !finalStep ? (
|
||||||
|
<IoCheckmarkSharp className="text-xl font-bold" size={25} />
|
||||||
|
) : (
|
||||||
|
<span className="text-lg font-semibold">{stepNumber}</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
51
src/components/ApprovalWorkflows/WorkflowStepSelects.tsx
Normal file
51
src/components/ApprovalWorkflows/WorkflowStepSelects.tsx
Normal file
@@ -0,0 +1,51 @@
|
|||||||
|
import Option from "@/interfaces/option";
|
||||||
|
import Select from "../Low/Select";
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
approvers: Option[];
|
||||||
|
selects: (Option | null | undefined)[];
|
||||||
|
placeholder: string;
|
||||||
|
onSelectChange: (numberOfSelects: number, index: number, value: Option | null) => void;
|
||||||
|
isCompleted: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function WorkflowStepSelects({
|
||||||
|
approvers,
|
||||||
|
selects,
|
||||||
|
placeholder,
|
||||||
|
onSelectChange,
|
||||||
|
isCompleted,
|
||||||
|
}: Props) {
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className={"flex flex-wrap gap-0"}
|
||||||
|
>
|
||||||
|
{selects.map((option, index) => {
|
||||||
|
let classes = "px-2 rounded-none";
|
||||||
|
if (index === 0 && selects.length === 1) {
|
||||||
|
classes += " rounded-l-2xl rounded-r-2xl";
|
||||||
|
} else if (index === 0) {
|
||||||
|
classes += " rounded-l-2xl";
|
||||||
|
} else if (index === selects.length - 1) {
|
||||||
|
classes += " rounded-r-2xl";
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div key={index} className="w-[275px]">
|
||||||
|
<Select
|
||||||
|
options={approvers}
|
||||||
|
value={option}
|
||||||
|
onChange={(option) => onSelectChange(selects.length, index, option)}
|
||||||
|
placeholder={placeholder}
|
||||||
|
flat
|
||||||
|
isClearable
|
||||||
|
className={classes}
|
||||||
|
disabled={isCompleted}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -1,17 +1,12 @@
|
|||||||
import {infoButtonStyle} from "@/constants/buttonStyles";
|
|
||||||
import {Module} from "@/interfaces";
|
|
||||||
import {User} from "@/interfaces/user";
|
import {User} from "@/interfaces/user";
|
||||||
import useExamStore from "@/stores/exam";
|
import useExamStore from "@/stores/exam";
|
||||||
import {getExam, getExamById} from "@/utils/exams";
|
import {getExam} from "@/utils/exams";
|
||||||
import {MODULE_ARRAY} from "@/utils/moduleUtils";
|
import {MODULE_ARRAY} from "@/utils/moduleUtils";
|
||||||
import {writingMarking} from "@/utils/score";
|
|
||||||
import {Menu} from "@headlessui/react";
|
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import {capitalize} from "lodash";
|
|
||||||
import {useRouter} from "next/router";
|
import {useRouter} from "next/router";
|
||||||
import {useEffect, useState} from "react";
|
import { useState} from "react";
|
||||||
import {BsBook, BsChevronDown, BsHeadphones, BsMegaphone, BsPen, BsQuestionSquare} from "react-icons/bs";
|
import { BsQuestionSquare} from "react-icons/bs";
|
||||||
import {toast} from "react-toastify";
|
import {toast} from "react-toastify";
|
||||||
import Button from "./Low/Button";
|
import Button from "./Low/Button";
|
||||||
import ModuleLevelSelector from "./Medium/ModuleLevelSelector";
|
import ModuleLevelSelector from "./Medium/ModuleLevelSelector";
|
||||||
|
|||||||
@@ -13,6 +13,7 @@ import validateBlanks from "../validateBlanks";
|
|||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
import setEditingAlert from "../../Shared/setEditingAlert";
|
import setEditingAlert from "../../Shared/setEditingAlert";
|
||||||
import PromptEdit from "../../Shared/PromptEdit";
|
import PromptEdit from "../../Shared/PromptEdit";
|
||||||
|
import { uuidv4 } from "@firebase/util";
|
||||||
|
|
||||||
interface Word {
|
interface Word {
|
||||||
letter: string;
|
letter: string;
|
||||||
@@ -72,6 +73,7 @@ const FillBlanksLetters: React.FC<{ exercise: FillBlanksExercise; sectionId: num
|
|||||||
...local,
|
...local,
|
||||||
text: blanksState.text,
|
text: blanksState.text,
|
||||||
solutions: Array.from(answers.entries()).map(([id, solution]) => ({
|
solutions: Array.from(answers.entries()).map(([id, solution]) => ({
|
||||||
|
uuid: local.solutions.find(sol => sol.id === id)?.uuid || uuidv4(),
|
||||||
id,
|
id,
|
||||||
solution
|
solution
|
||||||
}))
|
}))
|
||||||
@@ -145,6 +147,7 @@ const FillBlanksLetters: React.FC<{ exercise: FillBlanksExercise; sectionId: num
|
|||||||
setLocal(prev => ({
|
setLocal(prev => ({
|
||||||
...prev,
|
...prev,
|
||||||
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
||||||
|
uuid: prev.solutions.find(sol => sol.id === id)?.uuid || uuidv4(),
|
||||||
id,
|
id,
|
||||||
solution
|
solution
|
||||||
}))
|
}))
|
||||||
@@ -189,6 +192,7 @@ const FillBlanksLetters: React.FC<{ exercise: FillBlanksExercise; sectionId: num
|
|||||||
...prev,
|
...prev,
|
||||||
words: newWords,
|
words: newWords,
|
||||||
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
||||||
|
uuid: prev.solutions.find(sol => sol.id === id)?.uuid || uuidv4(),
|
||||||
id,
|
id,
|
||||||
solution
|
solution
|
||||||
}))
|
}))
|
||||||
@@ -217,6 +221,7 @@ const FillBlanksLetters: React.FC<{ exercise: FillBlanksExercise; sectionId: num
|
|||||||
...prev,
|
...prev,
|
||||||
words: newWords,
|
words: newWords,
|
||||||
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
||||||
|
uuid: prev.solutions.find(sol => sol.id === id)?.uuid || uuidv4(),
|
||||||
id,
|
id,
|
||||||
solution
|
solution
|
||||||
}))
|
}))
|
||||||
@@ -234,6 +239,7 @@ const FillBlanksLetters: React.FC<{ exercise: FillBlanksExercise; sectionId: num
|
|||||||
setLocal(prev => ({
|
setLocal(prev => ({
|
||||||
...prev,
|
...prev,
|
||||||
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
||||||
|
uuid: prev.solutions.find(sol => sol.id === id)?.uuid || uuidv4(),
|
||||||
id,
|
id,
|
||||||
solution
|
solution
|
||||||
}))
|
}))
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import { toast } from "react-toastify";
|
|||||||
import setEditingAlert from "../../Shared/setEditingAlert";
|
import setEditingAlert from "../../Shared/setEditingAlert";
|
||||||
import { MdEdit, MdEditOff } from "react-icons/md";
|
import { MdEdit, MdEditOff } from "react-icons/md";
|
||||||
import MCOption from "./MCOption";
|
import MCOption from "./MCOption";
|
||||||
|
import { uuidv4 } from "@firebase/util";
|
||||||
|
|
||||||
|
|
||||||
const FillBlanksMC: React.FC<{ exercise: FillBlanksExercise; sectionId: number }> = ({ exercise, sectionId }) => {
|
const FillBlanksMC: React.FC<{ exercise: FillBlanksExercise; sectionId: number }> = ({ exercise, sectionId }) => {
|
||||||
@@ -69,6 +70,7 @@ const FillBlanksMC: React.FC<{ exercise: FillBlanksExercise; sectionId: number }
|
|||||||
...local,
|
...local,
|
||||||
text: blanksState.text,
|
text: blanksState.text,
|
||||||
solutions: Array.from(answers.entries()).map(([id, solution]) => ({
|
solutions: Array.from(answers.entries()).map(([id, solution]) => ({
|
||||||
|
uuid: local.solutions.find(sol => sol.id === id)?.uuid || uuidv4(),
|
||||||
id,
|
id,
|
||||||
solution
|
solution
|
||||||
}))
|
}))
|
||||||
@@ -139,6 +141,7 @@ const FillBlanksMC: React.FC<{ exercise: FillBlanksExercise; sectionId: number }
|
|||||||
setLocal(prev => ({
|
setLocal(prev => ({
|
||||||
...prev,
|
...prev,
|
||||||
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
||||||
|
uuid: prev.solutions.find(sol => sol.id === id)?.uuid || uuidv4(),
|
||||||
id,
|
id,
|
||||||
solution
|
solution
|
||||||
}))
|
}))
|
||||||
@@ -168,6 +171,7 @@ const FillBlanksMC: React.FC<{ exercise: FillBlanksExercise; sectionId: number }
|
|||||||
...prev,
|
...prev,
|
||||||
words: newWords,
|
words: newWords,
|
||||||
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
||||||
|
uuid: prev.solutions.find(sol => sol.id === id)?.uuid || uuidv4(),
|
||||||
id,
|
id,
|
||||||
solution
|
solution
|
||||||
}))
|
}))
|
||||||
@@ -217,6 +221,7 @@ const FillBlanksMC: React.FC<{ exercise: FillBlanksExercise; sectionId: number }
|
|||||||
...prev,
|
...prev,
|
||||||
words: (prev.words as FillBlanksMCOption[]).filter(w => w.id !== blankId.toString()),
|
words: (prev.words as FillBlanksMCOption[]).filter(w => w.id !== blankId.toString()),
|
||||||
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
solutions: Array.from(newAnswers.entries()).map(([id, solution]) => ({
|
||||||
|
uuid: prev.solutions.find(sol => sol.id === id)?.uuid || uuidv4(),
|
||||||
id,
|
id,
|
||||||
solution
|
solution
|
||||||
}))
|
}))
|
||||||
@@ -234,6 +239,7 @@ const FillBlanksMC: React.FC<{ exercise: FillBlanksExercise; sectionId: number }
|
|||||||
|
|
||||||
blanksMissingWords.forEach(blank => {
|
blanksMissingWords.forEach(blank => {
|
||||||
const newMCOption: FillBlanksMCOption = {
|
const newMCOption: FillBlanksMCOption = {
|
||||||
|
uuid: uuidv4(),
|
||||||
id: blank.id.toString(),
|
id: blank.id.toString(),
|
||||||
options: {
|
options: {
|
||||||
A: 'Option A',
|
A: 'Option A',
|
||||||
@@ -249,6 +255,7 @@ const FillBlanksMC: React.FC<{ exercise: FillBlanksExercise; sectionId: number }
|
|||||||
...prev,
|
...prev,
|
||||||
words: newWords,
|
words: newWords,
|
||||||
solutions: Array.from(answers.entries()).map(([id, solution]) => ({
|
solutions: Array.from(answers.entries()).map(([id, solution]) => ({
|
||||||
|
uuid: prev.solutions.find(sol => sol.id === id)?.uuid || uuidv4(),
|
||||||
id,
|
id,
|
||||||
solution
|
solution
|
||||||
}))
|
}))
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ import { toast } from 'react-toastify';
|
|||||||
import { DragEndEvent } from '@dnd-kit/core';
|
import { DragEndEvent } from '@dnd-kit/core';
|
||||||
import { handleMatchSentencesReorder } from '@/stores/examEditor/reorder/local';
|
import { handleMatchSentencesReorder } from '@/stores/examEditor/reorder/local';
|
||||||
import PromptEdit from '../Shared/PromptEdit';
|
import PromptEdit from '../Shared/PromptEdit';
|
||||||
|
import { uuidv4 } from '@firebase/util';
|
||||||
|
|
||||||
const MatchSentences: React.FC<{ exercise: MatchSentencesExercise, sectionId: number }> = ({ exercise, sectionId }) => {
|
const MatchSentences: React.FC<{ exercise: MatchSentencesExercise, sectionId: number }> = ({ exercise, sectionId }) => {
|
||||||
const { currentModule, dispatch } = useExamEditorStore();
|
const { currentModule, dispatch } = useExamEditorStore();
|
||||||
@@ -98,6 +99,7 @@ const MatchSentences: React.FC<{ exercise: MatchSentencesExercise, sectionId: nu
|
|||||||
sentences: [
|
sentences: [
|
||||||
...local.sentences,
|
...local.sentences,
|
||||||
{
|
{
|
||||||
|
uuid: uuidv4(),
|
||||||
id: newId,
|
id: newId,
|
||||||
sentence: "",
|
sentence: "",
|
||||||
solution: ""
|
solution: ""
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import { useCallback, useEffect, useState } from "react";
|
|||||||
import { MdAdd } from "react-icons/md";
|
import { MdAdd } from "react-icons/md";
|
||||||
import Alert, { AlertItem } from "../../Shared/Alert";
|
import Alert, { AlertItem } from "../../Shared/Alert";
|
||||||
import PromptEdit from "../../Shared/PromptEdit";
|
import PromptEdit from "../../Shared/PromptEdit";
|
||||||
|
import { uuidv4 } from "@firebase/util";
|
||||||
|
|
||||||
|
|
||||||
const UnderlineMultipleChoice: React.FC<{exercise: MultipleChoiceExercise, sectionId: number}> = ({
|
const UnderlineMultipleChoice: React.FC<{exercise: MultipleChoiceExercise, sectionId: number}> = ({
|
||||||
@@ -57,6 +58,7 @@ const UnderlineMultipleChoice: React.FC<{exercise: MultipleChoiceExercise, secti
|
|||||||
{
|
{
|
||||||
prompt: "",
|
prompt: "",
|
||||||
solution: "",
|
solution: "",
|
||||||
|
uuid: uuidv4(),
|
||||||
id: newId,
|
id: newId,
|
||||||
options,
|
options,
|
||||||
variant: "text"
|
variant: "text"
|
||||||
|
|||||||
@@ -18,6 +18,7 @@ import SortableQuestion from '../../Shared/SortableQuestion';
|
|||||||
import setEditingAlert from '../../Shared/setEditingAlert';
|
import setEditingAlert from '../../Shared/setEditingAlert';
|
||||||
import { handleMultipleChoiceReorder } from '@/stores/examEditor/reorder/local';
|
import { handleMultipleChoiceReorder } from '@/stores/examEditor/reorder/local';
|
||||||
import PromptEdit from '../../Shared/PromptEdit';
|
import PromptEdit from '../../Shared/PromptEdit';
|
||||||
|
import { uuidv4 } from '@firebase/util';
|
||||||
|
|
||||||
interface MultipleChoiceProps {
|
interface MultipleChoiceProps {
|
||||||
exercise: MultipleChoiceExercise;
|
exercise: MultipleChoiceExercise;
|
||||||
@@ -120,6 +121,7 @@ const MultipleChoice: React.FC<MultipleChoiceProps> = ({ exercise, sectionId, op
|
|||||||
{
|
{
|
||||||
prompt: "",
|
prompt: "",
|
||||||
solution: "",
|
solution: "",
|
||||||
|
uuid: uuidv4(),
|
||||||
id: newId,
|
id: newId,
|
||||||
options,
|
options,
|
||||||
variant: "text"
|
variant: "text"
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ import setEditingAlert from '../Shared/setEditingAlert';
|
|||||||
import { DragEndEvent } from '@dnd-kit/core';
|
import { DragEndEvent } from '@dnd-kit/core';
|
||||||
import { handleTrueFalseReorder } from '@/stores/examEditor/reorder/local';
|
import { handleTrueFalseReorder } from '@/stores/examEditor/reorder/local';
|
||||||
import PromptEdit from '../Shared/PromptEdit';
|
import PromptEdit from '../Shared/PromptEdit';
|
||||||
|
import { uuidv4 } from '@firebase/util';
|
||||||
|
|
||||||
const TrueFalse: React.FC<{ exercise: TrueFalseExercise, sectionId: number }> = ({ exercise, sectionId }) => {
|
const TrueFalse: React.FC<{ exercise: TrueFalseExercise, sectionId: number }> = ({ exercise, sectionId }) => {
|
||||||
const { currentModule, dispatch } = useExamEditorStore();
|
const { currentModule, dispatch } = useExamEditorStore();
|
||||||
@@ -50,6 +51,7 @@ const TrueFalse: React.FC<{ exercise: TrueFalseExercise, sectionId: number }> =
|
|||||||
{
|
{
|
||||||
prompt: "",
|
prompt: "",
|
||||||
solution: undefined,
|
solution: undefined,
|
||||||
|
uuid: uuidv4(),
|
||||||
id: newId
|
id: newId
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
|
|||||||
@@ -22,6 +22,7 @@ import { validateEmptySolutions, validateQuestionText, validateWordCount } from
|
|||||||
import { handleWriteBlanksReorder } from '@/stores/examEditor/reorder/local';
|
import { handleWriteBlanksReorder } from '@/stores/examEditor/reorder/local';
|
||||||
import { ParsedQuestion, parseText, reconstructText } from './parsing';
|
import { ParsedQuestion, parseText, reconstructText } from './parsing';
|
||||||
import PromptEdit from '../Shared/PromptEdit';
|
import PromptEdit from '../Shared/PromptEdit';
|
||||||
|
import { uuidv4 } from '@firebase/util';
|
||||||
|
|
||||||
|
|
||||||
const WriteBlanks: React.FC<{ sectionId: number; exercise: WriteBlanksExercise; }> = ({ sectionId, exercise }) => {
|
const WriteBlanks: React.FC<{ sectionId: number; exercise: WriteBlanksExercise; }> = ({ sectionId, exercise }) => {
|
||||||
@@ -105,6 +106,7 @@ const WriteBlanks: React.FC<{ sectionId: number; exercise: WriteBlanksExercise;
|
|||||||
const newId = (Math.max(...existingIds, 0) + 1).toString();
|
const newId = (Math.max(...existingIds, 0) + 1).toString();
|
||||||
|
|
||||||
const newQuestion = {
|
const newQuestion = {
|
||||||
|
uuid: uuidv4(),
|
||||||
id: newId,
|
id: newId,
|
||||||
questionText: "New question"
|
questionText: "New question"
|
||||||
};
|
};
|
||||||
@@ -113,6 +115,7 @@ const WriteBlanks: React.FC<{ sectionId: number; exercise: WriteBlanksExercise;
|
|||||||
const updatedText = reconstructText(updatedQuestions);
|
const updatedText = reconstructText(updatedQuestions);
|
||||||
|
|
||||||
const updatedSolutions = [...local.solutions, {
|
const updatedSolutions = [...local.solutions, {
|
||||||
|
uuid: uuidv4(),
|
||||||
id: newId,
|
id: newId,
|
||||||
solution: [""]
|
solution: [""]
|
||||||
}];
|
}];
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ import { validateQuestions, validateEmptySolutions, validateWordCount } from "./
|
|||||||
import Header from "../../Shared/Header";
|
import Header from "../../Shared/Header";
|
||||||
import BlanksFormEditor from "./BlanksFormEditor";
|
import BlanksFormEditor from "./BlanksFormEditor";
|
||||||
import PromptEdit from "../Shared/PromptEdit";
|
import PromptEdit from "../Shared/PromptEdit";
|
||||||
|
import { uuidv4 } from "@firebase/util";
|
||||||
|
|
||||||
|
|
||||||
const WriteBlanksForm: React.FC<{ sectionId: number; exercise: WriteBlanksExercise }> = ({ sectionId, exercise }) => {
|
const WriteBlanksForm: React.FC<{ sectionId: number; exercise: WriteBlanksExercise }> = ({ sectionId, exercise }) => {
|
||||||
@@ -111,6 +112,7 @@ const WriteBlanksForm: React.FC<{ sectionId: number; exercise: WriteBlanksExerci
|
|||||||
|
|
||||||
const newLine = `New question with blank {{${newId}}}`;
|
const newLine = `New question with blank {{${newId}}}`;
|
||||||
const updatedQuestions = [...parsedQuestions, {
|
const updatedQuestions = [...parsedQuestions, {
|
||||||
|
uuid: uuidv4(),
|
||||||
id: newId,
|
id: newId,
|
||||||
parts: parseLine(newLine),
|
parts: parseLine(newLine),
|
||||||
editingPlaceholders: true
|
editingPlaceholders: true
|
||||||
@@ -121,6 +123,7 @@ const WriteBlanksForm: React.FC<{ sectionId: number; exercise: WriteBlanksExerci
|
|||||||
.join('\\n') + '\\n';
|
.join('\\n') + '\\n';
|
||||||
|
|
||||||
const updatedSolutions = [...local.solutions, {
|
const updatedSolutions = [...local.solutions, {
|
||||||
|
uuid: uuidv4(),
|
||||||
id: newId,
|
id: newId,
|
||||||
solution: [""]
|
solution: [""]
|
||||||
}];
|
}];
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ interface SettingsEditorProps {
|
|||||||
children?: ReactNode;
|
children?: ReactNode;
|
||||||
canPreview: boolean;
|
canPreview: boolean;
|
||||||
canSubmit: boolean;
|
canSubmit: boolean;
|
||||||
submitModule: () => void;
|
submitModule: (requiresApproval: boolean) => void;
|
||||||
preview: () => void;
|
preview: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -95,7 +95,7 @@ const SettingsEditor: React.FC<SettingsEditorProps> = ({
|
|||||||
}, [updateLocalAndScheduleGlobal]);
|
}, [updateLocalAndScheduleGlobal]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={`flex flex-col gap-8 border bg-ielts-${module}/20 rounded-3xl p-8 w-1/3 h-fit`}>
|
<div className={`flex flex-col gap-8 border bg-ielts-${module}/20 rounded-3xl p-8 w-1/3 h-fit -2xl:w-full`}>
|
||||||
<div className={`w-full flex justify-center text-ielts-${module} font-bold text-xl`}>{sectionLabel} Settings</div>
|
<div className={`w-full flex justify-center text-ielts-${module} font-bold text-xl`}>{sectionLabel} Settings</div>
|
||||||
<div className="flex flex-col gap-4">
|
<div className="flex flex-col gap-4">
|
||||||
<Dropdown
|
<Dropdown
|
||||||
@@ -148,18 +148,33 @@ const SettingsEditor: React.FC<SettingsEditorProps> = ({
|
|||||||
</div>
|
</div>
|
||||||
</Dropdown>
|
</Dropdown>
|
||||||
{children}
|
{children}
|
||||||
<div className="flex flex-row justify-between mt-4">
|
<div className="flex flex-col gap-3 mt-4">
|
||||||
<button
|
<button
|
||||||
className={clsx(
|
className={clsx(
|
||||||
"flex items-center justify-center px-4 py-2 text-white rounded-xl transition-colors duration-300",
|
"flex items-center justify-center px-4 py-2 text-white rounded-xl transition-colors duration-300",
|
||||||
`bg-ielts-${module}/70 border border-ielts-${module} hover:bg-ielts-${module} disabled:bg-ielts-${module}/30`,
|
`bg-ielts-${module}/70 border border-ielts-${module} hover:bg-ielts-${module} disabled:bg-ielts-${module}/30`,
|
||||||
"disabled:cursor-not-allowed disabled:text-gray-200"
|
"disabled:cursor-not-allowed disabled:text-gray-200"
|
||||||
)}
|
)}
|
||||||
onClick={submitModule}
|
onClick={() => submitModule(true)}
|
||||||
disabled={!canSubmit}
|
disabled={!canSubmit}
|
||||||
>
|
>
|
||||||
<FaFileUpload className="mr-2" size={18} />
|
<FaFileUpload className="mr-2" size={18} />
|
||||||
Submit Module as Exam
|
Submit module as exam for approval
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
className={clsx(
|
||||||
|
"flex items-center justify-center px-4 py-2 text-white rounded-xl transition-colors duration-300",
|
||||||
|
`bg-ielts-${module}/70 border border-ielts-${module} hover:bg-ielts-${module} disabled:bg-ielts-${module}/30`,
|
||||||
|
"disabled:cursor-not-allowed disabled:text-gray-200"
|
||||||
|
)}
|
||||||
|
onClick={() => {
|
||||||
|
if (!confirm(`Are you sure you want to skip the approval process for this exam?`)) return;
|
||||||
|
submitModule(false);
|
||||||
|
}}
|
||||||
|
disabled={!canSubmit}
|
||||||
|
>
|
||||||
|
<FaFileUpload className="mr-2" size={18} />
|
||||||
|
Submit module as exam and skip approval process
|
||||||
</button>
|
</button>
|
||||||
<button
|
<button
|
||||||
className={clsx(
|
className={clsx(
|
||||||
@@ -171,7 +186,7 @@ const SettingsEditor: React.FC<SettingsEditorProps> = ({
|
|||||||
disabled={!canPreview}
|
disabled={!canPreview}
|
||||||
>
|
>
|
||||||
<FaEye className="mr-2" size={18} />
|
<FaEye className="mr-2" size={18} />
|
||||||
Preview Module
|
Preview module
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -17,6 +17,7 @@ import ListeningComponents from "./listening/components";
|
|||||||
import ReadingComponents from "./reading/components";
|
import ReadingComponents from "./reading/components";
|
||||||
import SpeakingComponents from "./speaking/components";
|
import SpeakingComponents from "./speaking/components";
|
||||||
import SectionPicker from "./Shared/SectionPicker";
|
import SectionPicker from "./Shared/SectionPicker";
|
||||||
|
import { getExamById } from "@/utils/exams";
|
||||||
|
|
||||||
|
|
||||||
const LevelSettings: React.FC = () => {
|
const LevelSettings: React.FC = () => {
|
||||||
@@ -37,7 +38,7 @@ const LevelSettings: React.FC = () => {
|
|||||||
difficulty,
|
difficulty,
|
||||||
sections,
|
sections,
|
||||||
minTimer,
|
minTimer,
|
||||||
isPrivate,
|
access,
|
||||||
} = useExamEditorStore(state => state.modules[currentModule]);
|
} = useExamEditorStore(state => state.modules[currentModule]);
|
||||||
|
|
||||||
const { localSettings, updateLocalAndScheduleGlobal } = useSettingsState<LevelSectionSettings>(
|
const { localSettings, updateLocalAndScheduleGlobal } = useSettingsState<LevelSectionSettings>(
|
||||||
@@ -75,7 +76,7 @@ const LevelSettings: React.FC = () => {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
const submitLevel = async () => {
|
const submitLevel = async (requiresApproval: boolean) => {
|
||||||
if (title === "") {
|
if (title === "") {
|
||||||
toast.error("Enter a title for the exam!");
|
toast.error("Enter a title for the exam!");
|
||||||
return;
|
return;
|
||||||
@@ -194,17 +195,23 @@ const LevelSettings: React.FC = () => {
|
|||||||
category: s.settings.category
|
category: s.settings.category
|
||||||
};
|
};
|
||||||
}).filter(part => part.exercises.length > 0),
|
}).filter(part => part.exercises.length > 0),
|
||||||
|
requiresApproval: requiresApproval,
|
||||||
isDiagnostic: false,
|
isDiagnostic: false,
|
||||||
minTimer,
|
minTimer,
|
||||||
module: "level",
|
module: "level",
|
||||||
id: title,
|
id: title,
|
||||||
difficulty,
|
difficulty,
|
||||||
private: isPrivate,
|
access,
|
||||||
};
|
};
|
||||||
|
|
||||||
const result = await axios.post('/api/exam/level', exam);
|
const result = await axios.post('/api/exam/level', exam);
|
||||||
playSound("sent");
|
playSound("sent");
|
||||||
toast.success(`Submitted Exam ID: ${result.data.id}`);
|
// Successfully submitted exam
|
||||||
|
if (result.status === 200) {
|
||||||
|
toast.success(result.data.message);
|
||||||
|
} else if (result.status === 207) {
|
||||||
|
toast.warning(result.data.message);
|
||||||
|
}
|
||||||
|
|
||||||
Array.from(audioMap.values()).forEach(url => {
|
Array.from(audioMap.values()).forEach(url => {
|
||||||
URL.revokeObjectURL(url);
|
URL.revokeObjectURL(url);
|
||||||
@@ -237,7 +244,7 @@ const LevelSettings: React.FC = () => {
|
|||||||
isDiagnostic: false,
|
isDiagnostic: false,
|
||||||
variant: undefined,
|
variant: undefined,
|
||||||
difficulty,
|
difficulty,
|
||||||
private: isPrivate,
|
access,
|
||||||
} as LevelExam);
|
} as LevelExam);
|
||||||
setExerciseIndex(0);
|
setExerciseIndex(0);
|
||||||
setQuestionIndex(0);
|
setQuestionIndex(0);
|
||||||
|
|||||||
@@ -233,7 +233,7 @@ const ListeningComponents: React.FC<Props> = ({ currentSection, localSettings, u
|
|||||||
setIsOpen={(isOpen: boolean) => updateLocalAndScheduleGlobal({ isAudioContextOpen: isOpen }, false)}
|
setIsOpen={(isOpen: boolean) => updateLocalAndScheduleGlobal({ isAudioContextOpen: isOpen }, false)}
|
||||||
contentWrapperClassName={level ? `border border-ielts-listening` : ''}
|
contentWrapperClassName={level ? `border border-ielts-listening` : ''}
|
||||||
>
|
>
|
||||||
<div className="flex flex-row gap-2 items-center px-2 pb-4">
|
<div className="flex flex-row flex-wrap gap-2 items-center justify-center px-2 pb-4">
|
||||||
<div className="flex flex-col flex-grow gap-4 px-2">
|
<div className="flex flex-col flex-grow gap-4 px-2">
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Topic (Optional)</label>
|
<label className="font-normal text-base text-mti-gray-dim">Topic (Optional)</label>
|
||||||
<Input
|
<Input
|
||||||
|
|||||||
@@ -1,15 +1,9 @@
|
|||||||
import Dropdown from "../Shared/SettingsDropdown";
|
|
||||||
import ExercisePicker from "../../ExercisePicker";
|
|
||||||
import SettingsEditor from "..";
|
import SettingsEditor from "..";
|
||||||
import GenerateBtn from "../Shared/GenerateBtn";
|
import { ListeningSectionSettings } from "@/stores/examEditor/types";
|
||||||
import { useCallback, useState } from "react";
|
|
||||||
import { generate } from "../Shared/Generate";
|
|
||||||
import { Generating, LevelSectionSettings, ListeningSectionSettings } from "@/stores/examEditor/types";
|
|
||||||
import Option from "@/interfaces/option";
|
import Option from "@/interfaces/option";
|
||||||
import useExamEditorStore from "@/stores/examEditor";
|
import useExamEditorStore from "@/stores/examEditor";
|
||||||
import useSettingsState from "../../Hooks/useSettingsState";
|
import useSettingsState from "../../Hooks/useSettingsState";
|
||||||
import { ListeningExam, ListeningPart } from "@/interfaces/exam";
|
import { ListeningExam, ListeningPart } from "@/interfaces/exam";
|
||||||
import Input from "@/components/Low/Input";
|
|
||||||
import openDetachedTab from "@/utils/popout";
|
import openDetachedTab from "@/utils/popout";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
@@ -26,7 +20,7 @@ const ListeningSettings: React.FC = () => {
|
|||||||
difficulty,
|
difficulty,
|
||||||
sections,
|
sections,
|
||||||
minTimer,
|
minTimer,
|
||||||
isPrivate,
|
access,
|
||||||
instructionsState
|
instructionsState
|
||||||
} = useExamEditorStore(state => state.modules[currentModule]);
|
} = useExamEditorStore(state => state.modules[currentModule]);
|
||||||
|
|
||||||
@@ -64,7 +58,7 @@ const ListeningSettings: React.FC = () => {
|
|||||||
}
|
}
|
||||||
];
|
];
|
||||||
|
|
||||||
const submitListening = async () => {
|
const submitListening = async (requiresApproval: boolean) => {
|
||||||
if (title === "") {
|
if (title === "") {
|
||||||
toast.error("Enter a title for the exam!");
|
toast.error("Enter a title for the exam!");
|
||||||
return;
|
return;
|
||||||
@@ -137,19 +131,25 @@ const ListeningSettings: React.FC = () => {
|
|||||||
category: s.settings.category
|
category: s.settings.category
|
||||||
};
|
};
|
||||||
}),
|
}),
|
||||||
|
requiresApproval: requiresApproval,
|
||||||
isDiagnostic: false,
|
isDiagnostic: false,
|
||||||
minTimer,
|
minTimer,
|
||||||
module: "listening",
|
module: "listening",
|
||||||
id: title,
|
id: title,
|
||||||
variant: sections.length === 4 ? "full" : "partial",
|
variant: sections.length === 4 ? "full" : "partial",
|
||||||
difficulty,
|
difficulty,
|
||||||
private: isPrivate,
|
access,
|
||||||
instructions: instructionsURL
|
instructions: instructionsURL
|
||||||
};
|
};
|
||||||
|
|
||||||
const result = await axios.post('/api/exam/listening', exam);
|
const result = await axios.post('/api/exam/listening', exam);
|
||||||
playSound("sent");
|
playSound("sent");
|
||||||
toast.success(`Submitted Exam ID: ${result.data.id}`);
|
// Successfully submitted exam
|
||||||
|
if (result.status === 200) {
|
||||||
|
toast.success(result.data.message);
|
||||||
|
} else if (result.status === 207) {
|
||||||
|
toast.warning(result.data.message);
|
||||||
|
}
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
toast.error('No audio sections found in the exam! Please either import them or generate them.');
|
toast.error('No audio sections found in the exam! Please either import them or generate them.');
|
||||||
@@ -185,7 +185,7 @@ const ListeningSettings: React.FC = () => {
|
|||||||
isDiagnostic: false,
|
isDiagnostic: false,
|
||||||
variant: sections.length === 4 ? "full" : "partial",
|
variant: sections.length === 4 ? "full" : "partial",
|
||||||
difficulty,
|
difficulty,
|
||||||
private: isPrivate,
|
access,
|
||||||
instructions: instructionsState.currentInstructionsURL
|
instructions: instructionsState.currentInstructionsURL
|
||||||
} as ListeningExam);
|
} as ListeningExam);
|
||||||
setExerciseIndex(0);
|
setExerciseIndex(0);
|
||||||
|
|||||||
@@ -5,103 +5,140 @@ import ExercisePicker from "../../ExercisePicker";
|
|||||||
import { generate } from "../Shared/Generate";
|
import { generate } from "../Shared/Generate";
|
||||||
import GenerateBtn from "../Shared/GenerateBtn";
|
import GenerateBtn from "../Shared/GenerateBtn";
|
||||||
import { LevelPart, ReadingPart } from "@/interfaces/exam";
|
import { LevelPart, ReadingPart } from "@/interfaces/exam";
|
||||||
import { LevelSectionSettings, ReadingSectionSettings } from "@/stores/examEditor/types";
|
import {
|
||||||
|
LevelSectionSettings,
|
||||||
|
ReadingSectionSettings,
|
||||||
|
} from "@/stores/examEditor/types";
|
||||||
import useExamEditorStore from "@/stores/examEditor";
|
import useExamEditorStore from "@/stores/examEditor";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
localSettings: ReadingSectionSettings | LevelSectionSettings;
|
localSettings: ReadingSectionSettings | LevelSectionSettings;
|
||||||
updateLocalAndScheduleGlobal: (updates: Partial<ReadingSectionSettings | LevelSectionSettings>, schedule?: boolean) => void;
|
updateLocalAndScheduleGlobal: (
|
||||||
currentSection: ReadingPart | LevelPart;
|
updates: Partial<ReadingSectionSettings | LevelSectionSettings>,
|
||||||
generatePassageDisabled?: boolean;
|
schedule?: boolean
|
||||||
levelId?: number;
|
) => void;
|
||||||
level?: boolean;
|
currentSection: ReadingPart | LevelPart;
|
||||||
|
generatePassageDisabled?: boolean;
|
||||||
|
levelId?: number;
|
||||||
|
level?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
const ReadingComponents: React.FC<Props> = ({localSettings, updateLocalAndScheduleGlobal, currentSection, levelId, level = false, generatePassageDisabled = false}) => {
|
const ReadingComponents: React.FC<Props> = ({
|
||||||
const { currentModule } = useExamEditorStore();
|
localSettings,
|
||||||
const {
|
updateLocalAndScheduleGlobal,
|
||||||
focusedSection,
|
currentSection,
|
||||||
difficulty,
|
levelId,
|
||||||
} = useExamEditorStore(state => state.modules[currentModule]);
|
level = false,
|
||||||
|
generatePassageDisabled = false,
|
||||||
|
}) => {
|
||||||
|
const { currentModule } = useExamEditorStore();
|
||||||
|
const { focusedSection, difficulty } = useExamEditorStore(
|
||||||
|
(state) => state.modules[currentModule]
|
||||||
|
);
|
||||||
|
|
||||||
const generatePassage = useCallback(() => {
|
const generatePassage = useCallback(() => {
|
||||||
generate(
|
generate(
|
||||||
levelId ? levelId : focusedSection,
|
levelId ? levelId : focusedSection,
|
||||||
"reading",
|
"reading",
|
||||||
"passage",
|
"passage",
|
||||||
{
|
{
|
||||||
method: 'GET',
|
method: "GET",
|
||||||
queryParams: {
|
queryParams: {
|
||||||
difficulty,
|
difficulty,
|
||||||
...(localSettings.readingTopic && { topic: localSettings.readingTopic })
|
...(localSettings.readingTopic && {
|
||||||
}
|
topic: localSettings.readingTopic,
|
||||||
},
|
}),
|
||||||
(data: any) => [{
|
},
|
||||||
title: data.title,
|
},
|
||||||
text: data.text
|
(data: any) => [
|
||||||
}],
|
{
|
||||||
level ? focusedSection : undefined,
|
title: data.title,
|
||||||
level
|
text: data.text,
|
||||||
);
|
},
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
],
|
||||||
}, [localSettings.readingTopic, difficulty, focusedSection, levelId]);
|
level ? focusedSection : undefined,
|
||||||
|
level
|
||||||
const onTopicChange = useCallback((readingTopic: string) => {
|
|
||||||
updateLocalAndScheduleGlobal({ readingTopic });
|
|
||||||
}, [updateLocalAndScheduleGlobal]);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<>
|
|
||||||
<Dropdown
|
|
||||||
title="Generate Passage"
|
|
||||||
module="reading"
|
|
||||||
open={localSettings.isPassageOpen}
|
|
||||||
setIsOpen={(isOpen: boolean) => updateLocalAndScheduleGlobal({ isPassageOpen: isOpen }, false)}
|
|
||||||
contentWrapperClassName={level ? `border border-ielts-reading`: ''}
|
|
||||||
disabled={generatePassageDisabled}
|
|
||||||
>
|
|
||||||
<div className="flex flex-row gap-2 items-center px-2 pb-4">
|
|
||||||
<div className="flex flex-col flex-grow gap-4 px-2">
|
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Topic (Optional)</label>
|
|
||||||
<Input
|
|
||||||
key={`section-${focusedSection}`}
|
|
||||||
type="text"
|
|
||||||
placeholder="Topic"
|
|
||||||
name="category"
|
|
||||||
onChange={onTopicChange}
|
|
||||||
roundness="full"
|
|
||||||
value={localSettings.readingTopic}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className="flex self-end h-16 mb-1">
|
|
||||||
<GenerateBtn
|
|
||||||
module="reading"
|
|
||||||
genType="passage"
|
|
||||||
sectionId={focusedSection}
|
|
||||||
generateFnc={generatePassage}
|
|
||||||
level={level}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</Dropdown>
|
|
||||||
<Dropdown
|
|
||||||
title="Add Exercises"
|
|
||||||
module="reading"
|
|
||||||
open={localSettings.isReadingTopicOpean}
|
|
||||||
setIsOpen={(isOpen: boolean) => updateLocalAndScheduleGlobal({ isReadingTopicOpean: isOpen })}
|
|
||||||
contentWrapperClassName={level ? `border border-ielts-reading`: ''}
|
|
||||||
disabled={currentSection === undefined || currentSection.text === undefined || currentSection.text.content === "" || currentSection.text.title === ""}
|
|
||||||
>
|
|
||||||
<ExercisePicker
|
|
||||||
module="reading"
|
|
||||||
sectionId={levelId !== undefined ? levelId : focusedSection}
|
|
||||||
extraArgs={{ text: currentSection === undefined || currentSection.text === undefined ? "" : currentSection.text.content }}
|
|
||||||
levelSectionId={focusedSection}
|
|
||||||
level={level}
|
|
||||||
/>
|
|
||||||
</Dropdown>
|
|
||||||
</>
|
|
||||||
);
|
);
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, [localSettings.readingTopic, difficulty, focusedSection, levelId]);
|
||||||
|
|
||||||
|
const onTopicChange = useCallback(
|
||||||
|
(readingTopic: string) => {
|
||||||
|
updateLocalAndScheduleGlobal({ readingTopic });
|
||||||
|
},
|
||||||
|
[updateLocalAndScheduleGlobal]
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Dropdown
|
||||||
|
title="Generate Passage"
|
||||||
|
module="reading"
|
||||||
|
open={localSettings.isPassageOpen}
|
||||||
|
setIsOpen={(isOpen: boolean) =>
|
||||||
|
updateLocalAndScheduleGlobal({ isPassageOpen: isOpen }, false)
|
||||||
|
}
|
||||||
|
contentWrapperClassName={level ? `border border-ielts-reading` : ""}
|
||||||
|
disabled={generatePassageDisabled}
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
className="flex flex-row flex-wrap gap-2 items-center justify-center px-2 pb-4 "
|
||||||
|
>
|
||||||
|
<div className="flex flex-col flex-grow gap-4 px-2">
|
||||||
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
|
Topic (Optional)
|
||||||
|
</label>
|
||||||
|
<Input
|
||||||
|
key={`section-${focusedSection}`}
|
||||||
|
type="text"
|
||||||
|
placeholder="Topic"
|
||||||
|
name="category"
|
||||||
|
onChange={onTopicChange}
|
||||||
|
roundness="full"
|
||||||
|
value={localSettings.readingTopic}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="flex self-end h-16 mb-1">
|
||||||
|
<GenerateBtn
|
||||||
|
module="reading"
|
||||||
|
genType="passage"
|
||||||
|
sectionId={focusedSection}
|
||||||
|
generateFnc={generatePassage}
|
||||||
|
level={level}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</Dropdown>
|
||||||
|
<Dropdown
|
||||||
|
title="Add Exercises"
|
||||||
|
module="reading"
|
||||||
|
open={localSettings.isReadingTopicOpean}
|
||||||
|
setIsOpen={(isOpen: boolean) =>
|
||||||
|
updateLocalAndScheduleGlobal({ isReadingTopicOpean: isOpen })
|
||||||
|
}
|
||||||
|
contentWrapperClassName={level ? `border border-ielts-reading` : ""}
|
||||||
|
disabled={
|
||||||
|
currentSection === undefined ||
|
||||||
|
currentSection.text === undefined ||
|
||||||
|
currentSection.text.content === "" ||
|
||||||
|
currentSection.text.title === ""
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<ExercisePicker
|
||||||
|
module="reading"
|
||||||
|
sectionId={levelId !== undefined ? levelId : focusedSection}
|
||||||
|
extraArgs={{
|
||||||
|
text:
|
||||||
|
currentSection === undefined || currentSection.text === undefined
|
||||||
|
? ""
|
||||||
|
: currentSection.text.content,
|
||||||
|
}}
|
||||||
|
levelSectionId={focusedSection}
|
||||||
|
level={level}
|
||||||
|
/>
|
||||||
|
</Dropdown>
|
||||||
|
</>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default ReadingComponents;
|
export default ReadingComponents;
|
||||||
|
|||||||
@@ -14,130 +14,138 @@ import { toast } from "react-toastify";
|
|||||||
import ReadingComponents from "./components";
|
import ReadingComponents from "./components";
|
||||||
|
|
||||||
const ReadingSettings: React.FC = () => {
|
const ReadingSettings: React.FC = () => {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
const {
|
const {
|
||||||
setExam,
|
setExam,
|
||||||
setExerciseIndex,
|
setExerciseIndex,
|
||||||
setPartIndex,
|
setPartIndex,
|
||||||
setQuestionIndex,
|
setQuestionIndex,
|
||||||
setBgColor,
|
setBgColor,
|
||||||
} = usePersistentExamStore();
|
} = usePersistentExamStore();
|
||||||
|
|
||||||
const { currentModule, title } = useExamEditorStore();
|
const { currentModule, title } = useExamEditorStore();
|
||||||
const {
|
const { focusedSection, difficulty, sections, minTimer, access, type } =
|
||||||
focusedSection,
|
useExamEditorStore((state) => state.modules[currentModule]);
|
||||||
difficulty,
|
|
||||||
sections,
|
|
||||||
minTimer,
|
|
||||||
isPrivate,
|
|
||||||
type,
|
|
||||||
} = useExamEditorStore(state => state.modules[currentModule]);
|
|
||||||
|
|
||||||
const { localSettings, updateLocalAndScheduleGlobal } = useSettingsState<ReadingSectionSettings>(
|
const { localSettings, updateLocalAndScheduleGlobal } =
|
||||||
currentModule,
|
useSettingsState<ReadingSectionSettings>(currentModule, focusedSection);
|
||||||
focusedSection
|
|
||||||
);
|
|
||||||
|
|
||||||
const currentSection = sections.find((section) => section.sectionId == focusedSection)?.state as ReadingPart;
|
const currentSection = sections.find(
|
||||||
|
(section) => section.sectionId == focusedSection
|
||||||
|
)?.state as ReadingPart;
|
||||||
|
|
||||||
|
const defaultPresets: Option[] = [
|
||||||
|
{
|
||||||
|
label: "Preset: Reading Passage 1",
|
||||||
|
value:
|
||||||
|
"Welcome to {part} of the {label}. You will read texts relating to everyday topics and situations. These may include advertisements, brochures, manuals, or official documents. Answer questions that test your ability to locate specific information and understand main ideas.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Preset: Reading Passage 2",
|
||||||
|
value:
|
||||||
|
"Welcome to {part} of the {label}. You will read texts dealing with general interest topics that may include news articles, company policies, or workplace documents. Answer questions testing your understanding of main ideas, specific details, and the author's views.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Preset: Reading Passage 3",
|
||||||
|
value:
|
||||||
|
"Welcome to {part} of the {label}. You will read longer academic texts that may include journal articles, academic essays, or research papers. Answer questions testing your ability to understand complex arguments, identify key points, and follow the development of ideas.",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
const defaultPresets: Option[] = [
|
const canPreviewOrSubmit = sections.some(
|
||||||
{
|
(s) =>
|
||||||
label: "Preset: Reading Passage 1",
|
(s.state as ReadingPart).exercises &&
|
||||||
value: "Welcome to {part} of the {label}. You will read texts relating to everyday topics and situations. These may include advertisements, brochures, manuals, or official documents. Answer questions that test your ability to locate specific information and understand main ideas."
|
(s.state as ReadingPart).exercises.length > 0
|
||||||
},
|
);
|
||||||
{
|
|
||||||
label: "Preset: Reading Passage 2",
|
|
||||||
value: "Welcome to {part} of the {label}. You will read texts dealing with general interest topics that may include news articles, company policies, or workplace documents. Answer questions testing your understanding of main ideas, specific details, and the author's views."
|
|
||||||
},
|
|
||||||
{
|
|
||||||
label: "Preset: Reading Passage 3",
|
|
||||||
value: "Welcome to {part} of the {label}. You will read longer academic texts that may include journal articles, academic essays, or research papers. Answer questions testing your ability to understand complex arguments, identify key points, and follow the development of ideas."
|
|
||||||
}
|
|
||||||
];
|
|
||||||
|
|
||||||
const canPreviewOrSubmit = sections.some(
|
const submitReading = (requiresApproval: boolean) => {
|
||||||
(s) => (s.state as ReadingPart).exercises && (s.state as ReadingPart).exercises.length > 0
|
if (title === "") {
|
||||||
);
|
toast.error("Enter a title for the exam!");
|
||||||
|
return;
|
||||||
const submitReading = () => {
|
}
|
||||||
if (title === "") {
|
const exam: ReadingExam = {
|
||||||
toast.error("Enter a title for the exam!");
|
parts: sections.map((s) => {
|
||||||
return;
|
const exercise = s.state as ReadingPart;
|
||||||
}
|
return {
|
||||||
const exam: ReadingExam = {
|
...exercise,
|
||||||
parts: sections.map((s) => {
|
intro: localSettings.currentIntro,
|
||||||
const exercise = s.state as ReadingPart;
|
category: localSettings.category,
|
||||||
return {
|
|
||||||
...exercise,
|
|
||||||
intro: localSettings.currentIntro,
|
|
||||||
category: localSettings.category
|
|
||||||
};
|
|
||||||
}),
|
|
||||||
isDiagnostic: false,
|
|
||||||
minTimer,
|
|
||||||
module: "reading",
|
|
||||||
id: title,
|
|
||||||
variant: sections.length === 3 ? "full" : "partial",
|
|
||||||
difficulty,
|
|
||||||
private: isPrivate,
|
|
||||||
type: type!
|
|
||||||
};
|
};
|
||||||
|
}),
|
||||||
|
requiresApproval: requiresApproval,
|
||||||
|
isDiagnostic: false,
|
||||||
|
minTimer,
|
||||||
|
module: "reading",
|
||||||
|
id: title,
|
||||||
|
variant: sections.length === 3 ? "full" : "partial",
|
||||||
|
difficulty,
|
||||||
|
access,
|
||||||
|
type: type!,
|
||||||
|
};
|
||||||
|
|
||||||
axios.post(`/api/exam/reading`, exam)
|
axios
|
||||||
.then((result) => {
|
.post(`/api/exam/reading`, exam)
|
||||||
playSound("sent");
|
.then((result) => {
|
||||||
toast.success(`Submitted Exam ID: ${result.data.id}`);
|
playSound("sent");
|
||||||
})
|
// Successfully submitted exam
|
||||||
.catch((error) => {
|
if (result.status === 200) {
|
||||||
console.log(error);
|
toast.success(result.data.message);
|
||||||
toast.error(error.response.data.error || "Something went wrong while submitting, please try again later.");
|
} else if (result.status === 207) {
|
||||||
})
|
toast.warning(result.data.message);
|
||||||
}
|
}
|
||||||
|
})
|
||||||
|
.catch((error) => {
|
||||||
|
console.log(error);
|
||||||
|
toast.error(
|
||||||
|
error.response.data.error ||
|
||||||
|
"Something went wrong while submitting, please try again later."
|
||||||
|
);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
const preview = () => {
|
const preview = () => {
|
||||||
setExam({
|
setExam({
|
||||||
parts: sections.map((s) => {
|
parts: sections.map((s) => {
|
||||||
const exercises = s.state as ReadingPart;
|
const exercises = s.state as ReadingPart;
|
||||||
return {
|
return {
|
||||||
...exercises,
|
...exercises,
|
||||||
intro: s.settings.currentIntro,
|
intro: s.settings.currentIntro,
|
||||||
category: s.settings.category
|
category: s.settings.category,
|
||||||
};
|
};
|
||||||
}),
|
}),
|
||||||
minTimer,
|
minTimer,
|
||||||
module: "reading",
|
module: "reading",
|
||||||
id: title,
|
id: title,
|
||||||
isDiagnostic: false,
|
isDiagnostic: false,
|
||||||
variant: undefined,
|
variant: undefined,
|
||||||
difficulty,
|
difficulty,
|
||||||
private: isPrivate,
|
access: access,
|
||||||
type: type!
|
type: type!,
|
||||||
} as ReadingExam);
|
} as ReadingExam);
|
||||||
setExerciseIndex(0);
|
setExerciseIndex(0);
|
||||||
setQuestionIndex(0);
|
setQuestionIndex(0);
|
||||||
setPartIndex(0);
|
setPartIndex(0);
|
||||||
setBgColor("bg-white");
|
setBgColor("bg-white");
|
||||||
openDetachedTab("popout?type=Exam&module=reading", router)
|
openDetachedTab("popout?type=Exam&module=reading", router);
|
||||||
}
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<SettingsEditor
|
<SettingsEditor
|
||||||
sectionLabel={`Passage ${focusedSection}`}
|
sectionLabel={`Passage ${focusedSection}`}
|
||||||
sectionId={focusedSection}
|
sectionId={focusedSection}
|
||||||
module="reading"
|
module="reading"
|
||||||
introPresets={[defaultPresets[focusedSection - 1]]}
|
introPresets={[defaultPresets[focusedSection - 1]]}
|
||||||
preview={preview}
|
preview={preview}
|
||||||
canPreview={canPreviewOrSubmit}
|
canPreview={canPreviewOrSubmit}
|
||||||
canSubmit={canPreviewOrSubmit}
|
canSubmit={canPreviewOrSubmit}
|
||||||
submitModule={submitReading}
|
submitModule={submitReading}
|
||||||
>
|
>
|
||||||
<ReadingComponents
|
<ReadingComponents
|
||||||
{...{ localSettings, updateLocalAndScheduleGlobal, currentSection }}
|
{...{ localSettings, updateLocalAndScheduleGlobal, currentSection }}
|
||||||
/>
|
/>
|
||||||
</SettingsEditor>
|
</SettingsEditor>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default ReadingSettings;
|
export default ReadingSettings;
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import openDetachedTab from "@/utils/popout";
|
|||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import { playSound } from "@/utils/sound";
|
import { playSound } from "@/utils/sound";
|
||||||
import SpeakingComponents from "./components";
|
import SpeakingComponents from "./components";
|
||||||
|
import { getExamById } from "@/utils/exams";
|
||||||
|
|
||||||
export interface Avatar {
|
export interface Avatar {
|
||||||
name: string;
|
name: string;
|
||||||
@@ -29,7 +30,7 @@ const SpeakingSettings: React.FC = () => {
|
|||||||
} = usePersistentExamStore();
|
} = usePersistentExamStore();
|
||||||
|
|
||||||
const { title, currentModule } = useExamEditorStore();
|
const { title, currentModule } = useExamEditorStore();
|
||||||
const { focusedSection, difficulty, sections, minTimer, isPrivate } = useExamEditorStore((store) => store.modules[currentModule])
|
const { focusedSection, difficulty, sections, minTimer, access } = useExamEditorStore((store) => store.modules[currentModule])
|
||||||
|
|
||||||
const section = sections.find((section) => section.sectionId == focusedSection)?.state;
|
const section = sections.find((section) => section.sectionId == focusedSection)?.state;
|
||||||
|
|
||||||
@@ -83,7 +84,7 @@ const SpeakingSettings: React.FC = () => {
|
|||||||
});
|
});
|
||||||
})();
|
})();
|
||||||
|
|
||||||
const submitSpeaking = async () => {
|
const submitSpeaking = async (requiresApproval: boolean) => {
|
||||||
if (title === "") {
|
if (title === "") {
|
||||||
toast.error("Enter a title for the exam!");
|
toast.error("Enter a title for the exam!");
|
||||||
return;
|
return;
|
||||||
@@ -180,16 +181,22 @@ const SpeakingSettings: React.FC = () => {
|
|||||||
minTimer,
|
minTimer,
|
||||||
module: "speaking",
|
module: "speaking",
|
||||||
id: title,
|
id: title,
|
||||||
|
requiresApproval: requiresApproval,
|
||||||
isDiagnostic: false,
|
isDiagnostic: false,
|
||||||
variant: undefined,
|
variant: undefined,
|
||||||
difficulty,
|
difficulty,
|
||||||
instructorGender: "varied",
|
instructorGender: "varied",
|
||||||
private: isPrivate,
|
access,
|
||||||
};
|
};
|
||||||
|
|
||||||
const result = await axios.post('/api/exam/speaking', exam);
|
const result = await axios.post('/api/exam/speaking', exam);
|
||||||
playSound("sent");
|
playSound("sent");
|
||||||
toast.success(`Submitted Exam ID: ${result.data.id}`);
|
// Successfully submitted exam
|
||||||
|
if (result.status === 200) {
|
||||||
|
toast.success(result.data.message);
|
||||||
|
} else if (result.status === 207) {
|
||||||
|
toast.warning(result.data.message);
|
||||||
|
}
|
||||||
|
|
||||||
Array.from(urlMap.values()).forEach(url => {
|
Array.from(urlMap.values()).forEach(url => {
|
||||||
URL.revokeObjectURL(url);
|
URL.revokeObjectURL(url);
|
||||||
@@ -232,7 +239,7 @@ const SpeakingSettings: React.FC = () => {
|
|||||||
isDiagnostic: false,
|
isDiagnostic: false,
|
||||||
variant: undefined,
|
variant: undefined,
|
||||||
difficulty,
|
difficulty,
|
||||||
private: isPrivate,
|
access,
|
||||||
} as SpeakingExam);
|
} as SpeakingExam);
|
||||||
setExerciseIndex(0);
|
setExerciseIndex(0);
|
||||||
setQuestionIndex(0);
|
setQuestionIndex(0);
|
||||||
|
|||||||
@@ -12,6 +12,8 @@ import axios from "axios";
|
|||||||
import { playSound } from "@/utils/sound";
|
import { playSound } from "@/utils/sound";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
import WritingComponents from "./components";
|
import WritingComponents from "./components";
|
||||||
|
import { getExamById } from "@/utils/exams";
|
||||||
|
import { ApprovalWorkflow } from "@/interfaces/approval.workflow";
|
||||||
|
|
||||||
const WritingSettings: React.FC = () => {
|
const WritingSettings: React.FC = () => {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
@@ -21,7 +23,7 @@ const WritingSettings: React.FC = () => {
|
|||||||
const {
|
const {
|
||||||
minTimer,
|
minTimer,
|
||||||
difficulty,
|
difficulty,
|
||||||
isPrivate,
|
access,
|
||||||
sections,
|
sections,
|
||||||
focusedSection,
|
focusedSection,
|
||||||
type,
|
type,
|
||||||
@@ -79,14 +81,14 @@ const WritingSettings: React.FC = () => {
|
|||||||
isDiagnostic: false,
|
isDiagnostic: false,
|
||||||
variant: undefined,
|
variant: undefined,
|
||||||
difficulty,
|
difficulty,
|
||||||
private: isPrivate,
|
access,
|
||||||
type: type!
|
type: type!
|
||||||
});
|
});
|
||||||
setExerciseIndex(0);
|
setExerciseIndex(0);
|
||||||
openDetachedTab("popout?type=Exam&module=writing", router)
|
openDetachedTab("popout?type=Exam&module=writing", router)
|
||||||
}
|
}
|
||||||
|
|
||||||
const submitWriting = async () => {
|
const submitWriting = async (requiresApproval: boolean) => {
|
||||||
if (title === "") {
|
if (title === "") {
|
||||||
toast.error("Enter a title for the exam!");
|
toast.error("Enter a title for the exam!");
|
||||||
return;
|
return;
|
||||||
@@ -129,16 +131,22 @@ const WritingSettings: React.FC = () => {
|
|||||||
minTimer,
|
minTimer,
|
||||||
module: "writing",
|
module: "writing",
|
||||||
id: title,
|
id: title,
|
||||||
|
requiresApproval: requiresApproval,
|
||||||
isDiagnostic: false,
|
isDiagnostic: false,
|
||||||
variant: undefined,
|
variant: undefined,
|
||||||
difficulty,
|
difficulty,
|
||||||
private: isPrivate,
|
access,
|
||||||
type: type!
|
type: type!
|
||||||
};
|
};
|
||||||
|
|
||||||
const result = await axios.post(`/api/exam/writing`, exam)
|
const result = await axios.post(`/api/exam/writing`, exam)
|
||||||
playSound("sent");
|
playSound("sent");
|
||||||
toast.success(`Submitted Exam ID: ${result.data.id}`);
|
// Successfully submitted exam
|
||||||
|
if (result.status === 200) {
|
||||||
|
toast.success(result.data.message);
|
||||||
|
} else if (result.status === 207) {
|
||||||
|
toast.warning(result.data.message);
|
||||||
|
}
|
||||||
|
|
||||||
} catch (error: any) {
|
} catch (error: any) {
|
||||||
console.error('Error submitting exam:', error);
|
console.error('Error submitting exam:', error);
|
||||||
|
|||||||
@@ -19,8 +19,8 @@ const label = (type: string, firstId: string, lastId: string) => {
|
|||||||
const ExerciseLabel: React.FC<Props> = ({type, firstId, lastId, prompt}) => {
|
const ExerciseLabel: React.FC<Props> = ({type, firstId, lastId, prompt}) => {
|
||||||
return (
|
return (
|
||||||
<div className="flex w-full justify-between items-center mr-4">
|
<div className="flex w-full justify-between items-center mr-4">
|
||||||
<span className="font-semibold">{label(type, firstId, lastId)}</span>
|
<span className="font-semibold ellipsis-2">{label(type, firstId, lastId)}</span>
|
||||||
<div className="text-sm font-light italic">{previewLabel(prompt)}</div>
|
<div className="text-sm font-light italic ellipsis-2">{previewLabel(prompt)}</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,9 @@
|
|||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import SectionRenderer from "./SectionRenderer";
|
import SectionRenderer from "./SectionRenderer";
|
||||||
import Checkbox from "../Low/Checkbox";
|
|
||||||
import Input from "../Low/Input";
|
import Input from "../Low/Input";
|
||||||
import Select from "../Low/Select";
|
import Select from "../Low/Select";
|
||||||
import { capitalize } from "lodash";
|
import { capitalize } from "lodash";
|
||||||
import { Difficulty } from "@/interfaces/exam";
|
import { AccessType, ACCESSTYPE, Difficulty } from "@/interfaces/exam";
|
||||||
import { useCallback, useEffect, useMemo, useState } from "react";
|
import { useCallback, useEffect, useMemo, useState } from "react";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
import { ModuleState, SectionState } from "@/stores/examEditor/types";
|
import { ModuleState, SectionState } from "@/stores/examEditor/types";
|
||||||
@@ -20,220 +19,325 @@ import { defaultSectionSettings } from "@/stores/examEditor/defaults";
|
|||||||
import Button from "../Low/Button";
|
import Button from "../Low/Button";
|
||||||
import ResetModule from "./Standalone/ResetModule";
|
import ResetModule from "./Standalone/ResetModule";
|
||||||
import ListeningInstructions from "./Standalone/ListeningInstructions";
|
import ListeningInstructions from "./Standalone/ListeningInstructions";
|
||||||
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
|
import Option from "../../interfaces/option";
|
||||||
|
|
||||||
const DIFFICULTIES: Difficulty[] = ["A1", "A2", "B1", "B2", "C1", "C2"];
|
const DIFFICULTIES: Option[] = [
|
||||||
|
{ value: "A1", label: "A1" },
|
||||||
|
{ value: "A2", label: "A2" },
|
||||||
|
{ value: "B1", label: "B1" },
|
||||||
|
{ value: "B2", label: "B2" },
|
||||||
|
{ value: "C1", label: "C1" },
|
||||||
|
{ value: "C2", label: "C2" },
|
||||||
|
];
|
||||||
|
|
||||||
const ExamEditor: React.FC<{ levelParts?: number }> = ({ levelParts = 0 }) => {
|
const ModuleSettings: Record<Module, React.ComponentType> = {
|
||||||
const { currentModule, dispatch } = useExamEditorStore();
|
reading: ReadingSettings,
|
||||||
const {
|
writing: WritingSettings,
|
||||||
sections,
|
speaking: SpeakingSettings,
|
||||||
minTimer,
|
listening: ListeningSettings,
|
||||||
expandedSections,
|
level: LevelSettings,
|
||||||
examLabel,
|
};
|
||||||
isPrivate,
|
|
||||||
difficulty,
|
|
||||||
sectionLabels,
|
|
||||||
importModule
|
|
||||||
} = useExamEditorStore(state => state.modules[currentModule]);
|
|
||||||
|
|
||||||
const [numberOfLevelParts, setNumberOfLevelParts] = useState(levelParts !== 0 ? levelParts : 1);
|
const ExamEditor: React.FC<{
|
||||||
const [isResetModuleOpen, setIsResetModuleOpen] = useState(false);
|
levelParts?: number;
|
||||||
|
entitiesAllowEditPrivacy: EntityWithRoles[];
|
||||||
|
entitiesAllowConfExams: EntityWithRoles[];
|
||||||
|
entitiesAllowPublicExams: EntityWithRoles[];
|
||||||
|
}> = ({
|
||||||
|
levelParts = 0,
|
||||||
|
entitiesAllowEditPrivacy = [],
|
||||||
|
entitiesAllowConfExams = [],
|
||||||
|
entitiesAllowPublicExams = [],
|
||||||
|
}) => {
|
||||||
|
const { currentModule, dispatch } = useExamEditorStore();
|
||||||
|
const {
|
||||||
|
sections,
|
||||||
|
minTimer,
|
||||||
|
expandedSections,
|
||||||
|
examLabel,
|
||||||
|
access,
|
||||||
|
difficulty,
|
||||||
|
sectionLabels,
|
||||||
|
importModule,
|
||||||
|
} = useExamEditorStore((state) => state.modules[currentModule]);
|
||||||
|
|
||||||
// For exam edits
|
const [numberOfLevelParts, setNumberOfLevelParts] = useState(
|
||||||
useEffect(() => {
|
levelParts !== 0 ? levelParts : 1
|
||||||
if (levelParts !== 0) {
|
);
|
||||||
setNumberOfLevelParts(levelParts);
|
const [isResetModuleOpen, setIsResetModuleOpen] = useState(false);
|
||||||
dispatch({
|
|
||||||
type: 'UPDATE_MODULE',
|
|
||||||
payload: {
|
|
||||||
updates: {
|
|
||||||
sectionLabels: Array.from({ length: levelParts }).map((_, i) => ({
|
|
||||||
id: i + 1,
|
|
||||||
label: `Part ${i + 1}`
|
|
||||||
}))
|
|
||||||
},
|
|
||||||
module: "level"
|
|
||||||
}
|
|
||||||
})
|
|
||||||
}
|
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
||||||
}, [levelParts])
|
|
||||||
|
|
||||||
useEffect(() => {
|
// For exam edits
|
||||||
const currentSections = sections;
|
useEffect(() => {
|
||||||
const currentLabels = sectionLabels;
|
if (levelParts !== 0) {
|
||||||
let updatedSections: SectionState[];
|
setNumberOfLevelParts(levelParts);
|
||||||
let updatedLabels: any;
|
dispatch({
|
||||||
if (currentModule === "level" && currentSections.length !== currentLabels.length || numberOfLevelParts !== currentSections.length) {
|
type: "UPDATE_MODULE",
|
||||||
const newSections = [...currentSections];
|
payload: {
|
||||||
const newLabels = [...currentLabels];
|
updates: {
|
||||||
for (let i = currentLabels.length; i < numberOfLevelParts; i++) {
|
sectionLabels: Array.from({ length: levelParts }).map((_, i) => ({
|
||||||
if (currentSections.length !== numberOfLevelParts) newSections.push(defaultSectionSettings(currentModule, i + 1));
|
id: i + 1,
|
||||||
newLabels.push({
|
label: `Part ${i + 1}`,
|
||||||
id: i + 1,
|
})),
|
||||||
label: `Part ${i + 1}`
|
},
|
||||||
});
|
module: "level",
|
||||||
}
|
},
|
||||||
updatedSections = newSections;
|
});
|
||||||
updatedLabels = newLabels;
|
}
|
||||||
} else if (numberOfLevelParts < currentSections.length) {
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
updatedSections = currentSections.slice(0, numberOfLevelParts);
|
}, [levelParts]);
|
||||||
updatedLabels = currentLabels.slice(0, numberOfLevelParts);
|
|
||||||
} else {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const updatedExpandedSections = expandedSections.filter(
|
useEffect(() => {
|
||||||
sectionId => updatedSections.some(section => section.sectionId === sectionId)
|
const currentSections = sections;
|
||||||
);
|
const currentLabels = sectionLabels;
|
||||||
|
let updatedSections: SectionState[];
|
||||||
dispatch({
|
let updatedLabels: any;
|
||||||
type: 'UPDATE_MODULE',
|
if (
|
||||||
payload: {
|
(currentModule === "level" &&
|
||||||
updates: {
|
currentSections.length !== currentLabels.length) ||
|
||||||
sections: updatedSections,
|
numberOfLevelParts !== currentSections.length
|
||||||
sectionLabels: updatedLabels,
|
) {
|
||||||
expandedSections: updatedExpandedSections
|
const newSections = [...currentSections];
|
||||||
}
|
const newLabels = [...currentLabels];
|
||||||
}
|
for (let i = currentLabels.length; i < numberOfLevelParts; i++) {
|
||||||
|
if (currentSections.length !== numberOfLevelParts)
|
||||||
|
newSections.push(defaultSectionSettings(currentModule, i + 1));
|
||||||
|
newLabels.push({
|
||||||
|
id: i + 1,
|
||||||
|
label: `Part ${i + 1}`,
|
||||||
});
|
});
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
}
|
||||||
}, [numberOfLevelParts]);
|
updatedSections = newSections;
|
||||||
|
updatedLabels = newLabels;
|
||||||
const sectionIds = sections.map((section) => section.sectionId)
|
} else if (numberOfLevelParts < currentSections.length) {
|
||||||
|
updatedSections = currentSections.slice(0, numberOfLevelParts);
|
||||||
const updateModule = useCallback((updates: Partial<ModuleState>) => {
|
updatedLabels = currentLabels.slice(0, numberOfLevelParts);
|
||||||
dispatch({ type: 'UPDATE_MODULE', payload: { updates } });
|
} else {
|
||||||
}, [dispatch]);
|
return;
|
||||||
|
|
||||||
const toggleSection = (sectionId: number) => {
|
|
||||||
if (expandedSections.length === 1 && sectionIds.includes(sectionId)) {
|
|
||||||
toast.error("Include at least one section!");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
dispatch({ type: 'TOGGLE_SECTION', payload: { sectionId } });
|
|
||||||
};
|
|
||||||
|
|
||||||
const ModuleSettings: Record<Module, React.ComponentType> = {
|
|
||||||
reading: ReadingSettings,
|
|
||||||
writing: WritingSettings,
|
|
||||||
speaking: SpeakingSettings,
|
|
||||||
listening: ListeningSettings,
|
|
||||||
level: LevelSettings
|
|
||||||
};
|
|
||||||
|
|
||||||
const Settings = ModuleSettings[currentModule];
|
|
||||||
const showImport = importModule && ["reading", "listening", "level"].includes(currentModule);
|
|
||||||
|
|
||||||
const updateLevelParts = (parts: number) => {
|
|
||||||
setNumberOfLevelParts(parts);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
const updatedExpandedSections = expandedSections.filter((sectionId) =>
|
||||||
<>
|
updatedSections.some((section) => section.sectionId === sectionId)
|
||||||
{showImport ? <ImportOrStartFromScratch module={currentModule} setNumberOfLevelParts={updateLevelParts} /> : (
|
|
||||||
<>
|
|
||||||
{isResetModuleOpen && <ResetModule module={currentModule} isOpen={isResetModuleOpen} setIsOpen={setIsResetModuleOpen} setNumberOfLevelParts={setNumberOfLevelParts}/>}
|
|
||||||
<div className="flex gap-4 w-full items-center">
|
|
||||||
<div className="flex flex-col gap-3">
|
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Timer</label>
|
|
||||||
<Input
|
|
||||||
type="number"
|
|
||||||
name="minTimer"
|
|
||||||
onChange={(e) => updateModule({ minTimer: parseInt(e) < 15 ? 15 : parseInt(e) })}
|
|
||||||
value={minTimer}
|
|
||||||
className="max-w-[300px]"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
<div className="flex flex-col gap-3 flex-grow">
|
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Difficulty</label>
|
|
||||||
<Select
|
|
||||||
isMulti={true}
|
|
||||||
options={DIFFICULTIES.map((x) => ({
|
|
||||||
value: x,
|
|
||||||
label: capitalize(x)
|
|
||||||
}))}
|
|
||||||
onChange={(values) => {
|
|
||||||
const selectedDifficulties = values ? values.map(v => v.value as Difficulty) : [];
|
|
||||||
updateModule({ difficulty: selectedDifficulties });
|
|
||||||
}}
|
|
||||||
value={
|
|
||||||
difficulty
|
|
||||||
? difficulty.map(d => ({
|
|
||||||
value: d,
|
|
||||||
label: capitalize(d)
|
|
||||||
}))
|
|
||||||
: null
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
{(sectionLabels.length != 0 && currentModule !== "level") ? (
|
|
||||||
<div className="flex flex-col gap-3">
|
|
||||||
<label className="font-normal text-base text-mti-gray-dim">{sectionLabels[0].label.split(" ")[0]}</label>
|
|
||||||
<div className="flex flex-row gap-8">
|
|
||||||
{sectionLabels.map(({ id, label }) => (
|
|
||||||
<span
|
|
||||||
key={id}
|
|
||||||
className={clsx(
|
|
||||||
"px-6 py-4 w-48 h-[72px] flex justify-center items-center text-sm font-normal rounded-full border focus:outline-none cursor-pointer",
|
|
||||||
"transition duration-300 ease-in-out",
|
|
||||||
sectionIds.includes(id)
|
|
||||||
? `bg-ielts-${currentModule}/70 border-ielts-${currentModule} text-white`
|
|
||||||
: "bg-white border-mti-gray-platinum"
|
|
||||||
)}
|
|
||||||
onClick={() => toggleSection(id)}
|
|
||||||
>
|
|
||||||
{label}
|
|
||||||
</span>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
) : (
|
|
||||||
<div className="flex flex-col gap-3 w-1/3">
|
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Number of Parts</label>
|
|
||||||
<Input type="number" name="Number of Parts" min={1} onChange={(v) => setNumberOfLevelParts(parseInt(v))} value={numberOfLevelParts} />
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
<div className="flex flex-col gap-3 w-fit h-fit">
|
|
||||||
<div className="h-6" />
|
|
||||||
<Checkbox isChecked={isPrivate} onChange={(checked) => updateModule({ isPrivate: checked })}>
|
|
||||||
Privacy (Only available for Assignments)
|
|
||||||
</Checkbox>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div className="flex flex-row gap-3 w-full">
|
|
||||||
<div className="flex flex-col gap-3 flex-grow">
|
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Exam Label *</label>
|
|
||||||
<Input
|
|
||||||
type="text"
|
|
||||||
placeholder="Exam Label"
|
|
||||||
name="label"
|
|
||||||
onChange={(text) => updateModule({ examLabel: text })}
|
|
||||||
roundness="xl"
|
|
||||||
value={examLabel}
|
|
||||||
required
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
{currentModule === "listening" && <ListeningInstructions />}
|
|
||||||
<Button
|
|
||||||
onClick={() => setIsResetModuleOpen(true)}
|
|
||||||
customColor={`bg-ielts-${currentModule}/70 hover:bg-ielts-${currentModule} border-ielts-${currentModule}`}
|
|
||||||
className={`text-white self-end`}
|
|
||||||
>
|
|
||||||
Reset Module
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
<div className="flex flex-row gap-8">
|
|
||||||
<Settings />
|
|
||||||
<div className="flex-grow max-w-[66%]">
|
|
||||||
<SectionRenderer />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</>
|
|
||||||
);
|
);
|
||||||
|
|
||||||
|
dispatch({
|
||||||
|
type: "UPDATE_MODULE",
|
||||||
|
payload: {
|
||||||
|
updates: {
|
||||||
|
sections: updatedSections,
|
||||||
|
sectionLabels: updatedLabels,
|
||||||
|
expandedSections: updatedExpandedSections,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, [numberOfLevelParts]);
|
||||||
|
|
||||||
|
const sectionIds = useMemo(
|
||||||
|
() => sections.map((section) => section.sectionId),
|
||||||
|
[sections]
|
||||||
|
);
|
||||||
|
|
||||||
|
const updateModule = useCallback(
|
||||||
|
(updates: Partial<ModuleState>) => {
|
||||||
|
dispatch({ type: "UPDATE_MODULE", payload: { updates } });
|
||||||
|
},
|
||||||
|
[dispatch]
|
||||||
|
);
|
||||||
|
|
||||||
|
const toggleSection = useCallback(
|
||||||
|
(sectionId: number) => {
|
||||||
|
if (expandedSections.length === 1 && sectionIds.includes(sectionId)) {
|
||||||
|
toast.error("Include at least one section!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
dispatch({ type: "TOGGLE_SECTION", payload: { sectionId } });
|
||||||
|
},
|
||||||
|
[dispatch, expandedSections, sectionIds]
|
||||||
|
);
|
||||||
|
|
||||||
|
const Settings = useMemo(
|
||||||
|
() => ModuleSettings[currentModule],
|
||||||
|
[currentModule]
|
||||||
|
);
|
||||||
|
|
||||||
|
const showImport = useMemo(
|
||||||
|
() =>
|
||||||
|
importModule && ["reading", "listening", "level"].includes(currentModule),
|
||||||
|
[importModule, currentModule]
|
||||||
|
);
|
||||||
|
|
||||||
|
const accessTypeOptions = useMemo(() => {
|
||||||
|
let options: Option[] = [{ value: "private", label: "Private" }];
|
||||||
|
if (entitiesAllowConfExams.length > 0) {
|
||||||
|
options.push({ value: "confidential", label: "Confidential" });
|
||||||
|
}
|
||||||
|
if (entitiesAllowPublicExams.length > 0) {
|
||||||
|
options.push({ value: "public", label: "Public" });
|
||||||
|
}
|
||||||
|
return options;
|
||||||
|
}, [entitiesAllowConfExams.length, entitiesAllowPublicExams.length]);
|
||||||
|
|
||||||
|
const updateLevelParts = useCallback((parts: number) => {
|
||||||
|
setNumberOfLevelParts(parts);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
{showImport ? (
|
||||||
|
<ImportOrStartFromScratch
|
||||||
|
module={currentModule}
|
||||||
|
setNumberOfLevelParts={updateLevelParts}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
{isResetModuleOpen && (
|
||||||
|
<ResetModule
|
||||||
|
module={currentModule}
|
||||||
|
isOpen={isResetModuleOpen}
|
||||||
|
setIsOpen={setIsResetModuleOpen}
|
||||||
|
setNumberOfLevelParts={setNumberOfLevelParts}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
"flex gap-4 w-full",
|
||||||
|
sectionLabels.length > 3 ? "-2xl:flex-col" : "-xl:flex-col"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<div className="flex flex-row gap-3">
|
||||||
|
<div className="flex flex-col gap-3 ">
|
||||||
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
|
Timer
|
||||||
|
</label>
|
||||||
|
<Input
|
||||||
|
type="number"
|
||||||
|
name="minTimer"
|
||||||
|
onChange={(e) =>
|
||||||
|
updateModule({
|
||||||
|
minTimer: parseInt(e) < 15 ? 15 : parseInt(e),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
value={minTimer}
|
||||||
|
className="max-w-[125px] min-w-[100px] w-min"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col gap-3 ">
|
||||||
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
|
Difficulty
|
||||||
|
</label>
|
||||||
|
<Select
|
||||||
|
isMulti={true}
|
||||||
|
options={DIFFICULTIES}
|
||||||
|
onChange={(values) => {
|
||||||
|
const selectedDifficulties = values
|
||||||
|
? values.map((v) => v.value as Difficulty)
|
||||||
|
: [];
|
||||||
|
updateModule({ difficulty: selectedDifficulties });
|
||||||
|
}}
|
||||||
|
value={
|
||||||
|
difficulty
|
||||||
|
? (Array.isArray(difficulty)
|
||||||
|
? difficulty
|
||||||
|
: [difficulty]
|
||||||
|
).map((d) => ({
|
||||||
|
value: d,
|
||||||
|
label: capitalize(d),
|
||||||
|
}))
|
||||||
|
: null
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{sectionLabels.length != 0 && currentModule !== "level" ? (
|
||||||
|
<div className="flex flex-col gap-3 -xl:w-full">
|
||||||
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
|
{sectionLabels[0].label.split(" ")[0]}
|
||||||
|
</label>
|
||||||
|
<div className="flex flex-row gap-3">
|
||||||
|
{sectionLabels.map(({ id, label }) => (
|
||||||
|
<span
|
||||||
|
key={id}
|
||||||
|
className={clsx(
|
||||||
|
"px-6 py-4 w-40 2xl:w-48 h-[72px] flex justify-center items-center text-sm font-normal rounded-full border focus:outline-none cursor-pointer",
|
||||||
|
"transition duration-300 ease-in-out",
|
||||||
|
sectionIds.includes(id)
|
||||||
|
? `bg-ielts-${currentModule}/70 border-ielts-${currentModule} text-white`
|
||||||
|
: "bg-white border-mti-gray-platinum"
|
||||||
|
)}
|
||||||
|
onClick={() => toggleSection(id)}
|
||||||
|
>
|
||||||
|
{label}
|
||||||
|
</span>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="flex flex-col gap-3 w-1/3">
|
||||||
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
|
Number of Parts
|
||||||
|
</label>
|
||||||
|
<Input
|
||||||
|
type="number"
|
||||||
|
name="Number of Parts"
|
||||||
|
min={1}
|
||||||
|
onChange={(v) => setNumberOfLevelParts(parseInt(v))}
|
||||||
|
value={numberOfLevelParts}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<div className="max-w-[200px] w-full">
|
||||||
|
<Select
|
||||||
|
label="Access Type"
|
||||||
|
disabled={
|
||||||
|
accessTypeOptions.length === 0 ||
|
||||||
|
entitiesAllowEditPrivacy.length === 0
|
||||||
|
}
|
||||||
|
options={accessTypeOptions}
|
||||||
|
onChange={(value) => {
|
||||||
|
if (value?.value) {
|
||||||
|
updateModule({ access: value.value! as AccessType });
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
value={{ value: access, label: capitalize(access) }}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex flex-row gap-3 w-full">
|
||||||
|
<div className="flex flex-col gap-3 flex-grow">
|
||||||
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
|
Exam Label *
|
||||||
|
</label>
|
||||||
|
<Input
|
||||||
|
type="text"
|
||||||
|
placeholder="Exam Label"
|
||||||
|
name="label"
|
||||||
|
onChange={(text) => updateModule({ examLabel: text })}
|
||||||
|
roundness="xl"
|
||||||
|
value={examLabel}
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
{currentModule === "listening" && <ListeningInstructions />}
|
||||||
|
<Button
|
||||||
|
onClick={() => setIsResetModuleOpen(true)}
|
||||||
|
customColor={`bg-ielts-${currentModule}/70 hover:bg-ielts-${currentModule} border-ielts-${currentModule}`}
|
||||||
|
className={`text-white self-end`}
|
||||||
|
>
|
||||||
|
Reset Module
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-row gap-8 -xl:flex-col">
|
||||||
|
<Settings />
|
||||||
|
<div className="flex-grow max-w-[66%] -2xl:max-w-full">
|
||||||
|
<SectionRenderer />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default ExamEditor;
|
export default ExamEditor;
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ const DroppableQuestionArea: React.FC<DroppableQuestionAreaProps> = ({ question,
|
|||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
key={`answer_${question.id}_${answer}`}
|
key={`answer_${question.id}_${answer}`}
|
||||||
className={clsx("w-48 h-10 border rounded-xl flex items-center justify-center", isOver && "border-mti-purple-light")}>
|
className={clsx("w-48 h-10 border-2 border-mti-purple-light self-center rounded-xl flex items-center justify-center", isOver && "border-mti-purple-dark")}>
|
||||||
{answer && `Paragraph ${answer}`}
|
{answer && `Paragraph ${answer}`}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,7 +1,10 @@
|
|||||||
import { Session } from "@/hooks/useSessions";
|
import { Session } from "@/hooks/useSessions";
|
||||||
import { Assignment } from "@/interfaces/results";
|
import { Assignment } from "@/interfaces/results";
|
||||||
import { User } from "@/interfaces/user";
|
import { User } from "@/interfaces/user";
|
||||||
import { activeAssignmentFilter, futureAssignmentFilter } from "@/utils/assignments";
|
import {
|
||||||
|
activeAssignmentFilter,
|
||||||
|
futureAssignmentFilter,
|
||||||
|
} from "@/utils/assignments";
|
||||||
import { sortByModuleName } from "@/utils/moduleUtils";
|
import { sortByModuleName } from "@/utils/moduleUtils";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
@@ -11,102 +14,124 @@ import Button from "../Low/Button";
|
|||||||
import ModuleBadge from "../ModuleBadge";
|
import ModuleBadge from "../ModuleBadge";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
assignment: Assignment
|
assignment: Assignment;
|
||||||
user: User
|
user: User;
|
||||||
session?: Session
|
session?: Session;
|
||||||
startAssignment: (assignment: Assignment) => void
|
startAssignment: (assignment: Assignment) => void;
|
||||||
resumeAssignment: (session: Session) => void
|
resumeAssignment: (session: Session) => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function AssignmentCard({ user, assignment, session, startAssignment, resumeAssignment }: Props) {
|
export default function AssignmentCard({
|
||||||
const router = useRouter()
|
user,
|
||||||
|
assignment,
|
||||||
|
session,
|
||||||
|
startAssignment,
|
||||||
|
resumeAssignment,
|
||||||
|
}: Props) {
|
||||||
|
const hasBeenSubmitted = useMemo(
|
||||||
|
() => assignment.results.map((r) => r.user).includes(user.id),
|
||||||
|
[assignment.results, user.id]
|
||||||
|
);
|
||||||
|
|
||||||
const hasBeenSubmitted = useMemo(() => assignment.results.map((r) => r.user).includes(user.id), [assignment.results, user.id])
|
return (
|
||||||
|
<div
|
||||||
return (
|
className={clsx(
|
||||||
<div
|
"border-mti-gray-anti-flash flex min-w-[350px] flex-col gap-6 rounded-xl border p-4",
|
||||||
className={clsx(
|
assignment.results.map((r) => r.user).includes(user.id) &&
|
||||||
"border-mti-gray-anti-flash flex min-w-[350px] flex-col gap-6 rounded-xl border p-4",
|
"border-mti-green-light"
|
||||||
assignment.results.map((r) => r.user).includes(user.id) && "border-mti-green-light",
|
)}
|
||||||
)}
|
key={assignment.id}
|
||||||
key={assignment.id}>
|
>
|
||||||
<div className="flex flex-col gap-1">
|
<div className="flex flex-col gap-1">
|
||||||
<h3 className="text-mti-black/90 text-xl font-semibold">{assignment.name}</h3>
|
<h3 className="text-mti-black/90 text-xl font-semibold">
|
||||||
<span className="flex justify-between gap-1 text-lg">
|
{assignment.name}
|
||||||
<span>{moment(assignment.startDate).format("DD/MM/YY, HH:mm")}</span>
|
</h3>
|
||||||
<span>-</span>
|
<span className="flex justify-between gap-1 text-lg">
|
||||||
<span>{moment(assignment.endDate).format("DD/MM/YY, HH:mm")}</span>
|
<span>{moment(assignment.startDate).format("DD/MM/YY, HH:mm")}</span>
|
||||||
</span>
|
<span>-</span>
|
||||||
</div>
|
<span>{moment(assignment.endDate).format("DD/MM/YY, HH:mm")}</span>
|
||||||
<div className="flex w-full items-center justify-between">
|
</span>
|
||||||
<div className="-md:mt-2 grid w-fit min-w-[140px] grid-cols-2 grid-rows-2 place-items-center justify-between gap-4">
|
</div>
|
||||||
{assignment.exams
|
<div className="flex w-full items-center justify-between">
|
||||||
.filter((e) => e.assignee === user.id)
|
<div className="-md:mt-2 grid w-fit min-w-[140px] grid-cols-2 grid-rows-2 place-items-center justify-between gap-4">
|
||||||
.map((e) => e.module)
|
{assignment.exams
|
||||||
.sort(sortByModuleName)
|
.filter((e) => e.assignee === user.id)
|
||||||
.map((module) => (
|
.map((e) => e.module)
|
||||||
<ModuleBadge className="scale-110 w-full" key={module} module={module} />
|
.sort(sortByModuleName)
|
||||||
))}
|
.map((module) => (
|
||||||
</div>
|
<ModuleBadge
|
||||||
{futureAssignmentFilter(assignment) && !hasBeenSubmitted && (
|
className="scale-110 w-full"
|
||||||
<Button
|
key={module}
|
||||||
color="rose"
|
module={module}
|
||||||
className="-md:hidden h-full w-full max-w-[50%] !rounded-xl"
|
/>
|
||||||
disabled
|
))}
|
||||||
variant="outline">
|
</div>
|
||||||
Not yet started
|
{futureAssignmentFilter(assignment) && !hasBeenSubmitted && (
|
||||||
</Button>
|
<Button
|
||||||
)}
|
color="rose"
|
||||||
{activeAssignmentFilter(assignment) && !hasBeenSubmitted && (
|
className="-md:hidden h-full w-full max-w-[50%] !rounded-xl"
|
||||||
<>
|
disabled
|
||||||
<div
|
variant="outline"
|
||||||
className="tooltip flex h-full w-full items-center justify-end pl-8 md:hidden"
|
>
|
||||||
data-tip="Your screen size is too small to perform an assignment">
|
Not yet started
|
||||||
<Button className="h-full w-full !rounded-xl" variant="outline">
|
</Button>
|
||||||
Start
|
)}
|
||||||
</Button>
|
{activeAssignmentFilter(assignment) && !hasBeenSubmitted && (
|
||||||
</div>
|
<>
|
||||||
{!session && (
|
<div
|
||||||
<div
|
className="tooltip flex h-full w-full items-center justify-end pl-8 md:hidden"
|
||||||
data-tip="You have already started this assignment!"
|
data-tip="Your screen size is too small to perform an assignment"
|
||||||
className={clsx(
|
>
|
||||||
"-md:hidden h-full w-full max-w-[50%] cursor-pointer",
|
<Button className="h-full w-full !rounded-xl" variant="outline">
|
||||||
!!session && "tooltip",
|
Start
|
||||||
)}>
|
</Button>
|
||||||
<Button
|
</div>
|
||||||
className={clsx("w-full h-full !rounded-xl")}
|
{!session && (
|
||||||
onClick={() => startAssignment(assignment)}
|
<div
|
||||||
variant="outline">
|
data-tip="You have already started this assignment!"
|
||||||
Start
|
className={clsx(
|
||||||
</Button>
|
"-md:hidden h-full w-full max-w-[50%] cursor-pointer",
|
||||||
</div>
|
!!session && "tooltip"
|
||||||
)}
|
)}
|
||||||
{!!session && (
|
>
|
||||||
<div
|
<Button
|
||||||
className={clsx(
|
className={clsx("w-full h-full !rounded-xl")}
|
||||||
"-md:hidden h-full w-full max-w-[50%] cursor-pointer"
|
onClick={() => startAssignment(assignment)}
|
||||||
)}>
|
variant="outline"
|
||||||
<Button
|
>
|
||||||
className={clsx("w-full h-full !rounded-xl")}
|
Start
|
||||||
onClick={() => resumeAssignment(session)}
|
</Button>
|
||||||
color="green"
|
</div>
|
||||||
variant="outline">
|
)}
|
||||||
Resume
|
{!!session && (
|
||||||
</Button>
|
<div
|
||||||
</div>
|
className={clsx(
|
||||||
)}
|
"-md:hidden h-full w-full max-w-[50%] cursor-pointer"
|
||||||
</>
|
)}
|
||||||
)}
|
>
|
||||||
{hasBeenSubmitted && (
|
<Button
|
||||||
<Button
|
className={clsx("w-full h-full !rounded-xl")}
|
||||||
color="green"
|
onClick={() => resumeAssignment(session)}
|
||||||
className="-md:hidden h-full w-full max-w-[50%] !rounded-xl"
|
color="green"
|
||||||
disabled
|
variant="outline"
|
||||||
variant="outline">
|
>
|
||||||
Submitted
|
Resume
|
||||||
</Button>
|
</Button>
|
||||||
)}
|
</div>
|
||||||
</div>
|
)}
|
||||||
</div>
|
</>
|
||||||
)
|
)}
|
||||||
|
{hasBeenSubmitted && (
|
||||||
|
<Button
|
||||||
|
color="green"
|
||||||
|
className="-md:hidden h-full w-full max-w-[50%] !rounded-xl"
|
||||||
|
disabled
|
||||||
|
variant="outline"
|
||||||
|
>
|
||||||
|
Submitted
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,8 +2,6 @@ import {useListSearch} from "@/hooks/useListSearch";
|
|||||||
import usePagination from "@/hooks/usePagination";
|
import usePagination from "@/hooks/usePagination";
|
||||||
import { clsx } from "clsx";
|
import { clsx } from "clsx";
|
||||||
import {ReactNode} from "react";
|
import {ReactNode} from "react";
|
||||||
import Checkbox from "../Low/Checkbox";
|
|
||||||
import Separator from "../Low/Separator";
|
|
||||||
|
|
||||||
interface Props<T> {
|
interface Props<T> {
|
||||||
list: T[];
|
list: T[];
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
import useEntities from "@/hooks/useEntities";
|
|
||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
import { User } from "@/interfaces/user";
|
import { User } from "@/interfaces/user";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
@@ -6,66 +5,126 @@ import { useRouter } from "next/router";
|
|||||||
import { ToastContainer } from "react-toastify";
|
import { ToastContainer } from "react-toastify";
|
||||||
import Navbar from "../Navbar";
|
import Navbar from "../Navbar";
|
||||||
import Sidebar from "../Sidebar";
|
import Sidebar from "../Sidebar";
|
||||||
|
import React, { useEffect, useState } from "react";
|
||||||
|
|
||||||
|
export const LayoutContext = React.createContext({
|
||||||
|
onFocusLayerMouseEnter: () => {},
|
||||||
|
setOnFocusLayerMouseEnter: (() => {}) as React.Dispatch<
|
||||||
|
React.SetStateAction<() => void>
|
||||||
|
>,
|
||||||
|
navDisabled: false,
|
||||||
|
setNavDisabled: (() => {}) as React.Dispatch<React.SetStateAction<boolean>>,
|
||||||
|
focusMode: false,
|
||||||
|
setFocusMode: (() => {}) as React.Dispatch<React.SetStateAction<boolean>>,
|
||||||
|
hideSidebar: false,
|
||||||
|
setHideSidebar: (() => {}) as React.Dispatch<React.SetStateAction<boolean>>,
|
||||||
|
bgColor: "bg-white",
|
||||||
|
setBgColor: (() => {}) as React.Dispatch<React.SetStateAction<string>>,
|
||||||
|
className: "",
|
||||||
|
setClassName: (() => {}) as React.Dispatch<React.SetStateAction<string>>,
|
||||||
|
});
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
user: User;
|
user: User;
|
||||||
entities?: EntityWithRoles[]
|
entities?: EntityWithRoles[];
|
||||||
children: React.ReactNode;
|
children: React.ReactNode;
|
||||||
className?: string;
|
refreshPage?: boolean;
|
||||||
navDisabled?: boolean;
|
|
||||||
focusMode?: boolean;
|
|
||||||
hideSidebar?: boolean
|
|
||||||
bgColor?: string;
|
|
||||||
onFocusLayerMouseEnter?: () => void;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Layout({
|
export default function Layout({
|
||||||
user,
|
user,
|
||||||
children,
|
entities,
|
||||||
className,
|
children,
|
||||||
bgColor = "bg-white",
|
refreshPage,
|
||||||
hideSidebar,
|
|
||||||
navDisabled = false,
|
|
||||||
focusMode = false,
|
|
||||||
onFocusLayerMouseEnter
|
|
||||||
}: Props) {
|
}: Props) {
|
||||||
const router = useRouter();
|
const [onFocusLayerMouseEnter, setOnFocusLayerMouseEnter] = useState(
|
||||||
const { entities } = useEntities()
|
() => () => {}
|
||||||
|
);
|
||||||
|
const [navDisabled, setNavDisabled] = useState(false);
|
||||||
|
const [focusMode, setFocusMode] = useState(false);
|
||||||
|
const [hideSidebar, setHideSidebar] = useState(false);
|
||||||
|
const [bgColor, setBgColor] = useState("bg-white");
|
||||||
|
const [className, setClassName] = useState("");
|
||||||
|
|
||||||
return (
|
useEffect(() => {
|
||||||
<main className={clsx("w-full min-h-full h-screen flex flex-col bg-mti-gray-smoke relative")}>
|
if (refreshPage) {
|
||||||
<ToastContainer />
|
setClassName("");
|
||||||
{!hideSidebar && user && (
|
setBgColor("bg-white");
|
||||||
<Navbar
|
setFocusMode(false);
|
||||||
path={router.pathname}
|
setHideSidebar(false);
|
||||||
user={user}
|
setNavDisabled(false);
|
||||||
navDisabled={navDisabled}
|
setOnFocusLayerMouseEnter(() => () => {});
|
||||||
focusMode={focusMode}
|
}
|
||||||
onFocusLayerMouseEnter={onFocusLayerMouseEnter}
|
}, [refreshPage]);
|
||||||
/>
|
|
||||||
)}
|
const LayoutContextValue = React.useMemo(
|
||||||
<div className={clsx("h-full w-full flex gap-2")}>
|
() => ({
|
||||||
{!hideSidebar && user && (
|
onFocusLayerMouseEnter,
|
||||||
<Sidebar
|
setOnFocusLayerMouseEnter,
|
||||||
path={router.pathname}
|
navDisabled,
|
||||||
navDisabled={navDisabled}
|
setNavDisabled,
|
||||||
focusMode={focusMode}
|
focusMode,
|
||||||
onFocusLayerMouseEnter={onFocusLayerMouseEnter}
|
setFocusMode,
|
||||||
className="-md:hidden"
|
hideSidebar,
|
||||||
user={user}
|
setHideSidebar,
|
||||||
entities={entities}
|
bgColor,
|
||||||
/>
|
setBgColor,
|
||||||
)}
|
className,
|
||||||
<div
|
setClassName,
|
||||||
className={clsx(
|
}),
|
||||||
`w-full min-h-full ${bgColor} shadow-md rounded-2xl p-4 xl:p-10 pb-8 flex flex-col gap-8 relative overflow-hidden mt-2`,
|
[
|
||||||
bgColor !== "bg-white" ? "justify-center" : "h-fit",
|
bgColor,
|
||||||
hideSidebar ? "md:mx-8" : "md:mr-8",
|
className,
|
||||||
className,
|
focusMode,
|
||||||
)}>
|
hideSidebar,
|
||||||
{children}
|
navDisabled,
|
||||||
</div>
|
onFocusLayerMouseEnter,
|
||||||
</div>
|
]
|
||||||
</main>
|
);
|
||||||
);
|
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<LayoutContext.Provider value={LayoutContextValue}>
|
||||||
|
<main
|
||||||
|
className={clsx(
|
||||||
|
"w-full min-h-full h-screen flex flex-col bg-mti-gray-smoke relative"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<ToastContainer />
|
||||||
|
{!hideSidebar && user && (
|
||||||
|
<Navbar
|
||||||
|
path={router.pathname}
|
||||||
|
user={user}
|
||||||
|
navDisabled={navDisabled}
|
||||||
|
focusMode={focusMode}
|
||||||
|
onFocusLayerMouseEnter={onFocusLayerMouseEnter}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<div className={clsx("h-full w-full flex gap-2")}>
|
||||||
|
{!hideSidebar && user && (
|
||||||
|
<Sidebar
|
||||||
|
path={router.pathname}
|
||||||
|
navDisabled={navDisabled}
|
||||||
|
focusMode={focusMode}
|
||||||
|
onFocusLayerMouseEnter={onFocusLayerMouseEnter}
|
||||||
|
className="-md:hidden"
|
||||||
|
user={user}
|
||||||
|
entities={entities}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
`w-full min-h-full ${bgColor} shadow-md rounded-2xl p-4 xl:p-10 pb-8 flex flex-col gap-8 relative overflow-hidden mt-2`,
|
||||||
|
bgColor !== "bg-white" ? "justify-center" : "h-fit",
|
||||||
|
hideSidebar ? "md:mx-8" : "md:mr-8",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
|
</LayoutContext.Provider>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,109 +1,165 @@
|
|||||||
import { useListSearch } from "@/hooks/useListSearch"
|
import { useListSearch } from "@/hooks/useListSearch";
|
||||||
import { ColumnDef, flexRender, getCoreRowModel, getPaginationRowModel, getSortedRowModel, PaginationState, useReactTable } from "@tanstack/react-table"
|
import {
|
||||||
import clsx from "clsx"
|
ColumnDef,
|
||||||
import { useEffect, useState } from "react"
|
flexRender,
|
||||||
import { BsArrowDown, BsArrowUp } from "react-icons/bs"
|
getCoreRowModel,
|
||||||
import Button from "../Low/Button"
|
getPaginationRowModel,
|
||||||
|
getSortedRowModel,
|
||||||
|
PaginationState,
|
||||||
|
useReactTable,
|
||||||
|
} from "@tanstack/react-table";
|
||||||
|
import clsx from "clsx";
|
||||||
|
import { useEffect, useState } from "react";
|
||||||
|
import { BsArrowDown, BsArrowUp } from "react-icons/bs";
|
||||||
|
import Button from "../Low/Button";
|
||||||
|
|
||||||
interface Props<T> {
|
interface Props<T> {
|
||||||
data: T[]
|
data: T[];
|
||||||
columns: ColumnDef<any, any>[]
|
columns: ColumnDef<any, any>[];
|
||||||
searchFields: string[][]
|
searchFields: string[][];
|
||||||
size?: number
|
size?: number;
|
||||||
onDownload?: (rows: T[]) => void
|
onDownload?: (rows: T[]) => void;
|
||||||
isDownloadLoading?: boolean
|
isDownloadLoading?: boolean;
|
||||||
searchPlaceholder?: string
|
searchPlaceholder?: string;
|
||||||
|
isLoading?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Table<T>({ data, columns, searchFields, size = 16, onDownload, isDownloadLoading, searchPlaceholder }: Props<T>) {
|
export default function Table<T>({
|
||||||
const [pagination, setPagination] = useState<PaginationState>({
|
data,
|
||||||
pageIndex: 0,
|
columns,
|
||||||
pageSize: size,
|
searchFields,
|
||||||
})
|
size = 16,
|
||||||
|
onDownload,
|
||||||
|
isDownloadLoading,
|
||||||
|
searchPlaceholder,
|
||||||
|
isLoading,
|
||||||
|
}: Props<T>) {
|
||||||
|
const [pagination, setPagination] = useState<PaginationState>({
|
||||||
|
pageIndex: 0,
|
||||||
|
pageSize: size,
|
||||||
|
});
|
||||||
|
|
||||||
const { rows, renderSearch } = useListSearch<T>(searchFields, data, searchPlaceholder);
|
const { rows, renderSearch } = useListSearch<T>(
|
||||||
|
searchFields,
|
||||||
|
data,
|
||||||
|
searchPlaceholder
|
||||||
|
);
|
||||||
|
|
||||||
const table = useReactTable({
|
const table = useReactTable({
|
||||||
data: rows,
|
data: rows,
|
||||||
columns,
|
columns,
|
||||||
getCoreRowModel: getCoreRowModel(),
|
getCoreRowModel: getCoreRowModel(),
|
||||||
getSortedRowModel: getSortedRowModel(),
|
getSortedRowModel: getSortedRowModel(),
|
||||||
getPaginationRowModel: getPaginationRowModel(),
|
getPaginationRowModel: getPaginationRowModel(),
|
||||||
onPaginationChange: setPagination,
|
onPaginationChange: setPagination,
|
||||||
state: {
|
state: {
|
||||||
pagination
|
pagination,
|
||||||
}
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="w-full flex flex-col gap-2">
|
<div className="w-full flex flex-col gap-2">
|
||||||
<div className="w-full flex gap-2 items-end">
|
<div className="w-full flex gap-2 items-end">
|
||||||
{renderSearch()}
|
{renderSearch()}
|
||||||
{onDownload && (
|
{onDownload && (
|
||||||
<Button isLoading={isDownloadLoading} className="w-full max-w-[200px] mb-1" variant="outline" onClick={() => onDownload(rows)}>
|
<Button
|
||||||
Download
|
isLoading={isDownloadLoading}
|
||||||
</Button>
|
className="w-full max-w-[200px] mb-1"
|
||||||
)
|
variant="outline"
|
||||||
}
|
onClick={() => onDownload(rows)}
|
||||||
</div>
|
>
|
||||||
|
Download
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
<div className="w-full flex gap-2 justify-between items-center">
|
<div className="w-full flex gap-2 justify-between items-center">
|
||||||
<div className="flex items-center gap-4 w-fit">
|
<div className="flex items-center gap-4 w-fit">
|
||||||
<Button className="w-[200px] h-fit" disabled={!table.getCanPreviousPage()} onClick={() => table.previousPage()}>
|
<Button
|
||||||
Previous Page
|
className="w-[200px] h-fit"
|
||||||
</Button>
|
disabled={!table.getCanPreviousPage()}
|
||||||
</div>
|
onClick={() => table.previousPage()}
|
||||||
<div className="flex items-center gap-4 w-fit">
|
>
|
||||||
<span className="flex items-center gap-1">
|
Previous Page
|
||||||
<div>Page</div>
|
</Button>
|
||||||
<strong>
|
</div>
|
||||||
{table.getState().pagination.pageIndex + 1} of{' '}
|
<div className="flex items-center gap-4 w-fit">
|
||||||
{table.getPageCount().toLocaleString()}
|
<span className="flex items-center gap-1">
|
||||||
</strong>
|
<div>Page</div>
|
||||||
<div>| Total: {table.getRowCount().toLocaleString()}</div>
|
<strong>
|
||||||
</span>
|
{table.getState().pagination.pageIndex + 1} of{" "}
|
||||||
<Button className="w-[200px]" disabled={!table.getCanNextPage()} onClick={() => table.nextPage()}>
|
{table.getPageCount().toLocaleString()}
|
||||||
Next Page
|
</strong>
|
||||||
</Button>
|
<div>| Total: {table.getRowCount().toLocaleString()}</div>
|
||||||
</div>
|
</span>
|
||||||
</div>
|
<Button
|
||||||
|
className="w-[200px]"
|
||||||
|
disabled={!table.getCanNextPage()}
|
||||||
|
onClick={() => table.nextPage()}
|
||||||
|
>
|
||||||
|
Next Page
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<table className="rounded-xl bg-mti-purple-ultralight/40 w-full">
|
<table className="rounded-xl bg-mti-purple-ultralight/40 w-full">
|
||||||
<thead>
|
<thead>
|
||||||
{table.getHeaderGroups().map((headerGroup) => (
|
{table.getHeaderGroups().map((headerGroup) => (
|
||||||
<tr key={headerGroup.id}>
|
<tr key={headerGroup.id}>
|
||||||
{headerGroup.headers.map((header) => (
|
{headerGroup.headers.map((header) => (
|
||||||
<th className="py-4 px-4 text-left" key={header.id} colSpan={header.colSpan}>
|
<th
|
||||||
<div
|
className="py-4 px-4 text-left"
|
||||||
className={clsx(header.column.getCanSort() && 'cursor-pointer select-none', 'flex items-center gap-2')}
|
key={header.id}
|
||||||
onClick={header.column.getToggleSortingHandler()}
|
colSpan={header.colSpan}
|
||||||
>
|
>
|
||||||
{flexRender(
|
<div
|
||||||
header.column.columnDef.header,
|
className={clsx(
|
||||||
header.getContext()
|
header.column.getCanSort() &&
|
||||||
)}
|
"cursor-pointer select-none",
|
||||||
{{
|
"flex items-center gap-2"
|
||||||
asc: <BsArrowUp />,
|
)}
|
||||||
desc: <BsArrowDown />,
|
onClick={header.column.getToggleSortingHandler()}
|
||||||
}[header.column.getIsSorted() as string] ?? null}
|
>
|
||||||
</div>
|
{flexRender(
|
||||||
</th>
|
header.column.columnDef.header,
|
||||||
))}
|
header.getContext()
|
||||||
</tr>
|
)}
|
||||||
))}
|
{{
|
||||||
</thead>
|
asc: <BsArrowUp />,
|
||||||
<tbody className="px-2 w-full">
|
desc: <BsArrowDown />,
|
||||||
{table.getRowModel().rows.map((row) => (
|
}[header.column.getIsSorted() as string] ?? null}
|
||||||
<tr className="odd:bg-white even:bg-mti-purple-ultralight/40 rounded-lg py-2" key={row.id}>
|
</div>
|
||||||
{row.getVisibleCells().map((cell) => (
|
</th>
|
||||||
<td className="px-4 py-2 items-center w-fit" key={cell.id}>
|
))}
|
||||||
{flexRender(cell.column.columnDef.cell, cell.getContext())}
|
</tr>
|
||||||
</td>
|
))}
|
||||||
))}
|
</thead>
|
||||||
</tr>
|
<tbody className="px-2 w-full">
|
||||||
))}
|
{table.getRowModel().rows.map((row) => (
|
||||||
</tbody>
|
<tr
|
||||||
</table>
|
className="odd:bg-white even:bg-mti-purple-ultralight/40 rounded-lg py-2"
|
||||||
</div>
|
key={row.id}
|
||||||
)
|
>
|
||||||
|
{row.getVisibleCells().map((cell) => (
|
||||||
|
<td className="px-4 py-2 items-center w-fit" key={cell.id}>
|
||||||
|
{flexRender(cell.column.columnDef.cell, cell.getContext())}
|
||||||
|
</td>
|
||||||
|
))}
|
||||||
|
</tr>
|
||||||
|
))}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
{isLoading ? (
|
||||||
|
<div className="min-h-screen flex justify-center items-start">
|
||||||
|
<span className="loading loading-infinity w-32" />
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
rows.length === 0 && (
|
||||||
|
<div className="w-full flex justify-center items-start">
|
||||||
|
<span className="text-xl text-gray-500">No data found...</span>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,9 +1,7 @@
|
|||||||
import {useListSearch} from "@/hooks/useListSearch";
|
import {useListSearch} from "@/hooks/useListSearch";
|
||||||
import usePagination from "@/hooks/usePagination";
|
import usePagination from "@/hooks/usePagination";
|
||||||
import {Column, flexRender, getCoreRowModel, getSortedRowModel, useReactTable} from "@tanstack/react-table";
|
import { flexRender, getCoreRowModel, getSortedRowModel, useReactTable} from "@tanstack/react-table";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import {useMemo, useState} from "react";
|
|
||||||
import Button from "./Low/Button";
|
|
||||||
|
|
||||||
const SIZE = 25;
|
const SIZE = 25;
|
||||||
|
|
||||||
|
|||||||
@@ -48,6 +48,15 @@ export default function AsyncSelect({
|
|||||||
flat,
|
flat,
|
||||||
}: Props & (MultiProps | SingleProps)) {
|
}: Props & (MultiProps | SingleProps)) {
|
||||||
const [target, setTarget] = useState<HTMLElement>();
|
const [target, setTarget] = useState<HTMLElement>();
|
||||||
|
const [inputValue, setInputValue] = useState("");
|
||||||
|
|
||||||
|
//Implemented a debounce to prevent the API from being called too frequently
|
||||||
|
useEffect(() => {
|
||||||
|
const timer = setTimeout(() => {
|
||||||
|
loadOptions(inputValue);
|
||||||
|
}, 200);
|
||||||
|
return () => clearTimeout(timer);
|
||||||
|
}, [inputValue, loadOptions]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (document) setTarget(document.body);
|
if (document) setTarget(document.body);
|
||||||
@@ -77,7 +86,7 @@ export default function AsyncSelect({
|
|||||||
filterOption={null}
|
filterOption={null}
|
||||||
loadingMessage={() => "Loading..."}
|
loadingMessage={() => "Loading..."}
|
||||||
onInputChange={(inputValue) => {
|
onInputChange={(inputValue) => {
|
||||||
loadOptions(inputValue);
|
setInputValue(inputValue);
|
||||||
}}
|
}}
|
||||||
options={options}
|
options={options}
|
||||||
value={value}
|
value={value}
|
||||||
|
|||||||
@@ -3,8 +3,6 @@ import { checkAccess } from "@/utils/permissions";
|
|||||||
import Select from "../Low/Select";
|
import Select from "../Low/Select";
|
||||||
import { ReactNode, useEffect, useMemo, useState } from "react";
|
import { ReactNode, useEffect, useMemo, useState } from "react";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import useUsers from "@/hooks/useUsers";
|
|
||||||
import useGroups from "@/hooks/useGroups";
|
|
||||||
import useRecordStore from "@/stores/recordStore";
|
import useRecordStore from "@/stores/recordStore";
|
||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
import { mapBy } from "@/utils";
|
import { mapBy } from "@/utils";
|
||||||
@@ -44,13 +42,13 @@ const RecordFilter: React.FC<Props> = ({
|
|||||||
|
|
||||||
const [entity, setEntity] = useState<string>();
|
const [entity, setEntity] = useState<string>();
|
||||||
|
|
||||||
const [, setStatsUserId] = useRecordStore((state) => [
|
const [selectedUser, setStatsUserId] = useRecordStore((state) => [
|
||||||
state.selectedUser,
|
state.selectedUser,
|
||||||
state.setSelectedUser,
|
state.setSelectedUser,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
const entitiesToSearch = useMemo(() => {
|
const entitiesToSearch = useMemo(() => {
|
||||||
if(entity) return entity
|
if (entity) return entity;
|
||||||
if (isAdmin) return undefined;
|
if (isAdmin) return undefined;
|
||||||
return mapBy(entities, "id");
|
return mapBy(entities, "id");
|
||||||
}, [entities, entity, isAdmin]);
|
}, [entities, entity, isAdmin]);
|
||||||
@@ -68,7 +66,15 @@ const RecordFilter: React.FC<Props> = ({
|
|||||||
entities,
|
entities,
|
||||||
"view_student_record"
|
"view_student_record"
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const selectedUserValue = useMemo(
|
||||||
|
() =>
|
||||||
|
users.find((u) => u.id === selectedUser) || {
|
||||||
|
value: user.id,
|
||||||
|
label: `${user.name} - ${user.email}`,
|
||||||
|
},
|
||||||
|
[selectedUser, user, users]
|
||||||
|
);
|
||||||
|
|
||||||
useEffect(() => setStatsUserId(user.id), [setStatsUserId, user.id]);
|
useEffect(() => setStatsUserId(user.id), [setStatsUserId, user.id]);
|
||||||
|
|
||||||
@@ -118,10 +124,7 @@ const RecordFilter: React.FC<Props> = ({
|
|||||||
loadOptions={loadOptions}
|
loadOptions={loadOptions}
|
||||||
onMenuScrollToBottom={onScrollLoadMoreOptions}
|
onMenuScrollToBottom={onScrollLoadMoreOptions}
|
||||||
options={users}
|
options={users}
|
||||||
defaultValue={{
|
defaultValue={selectedUserValue}
|
||||||
value: user.id,
|
|
||||||
label: `${user.name} - ${user.email}`,
|
|
||||||
}}
|
|
||||||
onChange={(value) => setStatsUserId(value?.value!)}
|
onChange={(value) => setStatsUserId(value?.value!)}
|
||||||
styles={{
|
styles={{
|
||||||
menuPortal: (base) => ({ ...base, zIndex: 9999 }),
|
menuPortal: (base) => ({ ...base, zIndex: 9999 }),
|
||||||
|
|||||||
@@ -12,7 +12,6 @@ import { useRouter } from "next/router";
|
|||||||
import { uniqBy } from "lodash";
|
import { uniqBy } from "lodash";
|
||||||
import { sortByModule } from "@/utils/moduleUtils";
|
import { sortByModule } from "@/utils/moduleUtils";
|
||||||
import { getExamById } from "@/utils/exams";
|
import { getExamById } from "@/utils/exams";
|
||||||
import { Exam, UserSolution } from "@/interfaces/exam";
|
|
||||||
import ModuleBadge from "../ModuleBadge";
|
import ModuleBadge from "../ModuleBadge";
|
||||||
import useExamStore from "@/stores/exam";
|
import useExamStore from "@/stores/exam";
|
||||||
import { findBy } from "@/utils";
|
import { findBy } from "@/utils";
|
||||||
|
|||||||
60
src/components/Medium/UserProfileSkeleton.tsx
Normal file
60
src/components/Medium/UserProfileSkeleton.tsx
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
import React from "react";
|
||||||
|
|
||||||
|
export default function UserProfileSkeleton() {
|
||||||
|
return (
|
||||||
|
<div className="bg-white min-h-screen p-6">
|
||||||
|
<div className="mt-6 bg-white p-6 rounded-lg flex gap-4 items-center">
|
||||||
|
<div className="h-64 w-60 bg-gray-300 animate-pulse rounded"></div>
|
||||||
|
<div className="flex-1">
|
||||||
|
<div className="h-12 w-64 bg-gray-300 animate-pulse rounded"></div>
|
||||||
|
<div className="flex justify-between items-center mt-1">
|
||||||
|
<div className="h-4 w-60 bg-gray-300 animate-pulse mt-2 rounded"></div>
|
||||||
|
<div className="h-8 w-32 bg-gray-300 animate-pulse mt-2 rounded"></div>
|
||||||
|
</div>
|
||||||
|
<div className="h-4 w-100 bg-gray-300 animate-pulse mt-2 rounded"></div>
|
||||||
|
<div className="mt-6 grid grid-cols-4 justify-item-start gap-4">
|
||||||
|
<div className="bg-white p-4 rounded-lg text-center flex flex-row items-center justify-center">
|
||||||
|
<div className="h-12 w-12 mx-2 bg-gray-300 animate-pulse rounded"></div>
|
||||||
|
<div className="flex flex-col">
|
||||||
|
<div className="h-4 w-4 bg-gray-300 animate-pulse mt-2 rounded"></div>
|
||||||
|
<div className="h-4 w-16 bg-gray-300 animate-pulse mt-2 rounded"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="bg-white p-4 rounded-lg text-center flex flex-row items-center justify-center">
|
||||||
|
<div className="h-12 w-12 mx-2 bg-gray-300 animate-pulse rounded"></div>
|
||||||
|
<div className="flex flex-col">
|
||||||
|
<div className="h-4 w-4 bg-gray-300 animate-pulse mt-2 rounded"></div>
|
||||||
|
<div className="h-4 w-16 bg-gray-300 animate-pulse mt-2 rounded"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="bg-white p-4 rounded-lg text-center flex flex-row items-center justify-center">
|
||||||
|
<div className="h-12 w-12 mx-2 bg-gray-300 animate-pulse rounded"></div>
|
||||||
|
<div className="flex flex-col">
|
||||||
|
<div className="h-4 w-4 bg-gray-300 animate-pulse mt-2 rounded"></div>
|
||||||
|
<div className="h-4 w-16 bg-gray-300 animate-pulse mt-2 rounded"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="bg-white p-4 rounded-lg text-center flex flex-row items-center justify-center">
|
||||||
|
<div className="h-12 w-12 mx-2 bg-gray-300 animate-pulse rounded"></div>
|
||||||
|
<div className="flex flex-col">
|
||||||
|
<div className="h-4 w-4 bg-gray-300 animate-pulse mt-2 rounded"></div>
|
||||||
|
<div className="h-4 w-16 bg-gray-300 animate-pulse mt-2 rounded"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="mt-6 bg-white p-6 rounded-lg">
|
||||||
|
<div className="h-6 w-40 bg-gray-300 animate-pulse rounded mb-4"></div>
|
||||||
|
<div className="space-y-4">
|
||||||
|
{[...Array(4)].map((_, i) => (
|
||||||
|
<div key={i} className="flex justify-between items-center">
|
||||||
|
<div className="h-4 w-24 bg-gray-300 animate-pulse rounded"></div>
|
||||||
|
<div className="h-2 w-3/4 bg-gray-300 animate-pulse rounded"></div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -2,18 +2,18 @@ import clsx from "clsx";
|
|||||||
import { IconType } from "react-icons";
|
import { IconType } from "react-icons";
|
||||||
import { MdSpaceDashboard } from "react-icons/md";
|
import { MdSpaceDashboard } from "react-icons/md";
|
||||||
import {
|
import {
|
||||||
BsFileEarmarkText,
|
BsFileEarmarkText,
|
||||||
BsClockHistory,
|
BsClockHistory,
|
||||||
BsPencil,
|
BsGraphUp,
|
||||||
BsGraphUp,
|
BsChevronBarRight,
|
||||||
BsChevronBarRight,
|
BsChevronBarLeft,
|
||||||
BsChevronBarLeft,
|
BsShieldFill,
|
||||||
BsShieldFill,
|
BsCloudFill,
|
||||||
BsCloudFill,
|
BsCurrencyDollar,
|
||||||
BsCurrencyDollar,
|
BsClipboardData,
|
||||||
BsClipboardData,
|
BsPeople,
|
||||||
BsPeople,
|
|
||||||
} from "react-icons/bs";
|
} from "react-icons/bs";
|
||||||
|
import { GoWorkflow } from "react-icons/go";
|
||||||
import { CiDumbbell } from "react-icons/ci";
|
import { CiDumbbell } from "react-icons/ci";
|
||||||
import { RiLogoutBoxFill } from "react-icons/ri";
|
import { RiLogoutBoxFill } from "react-icons/ri";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
@@ -24,218 +24,487 @@ import { preventNavigation } from "@/utils/navigation.disabled";
|
|||||||
import usePreferencesStore from "@/stores/preferencesStore";
|
import usePreferencesStore from "@/stores/preferencesStore";
|
||||||
import { User } from "@/interfaces/user";
|
import { User } from "@/interfaces/user";
|
||||||
import useTicketsListener from "@/hooks/useTicketsListener";
|
import useTicketsListener from "@/hooks/useTicketsListener";
|
||||||
import { checkAccess, getTypesOfUser } from "@/utils/permissions";
|
import { getTypesOfUser } from "@/utils/permissions";
|
||||||
import usePermissions from "@/hooks/usePermissions";
|
import usePermissions from "@/hooks/usePermissions";
|
||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
import { useAllowedEntities, useAllowedEntitiesSomePermissions } from "@/hooks/useEntityPermissions";
|
import {
|
||||||
|
useAllowedEntities,
|
||||||
|
useAllowedEntitiesSomePermissions,
|
||||||
|
} from "@/hooks/useEntityPermissions";
|
||||||
import { useMemo } from "react";
|
import { useMemo } from "react";
|
||||||
|
import { PermissionType } from "../interfaces/permissions";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
path: string;
|
path: string;
|
||||||
navDisabled?: boolean;
|
navDisabled?: boolean;
|
||||||
focusMode?: boolean;
|
focusMode?: boolean;
|
||||||
onFocusLayerMouseEnter?: () => void;
|
onFocusLayerMouseEnter?: () => void;
|
||||||
className?: string;
|
className?: string;
|
||||||
user: User;
|
user: User;
|
||||||
entities?: EntityWithRoles[]
|
entities?: EntityWithRoles[];
|
||||||
}
|
}
|
||||||
|
|
||||||
interface NavProps {
|
interface NavProps {
|
||||||
Icon: IconType;
|
Icon: IconType;
|
||||||
label: string;
|
label: string;
|
||||||
path: string;
|
path: string;
|
||||||
keyPath: string;
|
keyPath: string;
|
||||||
disabled?: boolean;
|
disabled?: boolean;
|
||||||
isMinimized?: boolean;
|
isMinimized?: boolean;
|
||||||
badge?: number;
|
badge?: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
const Nav = ({ Icon, label, path, keyPath, disabled = false, isMinimized = false, badge }: NavProps) => {
|
const Nav = ({
|
||||||
return (
|
Icon,
|
||||||
<Link
|
label,
|
||||||
href={!disabled ? keyPath : ""}
|
path,
|
||||||
className={clsx(
|
keyPath,
|
||||||
"flex items-center gap-4 rounded-full p-4 text-gray-500 hover:text-white",
|
disabled = false,
|
||||||
"transition-all duration-300 ease-in-out relative",
|
isMinimized = false,
|
||||||
disabled ? "hover:bg-mti-gray-dim cursor-not-allowed" : "hover:bg-mti-purple-light cursor-pointer",
|
badge,
|
||||||
path.startsWith(keyPath) && "bg-mti-purple-light text-white",
|
}: NavProps) => {
|
||||||
isMinimized ? "w-fit" : "w-full min-w-[200px] px-8 2xl:min-w-[220px]",
|
return (
|
||||||
)}>
|
<Link
|
||||||
<Icon size={24} />
|
href={!disabled ? keyPath : ""}
|
||||||
{!isMinimized && <span className="text-lg font-semibold">{label}</span>}
|
className={clsx(
|
||||||
{!!badge && badge > 0 && (
|
"flex items-center gap-4 rounded-full p-4 text-gray-500 hover:text-white",
|
||||||
<div
|
"transition-all duration-300 ease-in-out relative",
|
||||||
className={clsx(
|
disabled
|
||||||
"bg-mti-purple-light h-5 w-5 text-xs rounded-full flex items-center justify-center text-white",
|
? "hover:bg-mti-gray-dim cursor-not-allowed"
|
||||||
"transition ease-in-out duration-300",
|
: "hover:bg-mti-purple-light cursor-pointer",
|
||||||
isMinimized && "absolute right-0 top-0",
|
path.startsWith(keyPath) && "bg-mti-purple-light text-white",
|
||||||
)}>
|
isMinimized ? "w-fit" : "w-full min-w-[200px] px-8 2xl:min-w-[220px]"
|
||||||
{badge}
|
)}
|
||||||
</div>
|
>
|
||||||
)}
|
<Icon size={24} />
|
||||||
</Link>
|
{!isMinimized && <span className="text-lg font-semibold">{label}</span>}
|
||||||
);
|
{!!badge && badge > 0 && (
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
"bg-mti-purple-light h-5 w-5 text-xs rounded-full flex items-center justify-center text-white",
|
||||||
|
"transition ease-in-out duration-300",
|
||||||
|
isMinimized && "absolute right-0 top-0"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{badge}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</Link>
|
||||||
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function Sidebar({
|
export default function Sidebar({
|
||||||
path,
|
path,
|
||||||
entities = [],
|
entities = [],
|
||||||
navDisabled = false,
|
navDisabled = false,
|
||||||
focusMode = false,
|
focusMode = false,
|
||||||
user,
|
user,
|
||||||
onFocusLayerMouseEnter,
|
onFocusLayerMouseEnter,
|
||||||
className
|
className,
|
||||||
}: Props) {
|
}: Props) {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
const isAdmin = useMemo(() => ['developer', 'admin'].includes(user?.type), [user?.type])
|
const isAdmin = useMemo(
|
||||||
|
() => ["developer", "admin"].includes(user?.type),
|
||||||
|
[user?.type]
|
||||||
|
);
|
||||||
|
|
||||||
const [isMinimized, toggleMinimize] = usePreferencesStore((state) => [state.isSidebarMinimized, state.toggleSidebarMinimized]);
|
const [isMinimized, toggleMinimize] = usePreferencesStore((state) => [
|
||||||
|
state.isSidebarMinimized,
|
||||||
|
state.toggleSidebarMinimized,
|
||||||
|
]);
|
||||||
|
|
||||||
const { totalAssignedTickets } = useTicketsListener(user.id);
|
const { permissions } = usePermissions(user.id);
|
||||||
const { permissions } = usePermissions(user.id);
|
|
||||||
|
|
||||||
const entitiesAllowStatistics = useAllowedEntities(user, entities, "view_statistics")
|
const entitiesAllowStatistics = useAllowedEntities(
|
||||||
const entitiesAllowPaymentRecord = useAllowedEntities(user, entities, "view_payment_record")
|
user,
|
||||||
|
entities,
|
||||||
|
"view_statistics"
|
||||||
|
);
|
||||||
|
|
||||||
const entitiesAllowGeneration = useAllowedEntitiesSomePermissions(user, entities, [
|
const entitiesAllowPaymentRecord = useAllowedEntities(
|
||||||
"generate_reading", "generate_listening", "generate_writing", "generate_speaking", "generate_level"
|
user,
|
||||||
])
|
entities,
|
||||||
|
"view_payment_record"
|
||||||
|
);
|
||||||
|
const entitiesAllowGeneration = useAllowedEntitiesSomePermissions(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
[
|
||||||
|
"generate_reading",
|
||||||
|
"generate_listening",
|
||||||
|
"generate_writing",
|
||||||
|
"generate_speaking",
|
||||||
|
"generate_level",
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
const logout = async () => {
|
const sidebarPermissions = useMemo<{ [key: string]: boolean }>(() => {
|
||||||
axios.post("/api/logout").finally(() => {
|
if (user.type === "developer") {
|
||||||
setTimeout(() => router.reload(), 500);
|
return {
|
||||||
});
|
viewExams: true,
|
||||||
};
|
viewStats: true,
|
||||||
|
viewRecords: true,
|
||||||
|
viewTickets: true,
|
||||||
|
viewClassrooms: true,
|
||||||
|
viewSettings: true,
|
||||||
|
viewPaymentRecords: true,
|
||||||
|
viewGeneration: true,
|
||||||
|
viewApprovalWorkflows: true,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
const sidebarPermissions: { [key: string]: boolean } = {
|
||||||
|
viewExams: false,
|
||||||
|
viewStats: false,
|
||||||
|
viewRecords: false,
|
||||||
|
viewTickets: false,
|
||||||
|
viewClassrooms: false,
|
||||||
|
viewSettings: false,
|
||||||
|
viewPaymentRecords: false,
|
||||||
|
viewGeneration: false,
|
||||||
|
viewApprovalWorkflows: false,
|
||||||
|
};
|
||||||
|
|
||||||
const disableNavigation = preventNavigation(navDisabled, focusMode);
|
if (!user || !user?.type) return sidebarPermissions;
|
||||||
|
|
||||||
return (
|
const neededPermissions = permissions.reduce((acc, curr) => {
|
||||||
<section
|
if (
|
||||||
className={clsx(
|
["viewExams", "viewRecords", "viewTickets"].includes(curr as string)
|
||||||
"relative flex h-full flex-col justify-between bg-transparent px-4 py-4 pb-8",
|
) {
|
||||||
isMinimized ? "w-fit" : "-xl:w-fit w-1/6",
|
acc.push(curr);
|
||||||
className,
|
}
|
||||||
)}>
|
return acc;
|
||||||
<div className="-xl:hidden flex-col gap-3 xl:flex">
|
}, [] as PermissionType[]);
|
||||||
<Nav disabled={disableNavigation} Icon={MdSpaceDashboard} label="Dashboard" path={path} keyPath="/dashboard" isMinimized={isMinimized} />
|
|
||||||
{checkAccess(user, ["student", "teacher", "developer"], permissions, "viewExams") && (
|
|
||||||
<Nav disabled={disableNavigation} Icon={BsFileEarmarkText} label="Practice" path={path} keyPath="/exam" isMinimized={isMinimized} />
|
|
||||||
)}
|
|
||||||
{checkAccess(user, getTypesOfUser(["agent"])) && entitiesAllowStatistics.length > 0 && (
|
|
||||||
<Nav disabled={disableNavigation} Icon={BsGraphUp} label="Stats" path={path} keyPath="/stats" isMinimized={isMinimized} />
|
|
||||||
)}
|
|
||||||
{checkAccess(user, ["developer", "admin", "mastercorporate", "corporate", "teacher", "student"], permissions) && (
|
|
||||||
<Nav
|
|
||||||
disabled={disableNavigation}
|
|
||||||
Icon={BsPeople}
|
|
||||||
label="Classrooms"
|
|
||||||
path={path}
|
|
||||||
keyPath="/classrooms"
|
|
||||||
isMinimized={isMinimized}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
{checkAccess(user, getTypesOfUser(["agent"]), permissions, "viewRecords") && (
|
|
||||||
<Nav disabled={disableNavigation} Icon={BsClockHistory} label="Record" path={path} keyPath="/record" isMinimized={isMinimized} />
|
|
||||||
)}
|
|
||||||
{checkAccess(user, getTypesOfUser(["agent"]), permissions, "viewRecords") && (
|
|
||||||
<Nav disabled={disableNavigation} Icon={CiDumbbell} label="Training" path={path} keyPath="/training" isMinimized={isMinimized} />
|
|
||||||
)}
|
|
||||||
{checkAccess(user, ["admin", "developer", "agent", "corporate", "mastercorporate"]) && entitiesAllowPaymentRecord.length > 0 && (
|
|
||||||
<Nav
|
|
||||||
disabled={disableNavigation}
|
|
||||||
Icon={BsCurrencyDollar}
|
|
||||||
label="Payment Record"
|
|
||||||
path={path}
|
|
||||||
keyPath="/payment-record"
|
|
||||||
isMinimized={isMinimized}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
{checkAccess(user, ["admin", "developer", "corporate", "teacher", "mastercorporate"]) && (
|
|
||||||
<Nav
|
|
||||||
disabled={disableNavigation}
|
|
||||||
Icon={BsShieldFill}
|
|
||||||
label="Settings"
|
|
||||||
path={path}
|
|
||||||
keyPath="/settings"
|
|
||||||
isMinimized={isMinimized}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
{checkAccess(user, ["admin", "developer", "agent"], permissions, "viewTickets") && (
|
|
||||||
<Nav
|
|
||||||
disabled={disableNavigation}
|
|
||||||
Icon={BsClipboardData}
|
|
||||||
label="Tickets"
|
|
||||||
path={path}
|
|
||||||
keyPath="/tickets"
|
|
||||||
isMinimized={isMinimized}
|
|
||||||
badge={totalAssignedTickets}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
{checkAccess(user, ["admin", "developer", "teacher", 'corporate', 'mastercorporate'])
|
|
||||||
&& (entitiesAllowGeneration.length > 0 || isAdmin) && (
|
|
||||||
<Nav
|
|
||||||
disabled={disableNavigation}
|
|
||||||
Icon={BsCloudFill}
|
|
||||||
label="Generation"
|
|
||||||
path={path}
|
|
||||||
keyPath="/generation"
|
|
||||||
isMinimized={isMinimized}
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
<div className="-xl:flex flex-col gap-3 xl:hidden">
|
|
||||||
<Nav disabled={disableNavigation} Icon={MdSpaceDashboard} label="Dashboard" path={path} keyPath="/" isMinimized />
|
|
||||||
<Nav disabled={disableNavigation} Icon={BsFileEarmarkText} label="Exams" path={path} keyPath="/exam" isMinimized />
|
|
||||||
{checkAccess(user, getTypesOfUser(["agent"]), permissions, "viewStats") && (
|
|
||||||
<Nav disabled={disableNavigation} Icon={BsGraphUp} label="Stats" path={path} keyPath="/stats" isMinimized />
|
|
||||||
)}
|
|
||||||
{checkAccess(user, getTypesOfUser(["agent"]), permissions, "viewRecords") && (
|
|
||||||
<Nav disabled={disableNavigation} Icon={BsClockHistory} label="Record" path={path} keyPath="/record" isMinimized />
|
|
||||||
)}
|
|
||||||
{checkAccess(user, getTypesOfUser(["agent"]), permissions, "viewRecords") && (
|
|
||||||
<Nav disabled={disableNavigation} Icon={CiDumbbell} label="Training" path={path} keyPath="/training" isMinimized />
|
|
||||||
)}
|
|
||||||
{checkAccess(user, getTypesOfUser(["student"])) && (
|
|
||||||
<Nav disabled={disableNavigation} Icon={BsShieldFill} label="Settings" path={path} keyPath="/settings" isMinimized />
|
|
||||||
)}
|
|
||||||
{entitiesAllowGeneration.length > 0 && (
|
|
||||||
<Nav
|
|
||||||
disabled={disableNavigation}
|
|
||||||
Icon={BsCloudFill}
|
|
||||||
label="Generation"
|
|
||||||
path={path}
|
|
||||||
keyPath="/generation"
|
|
||||||
isMinimized
|
|
||||||
/>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="2xl:fixed bottom-12 flex flex-col gap-0 -2xl:mt-8">
|
if (
|
||||||
<div
|
["student", "teacher", "developer"].includes(user.type) &&
|
||||||
role="button"
|
neededPermissions.includes("viewExams")
|
||||||
tabIndex={1}
|
) {
|
||||||
onClick={toggleMinimize}
|
sidebarPermissions["viewExams"] = true;
|
||||||
className={clsx(
|
}
|
||||||
"hover:text-mti-rose -xl:hidden flex cursor-pointer items-center gap-4 rounded-full p-4 text-black transition duration-300 ease-in-out",
|
if (
|
||||||
isMinimized ? "w-fit" : "w-full min-w-[250px] px-8",
|
getTypesOfUser(["agent"]).includes(user.type) &&
|
||||||
)}>
|
(entitiesAllowStatistics.length > 0 ||
|
||||||
{isMinimized ? <BsChevronBarRight size={24} /> : <BsChevronBarLeft size={24} />}
|
neededPermissions.includes("viewStats"))
|
||||||
{!isMinimized && <span className="text-lg font-medium">Minimize</span>}
|
) {
|
||||||
</div>
|
sidebarPermissions["viewStats"] = true;
|
||||||
<div
|
}
|
||||||
role="button"
|
if (
|
||||||
tabIndex={1}
|
[
|
||||||
onClick={focusMode ? () => { } : logout}
|
"admin",
|
||||||
className={clsx(
|
"developer",
|
||||||
"hover:text-mti-rose flex cursor-pointer items-center gap-4 rounded-full p-4 text-black transition duration-300 ease-in-out",
|
"teacher",
|
||||||
isMinimized ? "w-fit" : "w-full min-w-[250px] px-8",
|
"corporate",
|
||||||
)}>
|
"mastercorporate",
|
||||||
<RiLogoutBoxFill size={24} />
|
].includes(user.type) &&
|
||||||
{!isMinimized && <span className="-xl:hidden text-lg font-medium">Log Out</span>}
|
(entitiesAllowGeneration.length > 0 || isAdmin)
|
||||||
</div>
|
) {
|
||||||
</div>
|
sidebarPermissions["viewGeneration"] = true;
|
||||||
{focusMode && <FocusLayer onFocusLayerMouseEnter={onFocusLayerMouseEnter} />}
|
sidebarPermissions["viewApprovalWorkflows"] = true;
|
||||||
</section>
|
}
|
||||||
);
|
if (
|
||||||
|
getTypesOfUser(["agent"]).includes(user.type) &&
|
||||||
|
neededPermissions.includes("viewRecords")
|
||||||
|
) {
|
||||||
|
sidebarPermissions["viewRecords"] = true;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
["admin", "developer", "agent"].includes(user.type) &&
|
||||||
|
neededPermissions.includes("viewTickets")
|
||||||
|
) {
|
||||||
|
sidebarPermissions["viewTickets"] = true;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
[
|
||||||
|
"admin",
|
||||||
|
"mastercorporate",
|
||||||
|
"developer",
|
||||||
|
"corporate",
|
||||||
|
"teacher",
|
||||||
|
"student",
|
||||||
|
].includes(user.type)
|
||||||
|
) {
|
||||||
|
sidebarPermissions["viewClassrooms"] = true;
|
||||||
|
}
|
||||||
|
if (getTypesOfUser(["student", "agent"]).includes(user.type)) {
|
||||||
|
sidebarPermissions["viewSettings"] = true;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
["admin", "developer", "agent", "corporate", "mastercorporate"].includes(
|
||||||
|
user.type
|
||||||
|
) &&
|
||||||
|
entitiesAllowPaymentRecord.length > 0
|
||||||
|
) {
|
||||||
|
sidebarPermissions["viewPaymentRecords"] = true;
|
||||||
|
}
|
||||||
|
return sidebarPermissions;
|
||||||
|
}, [
|
||||||
|
entitiesAllowGeneration.length,
|
||||||
|
entitiesAllowPaymentRecord.length,
|
||||||
|
entitiesAllowStatistics.length,
|
||||||
|
isAdmin,
|
||||||
|
permissions,
|
||||||
|
user,
|
||||||
|
]);
|
||||||
|
|
||||||
|
const { totalAssignedTickets } = useTicketsListener(
|
||||||
|
user.id,
|
||||||
|
sidebarPermissions["viewTickets"]
|
||||||
|
);
|
||||||
|
|
||||||
|
const logout = async () => {
|
||||||
|
axios.post("/api/logout").finally(() => {
|
||||||
|
setTimeout(() => router.reload(), 500);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const disableNavigation = preventNavigation(navDisabled, focusMode);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<section
|
||||||
|
className={clsx(
|
||||||
|
"relative flex h-full flex-col justify-between bg-transparent px-4 py-4 pb-8",
|
||||||
|
isMinimized ? "w-fit" : "-xl:w-20 w-1/6",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<div className="-xl:hidden flex-col gap-3 xl:flex">
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={MdSpaceDashboard}
|
||||||
|
label="Dashboard"
|
||||||
|
path={path}
|
||||||
|
keyPath="/dashboard"
|
||||||
|
isMinimized={isMinimized}
|
||||||
|
/>
|
||||||
|
{sidebarPermissions["viewExams"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsFileEarmarkText}
|
||||||
|
label="Practice"
|
||||||
|
path={path}
|
||||||
|
keyPath="/exam"
|
||||||
|
isMinimized={isMinimized}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewStats"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsGraphUp}
|
||||||
|
label="Stats"
|
||||||
|
path={path}
|
||||||
|
keyPath="/stats"
|
||||||
|
isMinimized={isMinimized}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewClassrooms"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsPeople}
|
||||||
|
label="Classrooms"
|
||||||
|
path={path}
|
||||||
|
keyPath="/classrooms"
|
||||||
|
isMinimized={isMinimized}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewRecords"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsClockHistory}
|
||||||
|
label="Record"
|
||||||
|
path={path}
|
||||||
|
keyPath="/record"
|
||||||
|
isMinimized={isMinimized}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewRecords"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={CiDumbbell}
|
||||||
|
label="Training"
|
||||||
|
path={path}
|
||||||
|
keyPath="/training"
|
||||||
|
isMinimized={isMinimized}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewPaymentRecords"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsCurrencyDollar}
|
||||||
|
label="Payment Record"
|
||||||
|
path={path}
|
||||||
|
keyPath="/payment-record"
|
||||||
|
isMinimized={isMinimized}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewSettings"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsShieldFill}
|
||||||
|
label="Settings"
|
||||||
|
path={path}
|
||||||
|
keyPath="/settings"
|
||||||
|
isMinimized={isMinimized}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewTickets"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsClipboardData}
|
||||||
|
label="Tickets"
|
||||||
|
path={path}
|
||||||
|
keyPath="/tickets"
|
||||||
|
isMinimized={isMinimized}
|
||||||
|
badge={totalAssignedTickets}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewGeneration"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsCloudFill}
|
||||||
|
label="Generation"
|
||||||
|
path={path}
|
||||||
|
keyPath="/generation"
|
||||||
|
isMinimized={isMinimized}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewApprovalWorkflows"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={GoWorkflow}
|
||||||
|
label="Approval Workflows"
|
||||||
|
path={path}
|
||||||
|
keyPath="/approval-workflows"
|
||||||
|
isMinimized={isMinimized}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="-xl:flex flex-col gap-3 xl:hidden">
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={MdSpaceDashboard}
|
||||||
|
label="Dashboard"
|
||||||
|
path={path}
|
||||||
|
keyPath="/"
|
||||||
|
isMinimized
|
||||||
|
/>
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsFileEarmarkText}
|
||||||
|
label="Exams"
|
||||||
|
path={path}
|
||||||
|
keyPath="/exam"
|
||||||
|
isMinimized
|
||||||
|
/>
|
||||||
|
{sidebarPermissions["viewStats"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsGraphUp}
|
||||||
|
label="Stats"
|
||||||
|
path={path}
|
||||||
|
keyPath="/stats"
|
||||||
|
isMinimized
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewRecords"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsClockHistory}
|
||||||
|
label="Record"
|
||||||
|
path={path}
|
||||||
|
keyPath="/record"
|
||||||
|
isMinimized
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewRecords"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={CiDumbbell}
|
||||||
|
label="Training"
|
||||||
|
path={path}
|
||||||
|
keyPath="/training"
|
||||||
|
isMinimized
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewPaymentRecords"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsCurrencyDollar}
|
||||||
|
label="Payment Record"
|
||||||
|
path={path}
|
||||||
|
keyPath="/payment-record"
|
||||||
|
isMinimized
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewSettings"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsShieldFill}
|
||||||
|
label="Settings"
|
||||||
|
path={path}
|
||||||
|
keyPath="/settings"
|
||||||
|
isMinimized
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewGeneration"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={BsCloudFill}
|
||||||
|
label="Generation"
|
||||||
|
path={path}
|
||||||
|
keyPath="/generation"
|
||||||
|
isMinimized
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{sidebarPermissions["viewApprovalWorkflows"] && (
|
||||||
|
<Nav
|
||||||
|
disabled={disableNavigation}
|
||||||
|
Icon={GoWorkflow}
|
||||||
|
label="Approval Workflows"
|
||||||
|
path={path}
|
||||||
|
keyPath="/approval-workflows"
|
||||||
|
isMinimized
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="2xl:fixed bottom-12 flex flex-col gap-0 -2xl:mt-8 ">
|
||||||
|
<div
|
||||||
|
role="button"
|
||||||
|
tabIndex={1}
|
||||||
|
onClick={toggleMinimize}
|
||||||
|
className={clsx(
|
||||||
|
"hover:text-mti-rose -xl:hidden flex cursor-pointer items-center gap-4 rounded-full p-4 text-black transition duration-300 ease-in-out",
|
||||||
|
isMinimized ? "w-fit" : "w-full min-w-[250px] px-8"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{isMinimized ? (
|
||||||
|
<BsChevronBarRight size={24} />
|
||||||
|
) : (
|
||||||
|
<BsChevronBarLeft size={24} />
|
||||||
|
)}
|
||||||
|
{!isMinimized && (
|
||||||
|
<span className="text-lg font-medium">Minimize</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
role="button"
|
||||||
|
tabIndex={1}
|
||||||
|
onClick={focusMode ? () => {} : logout}
|
||||||
|
className={clsx(
|
||||||
|
"hover:text-mti-rose flex cursor-pointer items-center gap-4 rounded-full p-4 text-black transition duration-300 ease-in-out -xl:px-4",
|
||||||
|
isMinimized ? "w-fit" : "w-full min-w-[250px] px-8"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<RiLogoutBoxFill size={24} />
|
||||||
|
{!isMinimized && (
|
||||||
|
<span className="-xl:hidden text-lg font-medium">Log Out</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{focusMode && (
|
||||||
|
<FocusLayer onFocusLayerMouseEnter={onFocusLayerMouseEnter} />
|
||||||
|
)}
|
||||||
|
</section>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -29,7 +29,7 @@ function QuestionSolutionArea({
|
|||||||
</div>
|
</div>
|
||||||
<div
|
<div
|
||||||
className={clsx(
|
className={clsx(
|
||||||
"w-56 h-10 border rounded-xl items-center justify-center flex gap-3 px-2",
|
"w-56 h-10 border self-center rounded-xl items-center justify-center flex gap-3 px-2",
|
||||||
!userSolution
|
!userSolution
|
||||||
? "border-mti-gray-davy"
|
? "border-mti-gray-davy"
|
||||||
: userSolution.option.toString() === question.solution.toString()
|
: userSolution.option.toString() === question.solution.toString()
|
||||||
|
|||||||
204
src/demo/approval_workflows.json
Normal file
204
src/demo/approval_workflows.json
Normal file
@@ -0,0 +1,204 @@
|
|||||||
|
[
|
||||||
|
{
|
||||||
|
"id": "kajhfakscbka-asacaca-acawesae",
|
||||||
|
"name": "English Exam 1st Quarter 2025",
|
||||||
|
"entityId": "64a92896-fa8c-4908-95f3-23ffe05560c5",
|
||||||
|
"modules": [
|
||||||
|
"reading",
|
||||||
|
"writing"
|
||||||
|
],
|
||||||
|
"requester": "ffdIipRyXTRmm10Sq2eg7P97rLB2",
|
||||||
|
"startDate": 1737712243906,
|
||||||
|
"status": "pending",
|
||||||
|
"steps": [
|
||||||
|
{
|
||||||
|
"stepType": "form-intake",
|
||||||
|
"stepNumber": 1,
|
||||||
|
"completed": true,
|
||||||
|
"completedBy": "5fZibjknlJdfIZVndlV2FIdamtn1",
|
||||||
|
"completedDate": 1737712243906,
|
||||||
|
"firstStep": true,
|
||||||
|
"assignees": [
|
||||||
|
"5fZibjknlJdfIZVndlV2FIdamtn1",
|
||||||
|
"50jqJuESQNX0Qas64B5JZBQTIiq1",
|
||||||
|
"2rtgJKmBXfWFzrtG8AjFgyrGBcp1"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment\nThis is a random comment\nThis is a random comment\nThis is a random comment\nThis is a random comment\n"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"stepType": "approval-by",
|
||||||
|
"stepNumber": 2,
|
||||||
|
"completed": true,
|
||||||
|
"completedBy": "50jqJuESQNX0Qas64B5JZBQTIiq1",
|
||||||
|
"completedDate": 1737712243906,
|
||||||
|
"assignees": [
|
||||||
|
"5fZibjknlJdfIZVndlV2FIdamtn1",
|
||||||
|
"50jqJuESQNX0Qas64B5JZBQTIiq1",
|
||||||
|
"2rtgJKmBXfWFzrtG8AjFgyrGBcp1"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"stepType": "approval-by",
|
||||||
|
"stepNumber": 3,
|
||||||
|
"completed": false,
|
||||||
|
"assignees": [
|
||||||
|
"5fZibjknlJdfIZVndlV2FIdamtn1",
|
||||||
|
"50jqJuESQNX0Qas64B5JZBQTIiq1",
|
||||||
|
"2rtgJKmBXfWFzrtG8AjFgyrGBcp1"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"stepType": "approval-by",
|
||||||
|
"stepNumber": 4,
|
||||||
|
"completed": false,
|
||||||
|
"assignees": [
|
||||||
|
"50jqJuESQNX0Qas64B5JZBQTIiq1"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"stepType": "approval-by",
|
||||||
|
"stepNumber": 5,
|
||||||
|
"completed": false,
|
||||||
|
"finalStep": true,
|
||||||
|
"assignees": [
|
||||||
|
"50jqJuESQNX0Qas64B5JZBQTIiq1",
|
||||||
|
"2rtgJKmBXfWFzrtG8AjFgyrGBcp1"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "aaaaaakscbka-asacaca-acawesae",
|
||||||
|
"name": "English Exam 2nd Quarter 2025",
|
||||||
|
"entityId": "64a92896-fa8c-4908-95f3-23ffe05560c5",
|
||||||
|
"modules": [
|
||||||
|
"reading",
|
||||||
|
"writing",
|
||||||
|
"level",
|
||||||
|
"speaking",
|
||||||
|
"listening"
|
||||||
|
],
|
||||||
|
"requester": "231c84b2-a65a-49a9-803c-c664d84b13e0",
|
||||||
|
"startDate": 1737712243906,
|
||||||
|
"status": "approved",
|
||||||
|
"steps": [
|
||||||
|
{
|
||||||
|
"stepType": "form-intake",
|
||||||
|
"stepNumber": 1,
|
||||||
|
"completed": true,
|
||||||
|
"completedBy": "fd5fce42-4bcc-4150-a143-b484e750b265",
|
||||||
|
"completedDate": 1737712243906,
|
||||||
|
"firstStep": true,
|
||||||
|
"assignees": [
|
||||||
|
"fd5fce42-4bcc-4150-a143-b484e750b265",
|
||||||
|
"231c84b2-a65a-49a9-803c-c664d84b13e0",
|
||||||
|
"c5fc1514-1a94-4f8c-a046-a62099097a50"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"stepType": "approval-by",
|
||||||
|
"stepNumber": 2,
|
||||||
|
"completed": true,
|
||||||
|
"completedBy": "rTh9yz6Z1WOidHlVOSGInlpoxrk1",
|
||||||
|
"completedDate": 1737712243906,
|
||||||
|
"assignees": [
|
||||||
|
"fd5fce42-4bcc-4150-a143-b484e750b265",
|
||||||
|
"rTh9yz6Z1WOidHlVOSGInlpoxrk1",
|
||||||
|
"c5fc1514-1a94-4f8c-a046-a62099097a50"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"stepType": "approval-by",
|
||||||
|
"stepNumber": 3,
|
||||||
|
"completed": true,
|
||||||
|
"completedBy": "231c84b2-a65a-49a9-803c-c664d84b13e0",
|
||||||
|
"completedDate": 1737712243906,
|
||||||
|
"assignees": [
|
||||||
|
"fd5fce42-4bcc-4150-a143-b484e750b265",
|
||||||
|
"231c84b2-a65a-49a9-803c-c664d84b13e0",
|
||||||
|
"c5fc1514-1a94-4f8c-a046-a62099097a50"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"stepType": "approval-by",
|
||||||
|
"stepNumber": 4,
|
||||||
|
"completed": true,
|
||||||
|
"completedBy": "231c84b2-a65a-49a9-803c-c664d84b13e0",
|
||||||
|
"completedDate": 1737712243906,
|
||||||
|
"assignees": [
|
||||||
|
"fd5fce42-4bcc-4150-a143-b484e750b265"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"stepType": "approval-by",
|
||||||
|
"stepNumber": 5,
|
||||||
|
"completed": true,
|
||||||
|
"completedBy": "c5fc1514-1a94-4f8c-a046-a62099097a50",
|
||||||
|
"completedDate": 1737712243906,
|
||||||
|
"finalStep": true,
|
||||||
|
"assignees": [
|
||||||
|
"rTh9yz6Z1WOidHlVOSGInlpoxrk1",
|
||||||
|
"c5fc1514-1a94-4f8c-a046-a62099097a50"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "bbbbkscbka-asacaca-acawesae",
|
||||||
|
"name": "English Exam 3rd Quarter 2025",
|
||||||
|
"entityId": "49ed2f0c-7d0d-46e4-9576-7cf19edc4980",
|
||||||
|
"modules": [
|
||||||
|
"reading"
|
||||||
|
],
|
||||||
|
"requester": "rTh9yz6Z1WOidHlVOSGInlpoxrk1",
|
||||||
|
"startDate": 1737712243906,
|
||||||
|
"status": "rejected",
|
||||||
|
"steps": [
|
||||||
|
{
|
||||||
|
"stepType": "form-intake",
|
||||||
|
"stepNumber": 1,
|
||||||
|
"completed": true,
|
||||||
|
"completedBy": "231c84b2-a65a-49a9-803c-c664d84b13e0",
|
||||||
|
"completedDate": 1737712243906,
|
||||||
|
"firstStep": true,
|
||||||
|
"assignees": [
|
||||||
|
"fd5fce42-4bcc-4150-a143-b484e750b265",
|
||||||
|
"231c84b2-a65a-49a9-803c-c664d84b13e0",
|
||||||
|
"c5fc1514-1a94-4f8c-a046-a62099097a50"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment\nThis is a random comment\nThis is a random comment\nThis is a random comment\nThis is a random comment\n"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"stepType": "approval-by",
|
||||||
|
"stepNumber": 2,
|
||||||
|
"completed": true,
|
||||||
|
"completedBy": "rTh9yz6Z1WOidHlVOSGInlpoxrk1",
|
||||||
|
"completedDate": 1737712243906,
|
||||||
|
"assignees": [
|
||||||
|
"rTh9yz6Z1WOidHlVOSGInlpoxrk1",
|
||||||
|
"c5fc1514-1a94-4f8c-a046-a62099097a50"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"stepType": "approval-by",
|
||||||
|
"stepNumber": 3,
|
||||||
|
"completed": false,
|
||||||
|
"finalStep": true,
|
||||||
|
"assignees": [
|
||||||
|
"rTh9yz6Z1WOidHlVOSGInlpoxrk1"
|
||||||
|
],
|
||||||
|
"comments": "This is a random comment"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
@@ -14,8 +14,6 @@ import {
|
|||||||
BsPen,
|
BsPen,
|
||||||
BsXCircle,
|
BsXCircle,
|
||||||
} from "react-icons/bs";
|
} from "react-icons/bs";
|
||||||
import { totalExamsByModule } from "@/utils/stats";
|
|
||||||
import useFilterRecordsByUser from "@/hooks/useFilterRecordsByUser";
|
|
||||||
import Button from "@/components/Low/Button";
|
import Button from "@/components/Low/Button";
|
||||||
import { sortByModuleName } from "@/utils/moduleUtils";
|
import { sortByModuleName } from "@/utils/moduleUtils";
|
||||||
import { capitalize } from "lodash";
|
import { capitalize } from "lodash";
|
||||||
@@ -24,7 +22,7 @@ import { Variant } from "@/interfaces/exam";
|
|||||||
import useSessions, { Session } from "@/hooks/useSessions";
|
import useSessions, { Session } from "@/hooks/useSessions";
|
||||||
import SessionCard from "@/components/Medium/SessionCard";
|
import SessionCard from "@/components/Medium/SessionCard";
|
||||||
import useExamStore from "@/stores/exam";
|
import useExamStore from "@/stores/exam";
|
||||||
import moment from "moment";
|
import useStats from "../hooks/useStats";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
user: User;
|
user: User;
|
||||||
@@ -41,7 +39,21 @@ export default function Selection({ user, page, onStart }: Props) {
|
|||||||
const [avoidRepeatedExams, setAvoidRepeatedExams] = useState(true);
|
const [avoidRepeatedExams, setAvoidRepeatedExams] = useState(true);
|
||||||
const [variant, setVariant] = useState<Variant>("full");
|
const [variant, setVariant] = useState<Variant>("full");
|
||||||
|
|
||||||
const { data: stats } = useFilterRecordsByUser<Stat[]>(user?.id);
|
const {
|
||||||
|
data: {
|
||||||
|
allStats = [],
|
||||||
|
moduleCount: { reading, listening, writing, speaking, level } = {
|
||||||
|
reading: 0,
|
||||||
|
listening: 0,
|
||||||
|
writing: 0,
|
||||||
|
speaking: 0,
|
||||||
|
level: 0,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
} = useStats<{
|
||||||
|
allStats: Stat[];
|
||||||
|
moduleCount: Record<Module, number>;
|
||||||
|
}>(user?.id, !user?.id, "byModule");
|
||||||
const { sessions, isLoading, reload } = useSessions(user.id);
|
const { sessions, isLoading, reload } = useSessions(user.id);
|
||||||
|
|
||||||
const dispatch = useExamStore((state) => state.dispatch);
|
const dispatch = useExamStore((state) => state.dispatch);
|
||||||
@@ -77,7 +89,7 @@ export default function Selection({ user, page, onStart }: Props) {
|
|||||||
<BsBook className="text-ielts-reading h-6 w-6 md:h-8 md:w-8" />
|
<BsBook className="text-ielts-reading h-6 w-6 md:h-8 md:w-8" />
|
||||||
),
|
),
|
||||||
label: "Reading",
|
label: "Reading",
|
||||||
value: totalExamsByModule(stats, "reading"),
|
value: reading || 0,
|
||||||
tooltip: "The amount of reading exams performed.",
|
tooltip: "The amount of reading exams performed.",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -85,7 +97,7 @@ export default function Selection({ user, page, onStart }: Props) {
|
|||||||
<BsHeadphones className="text-ielts-listening h-6 w-6 md:h-8 md:w-8" />
|
<BsHeadphones className="text-ielts-listening h-6 w-6 md:h-8 md:w-8" />
|
||||||
),
|
),
|
||||||
label: "Listening",
|
label: "Listening",
|
||||||
value: totalExamsByModule(stats, "listening"),
|
value: listening || 0,
|
||||||
tooltip: "The amount of listening exams performed.",
|
tooltip: "The amount of listening exams performed.",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -93,7 +105,7 @@ export default function Selection({ user, page, onStart }: Props) {
|
|||||||
<BsPen className="text-ielts-writing h-6 w-6 md:h-8 md:w-8" />
|
<BsPen className="text-ielts-writing h-6 w-6 md:h-8 md:w-8" />
|
||||||
),
|
),
|
||||||
label: "Writing",
|
label: "Writing",
|
||||||
value: totalExamsByModule(stats, "writing"),
|
value: writing || 0,
|
||||||
tooltip: "The amount of writing exams performed.",
|
tooltip: "The amount of writing exams performed.",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -101,7 +113,7 @@ export default function Selection({ user, page, onStart }: Props) {
|
|||||||
<BsMegaphone className="text-ielts-speaking h-6 w-6 md:h-8 md:w-8" />
|
<BsMegaphone className="text-ielts-speaking h-6 w-6 md:h-8 md:w-8" />
|
||||||
),
|
),
|
||||||
label: "Speaking",
|
label: "Speaking",
|
||||||
value: totalExamsByModule(stats, "speaking"),
|
value: speaking || 0,
|
||||||
tooltip: "The amount of speaking exams performed.",
|
tooltip: "The amount of speaking exams performed.",
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -109,7 +121,7 @@ export default function Selection({ user, page, onStart }: Props) {
|
|||||||
<BsClipboard className="text-ielts-level h-6 w-6 md:h-8 md:w-8" />
|
<BsClipboard className="text-ielts-level h-6 w-6 md:h-8 md:w-8" />
|
||||||
),
|
),
|
||||||
label: "Level",
|
label: "Level",
|
||||||
value: totalExamsByModule(stats, "level"),
|
value: level || 0,
|
||||||
tooltip: "The amount of level exams performed.",
|
tooltip: "The amount of level exams performed.",
|
||||||
},
|
},
|
||||||
]}
|
]}
|
||||||
|
|||||||
24
src/hooks/useApprovalWorkflow.tsx
Normal file
24
src/hooks/useApprovalWorkflow.tsx
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
import { ApprovalWorkflow } from "@/interfaces/approval.workflow";
|
||||||
|
import axios from "axios";
|
||||||
|
import { useCallback, useEffect, useState } from "react";
|
||||||
|
|
||||||
|
export default function useApprovalWorkflow(id: string) {
|
||||||
|
const [workflow, setWorkflow] = useState<ApprovalWorkflow>();
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
const [isError, setIsError] = useState(false);
|
||||||
|
|
||||||
|
const getData = useCallback(() => {
|
||||||
|
setIsLoading(true);
|
||||||
|
axios
|
||||||
|
.get<ApprovalWorkflow>(`/api/approval-workflows/${id}`)
|
||||||
|
.then((response) => setWorkflow(response.data))
|
||||||
|
.catch((error) => {
|
||||||
|
setIsError(true);
|
||||||
|
})
|
||||||
|
.finally(() => setIsLoading(false));
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(getData, [getData]);
|
||||||
|
|
||||||
|
return { workflow, isLoading, isError, reload: getData };
|
||||||
|
}
|
||||||
24
src/hooks/useApprovalWorkflows.tsx
Normal file
24
src/hooks/useApprovalWorkflows.tsx
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
import { ApprovalWorkflow } from "@/interfaces/approval.workflow";
|
||||||
|
import axios from "axios";
|
||||||
|
import { useCallback, useEffect, useState } from "react";
|
||||||
|
|
||||||
|
export default function useApprovalWorkflows(entitiesString?: string) {
|
||||||
|
const [workflows, setWorkflows] = useState<ApprovalWorkflow[]>([]);
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
const [isError, setIsError] = useState(false);
|
||||||
|
|
||||||
|
const getData = useCallback(() => {
|
||||||
|
setIsLoading(true);
|
||||||
|
axios
|
||||||
|
.get<ApprovalWorkflow[]>(`/api/approval-workflows`, {params: { entityIds: entitiesString }})
|
||||||
|
.then((response) => setWorkflows(response.data))
|
||||||
|
.catch((error) => {
|
||||||
|
setIsError(true);
|
||||||
|
})
|
||||||
|
.finally(() => setIsLoading(false));
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(getData, [getData]);
|
||||||
|
|
||||||
|
return { workflows, isLoading, isError, reload: getData };
|
||||||
|
}
|
||||||
@@ -1,23 +1,22 @@
|
|||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
import { Discount } from "@/interfaces/paypal";
|
|
||||||
import { Code, Group, User } from "@/interfaces/user";
|
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import { useEffect, useState } from "react";
|
import { useCallback, useEffect, useState } from "react";
|
||||||
|
|
||||||
export default function useEntities() {
|
export default function useEntities(shouldNot?: boolean) {
|
||||||
const [entities, setEntities] = useState<EntityWithRoles[]>([]);
|
const [entities, setEntities] = useState<EntityWithRoles[]>([]);
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [isError, setIsError] = useState(false);
|
const [isError, setIsError] = useState(false);
|
||||||
|
|
||||||
const getData = () => {
|
const getData = useCallback(() => {
|
||||||
|
if (shouldNot) return;
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
axios
|
axios
|
||||||
.get<EntityWithRoles[]>("/api/entities?showRoles=true")
|
.get<EntityWithRoles[]>("/api/entities?showRoles=true")
|
||||||
.then((response) => setEntities(response.data))
|
.then((response) => setEntities(response.data))
|
||||||
.finally(() => setIsLoading(false));
|
.finally(() => setIsLoading(false));
|
||||||
};
|
}, [shouldNot]);
|
||||||
|
|
||||||
useEffect(getData, []);
|
useEffect(getData, [getData])
|
||||||
|
|
||||||
return { entities, isLoading, isError, reload: getData };
|
return { entities, isLoading, isError, reload: getData };
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import { EntityWithRoles, WithLabeledEntities } from "@/interfaces/entity";
|
import { WithLabeledEntities } from "@/interfaces/entity";
|
||||||
import { Discount } from "@/interfaces/paypal";
|
import { Type, User } from "@/interfaces/user";
|
||||||
import { Code, Group, Type, User } from "@/interfaces/user";
|
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
|
|
||||||
@@ -12,7 +11,9 @@ export default function useEntitiesUsers(type?: Type) {
|
|||||||
const getData = () => {
|
const getData = () => {
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
axios
|
axios
|
||||||
.get<WithLabeledEntities<User>[]>(`/api/entities/users${type ? "?type=" + type : ""}`)
|
.get<WithLabeledEntities<User>[]>(
|
||||||
|
`/api/entities/users${type ? "?type=" + type : ""}`
|
||||||
|
)
|
||||||
.then((response) => setUsers(response.data))
|
.then((response) => setUsers(response.data))
|
||||||
.finally(() => setIsLoading(false));
|
.finally(() => setIsLoading(false));
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -1,21 +1,21 @@
|
|||||||
import {Exam} from "@/interfaces/exam";
|
import { Exam } from "@/interfaces/exam";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import {useEffect, useState} from "react";
|
import { useEffect, useState } from "react";
|
||||||
|
|
||||||
export default function useExams() {
|
export default function useExams() {
|
||||||
const [exams, setExams] = useState<Exam[]>([]);
|
const [exams, setExams] = useState<Exam[]>([]);
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [isError, setIsError] = useState(false);
|
const [isError, setIsError] = useState(false);
|
||||||
|
|
||||||
const getData = () => {
|
const getData = () => {
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
axios
|
axios
|
||||||
.get<Exam[]>("/api/exam")
|
.get<Exam[]>(`/api/exam`)
|
||||||
.then((response) => setExams(response.data))
|
.then((response) => setExams(response.data))
|
||||||
.finally(() => setIsLoading(false));
|
.finally(() => setIsLoading(false));
|
||||||
};
|
};
|
||||||
|
|
||||||
useEffect(getData, []);
|
useEffect(getData, []);
|
||||||
|
|
||||||
return {exams, isLoading, isError, reload: getData};
|
return { exams, isLoading, isError, reload: getData };
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,60 +1,138 @@
|
|||||||
import Button from "@/components/Low/Button";
|
import Button from "@/components/Low/Button";
|
||||||
import {useMemo, useState} from "react";
|
import { useEffect, useMemo, useState } from "react";
|
||||||
import {BiChevronLeft} from "react-icons/bi";
|
import {
|
||||||
import {BsChevronDoubleLeft, BsChevronDoubleRight, BsChevronLeft, BsChevronRight} from "react-icons/bs";
|
BsChevronDoubleLeft,
|
||||||
|
BsChevronDoubleRight,
|
||||||
|
BsChevronLeft,
|
||||||
|
BsChevronRight,
|
||||||
|
} from "react-icons/bs";
|
||||||
|
import Select from "../components/Low/Select";
|
||||||
|
|
||||||
export default function usePagination<T>(list: T[], size = 25) {
|
export default function usePagination<T>(list: T[], size = 25) {
|
||||||
const [page, setPage] = useState(0);
|
const [page, setPage] = useState(0);
|
||||||
|
const [itemsPerPage, setItemsPerPage] = useState(size);
|
||||||
|
|
||||||
const items = useMemo(() => list.slice(page * size, (page + 1) * size), [page, size, list]);
|
const items = useMemo(
|
||||||
|
() => list.slice(page * itemsPerPage, (page + 1) * itemsPerPage),
|
||||||
|
[list, page, itemsPerPage]
|
||||||
|
);
|
||||||
|
useEffect(() => {
|
||||||
|
if (page * itemsPerPage >= list.length) setPage(0);
|
||||||
|
}, [items, itemsPerPage, list.length, page]);
|
||||||
|
|
||||||
const render = () => (
|
const itemsPerPageOptions = [25, 50, 100, 200];
|
||||||
<div className="w-full flex gap-2 justify-between items-center">
|
|
||||||
<div className="flex items-center gap-4 w-fit">
|
|
||||||
<Button className="w-[200px] h-fit" disabled={page === 0} onClick={() => setPage((prev) => prev - 1)}>
|
|
||||||
Previous Page
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
<div className="flex items-center gap-4 w-fit">
|
|
||||||
<span className="opacity-80">
|
|
||||||
{page * size + 1} - {(page + 1) * size > list.length ? list.length : (page + 1) * size} / {list.length}
|
|
||||||
</span>
|
|
||||||
<Button className="w-[200px]" disabled={(page + 1) * size >= list.length} onClick={() => setPage((prev) => prev + 1)}>
|
|
||||||
Next Page
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
|
|
||||||
const renderMinimal = () => (
|
const render = () => (
|
||||||
<div className="flex gap-4 items-center">
|
<div className="w-full flex gap-2 justify-between items-center">
|
||||||
<div className="flex gap-2 items-center">
|
<div className="flex items-center gap-4 w-fit">
|
||||||
<button disabled={page === 0} onClick={() => setPage(0)} className="disabled:opacity-60 disabled:cursor-not-allowed">
|
<Button
|
||||||
<BsChevronDoubleLeft />
|
className="w-[200px] h-fit"
|
||||||
</button>
|
disabled={page === 0}
|
||||||
<button disabled={page === 0} onClick={() => setPage((prev) => prev - 1)} className="disabled:opacity-60 disabled:cursor-not-allowed">
|
onClick={() => setPage((prev) => prev - 1)}
|
||||||
<BsChevronLeft />
|
>
|
||||||
</button>
|
Previous Page
|
||||||
</div>
|
</Button>
|
||||||
<span className="opacity-80 w-32 text-center">
|
</div>
|
||||||
{page * size + 1} - {(page + 1) * size > list.length ? list.length : (page + 1) * size} / {list.length}
|
<div className="flex items-center gap-4 w-fit">
|
||||||
</span>
|
<div className="flex flex-row items-center gap-1 w-56">
|
||||||
<div className="flex gap-2 items-center">
|
<Select
|
||||||
<button
|
value={{
|
||||||
disabled={(page + 1) * size >= list.length}
|
value: itemsPerPage.toString(),
|
||||||
onClick={() => setPage((prev) => prev + 1)}
|
label: itemsPerPage.toString(),
|
||||||
className="disabled:opacity-60 disabled:cursor-not-allowed">
|
}}
|
||||||
<BsChevronRight />
|
onChange={(value) =>
|
||||||
</button>
|
setItemsPerPage(parseInt(value!.value ?? "25"))
|
||||||
<button
|
}
|
||||||
disabled={(page + 1) * size >= list.length}
|
options={itemsPerPageOptions.map((size) => ({
|
||||||
onClick={() => setPage(Math.floor(list.length / size))}
|
label: size.toString(),
|
||||||
className="disabled:opacity-60 disabled:cursor-not-allowed">
|
value: size.toString(),
|
||||||
<BsChevronDoubleRight />
|
}))}
|
||||||
</button>
|
isClearable={false}
|
||||||
</div>
|
styles={{
|
||||||
</div>
|
control: (styles) => ({ ...styles, width: "100px" }),
|
||||||
);
|
container: (styles) => ({ ...styles, width: "100px" }),
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<span className="opacity-80 w-32 text-center">
|
||||||
|
{page * itemsPerPage + 1} -{" "}
|
||||||
|
{itemsPerPage * (page + 1) > list.length
|
||||||
|
? list.length
|
||||||
|
: itemsPerPage * (page + 1)}
|
||||||
|
{list.length}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<Button
|
||||||
|
className="w-[200px]"
|
||||||
|
disabled={(page + 1) * itemsPerPage >= list.length}
|
||||||
|
onClick={() => setPage((prev) => prev + 1)}
|
||||||
|
>
|
||||||
|
Next Page
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
return {page, items, setPage, render, renderMinimal};
|
const renderMinimal = () => (
|
||||||
|
<div className="flex gap-4 items-center">
|
||||||
|
<div className="flex gap-2 items-center">
|
||||||
|
<button
|
||||||
|
disabled={page === 0}
|
||||||
|
onClick={() => setPage(0)}
|
||||||
|
className="disabled:opacity-60 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
<BsChevronDoubleLeft />
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
disabled={page === 0}
|
||||||
|
onClick={() => setPage((prev) => prev - 1)}
|
||||||
|
className="disabled:opacity-60 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
<BsChevronLeft />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-row items-center gap-1 w-56">
|
||||||
|
<Select
|
||||||
|
value={{
|
||||||
|
value: itemsPerPage.toString(),
|
||||||
|
label: itemsPerPage.toString(),
|
||||||
|
}}
|
||||||
|
onChange={(value) => setItemsPerPage(parseInt(value!.value ?? "25"))}
|
||||||
|
options={itemsPerPageOptions.map((size) => ({
|
||||||
|
label: size.toString(),
|
||||||
|
value: size.toString(),
|
||||||
|
}))}
|
||||||
|
isClearable={false}
|
||||||
|
styles={{
|
||||||
|
control: (styles) => ({ ...styles, width: "100px" }),
|
||||||
|
container: (styles) => ({ ...styles, width: "100px" }),
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<span className="opacity-80 w-32 text-center">
|
||||||
|
{page * itemsPerPage + 1} -{" "}
|
||||||
|
{itemsPerPage * (page + 1) > list.length
|
||||||
|
? list.length
|
||||||
|
: itemsPerPage * (page + 1)}
|
||||||
|
/ {list.length}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className="flex gap-2 items-center">
|
||||||
|
<button
|
||||||
|
disabled={(page + 1) * itemsPerPage >= list.length}
|
||||||
|
onClick={() => setPage((prev) => prev + 1)}
|
||||||
|
className="disabled:opacity-60 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
<BsChevronRight />
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
disabled={(page + 1) * itemsPerPage >= list.length}
|
||||||
|
onClick={() => setPage(Math.floor(list.length / itemsPerPage))}
|
||||||
|
className="disabled:opacity-60 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
<BsChevronDoubleRight />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
|
||||||
|
return { page, items, setPage, render, renderMinimal };
|
||||||
}
|
}
|
||||||
|
|||||||
42
src/hooks/useStats.tsx
Normal file
42
src/hooks/useStats.tsx
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
import axios from "axios";
|
||||||
|
import { useCallback, useEffect, useState } from "react";
|
||||||
|
|
||||||
|
export default function useStats<T extends any>(
|
||||||
|
id?: string,
|
||||||
|
shouldNotQuery: boolean = !id,
|
||||||
|
queryType: string = "stats"
|
||||||
|
) {
|
||||||
|
type ElementType = T extends (infer U)[] ? U : never;
|
||||||
|
|
||||||
|
const [data, setData] = useState<T>({} as unknown as T);
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
const [isError, setIsError] = useState(false);
|
||||||
|
|
||||||
|
const getData = useCallback(() => {
|
||||||
|
if (shouldNotQuery) return;
|
||||||
|
|
||||||
|
setIsLoading(true);
|
||||||
|
setIsError(false);
|
||||||
|
let endpoint = `/api/stats/user/${id}`;
|
||||||
|
if (queryType) endpoint += `?query=${queryType}`;
|
||||||
|
axios
|
||||||
|
.get<T>(endpoint)
|
||||||
|
.then((response) => {
|
||||||
|
console.log(response.data);
|
||||||
|
setData(response.data);
|
||||||
|
})
|
||||||
|
.catch(() => setIsError(true))
|
||||||
|
.finally(() => setIsLoading(false));
|
||||||
|
}, [id, shouldNotQuery, queryType]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
getData();
|
||||||
|
}, [getData]);
|
||||||
|
|
||||||
|
return {
|
||||||
|
data,
|
||||||
|
reload: getData,
|
||||||
|
isLoading,
|
||||||
|
isError,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -1,26 +1,28 @@
|
|||||||
import { useState, useEffect } from "react";
|
import { useState, useEffect, useCallback } from "react";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
|
|
||||||
const useTicketsListener = (userId?: string) => {
|
const useTicketsListener = (userId?: string, canFetch?: boolean) => {
|
||||||
const [assignedTickets, setAssignedTickets] = useState([]);
|
const [assignedTickets, setAssignedTickets] = useState([]);
|
||||||
|
|
||||||
const getData = () => {
|
const getData = useCallback(() => {
|
||||||
axios
|
axios
|
||||||
.get("/api/tickets/assignedToUser")
|
.get("/api/tickets/assignedToUser")
|
||||||
.then((response) => setAssignedTickets(response.data));
|
.then((response) => setAssignedTickets(response.data));
|
||||||
};
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
getData();
|
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
|
if (!canFetch) return;
|
||||||
|
getData();
|
||||||
|
}, [canFetch, getData]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (!canFetch) return;
|
||||||
const intervalId = setInterval(() => {
|
const intervalId = setInterval(() => {
|
||||||
getData();
|
getData();
|
||||||
}, 60 * 1000);
|
}, 60 * 1000);
|
||||||
|
|
||||||
return () => clearInterval(intervalId);
|
return () => clearInterval(intervalId);
|
||||||
}, [assignedTickets]);
|
}, [assignedTickets, canFetch, getData]);
|
||||||
|
|
||||||
if (userId) {
|
if (userId) {
|
||||||
return {
|
return {
|
||||||
|
|||||||
72
src/interfaces/approval.workflow.ts
Normal file
72
src/interfaces/approval.workflow.ts
Normal file
@@ -0,0 +1,72 @@
|
|||||||
|
import { ObjectId } from "mongodb";
|
||||||
|
import { Module } from ".";
|
||||||
|
import { Type, User, userTypeLabels, userTypeLabelsShort } from "./user";
|
||||||
|
|
||||||
|
export interface ApprovalWorkflow {
|
||||||
|
_id?: ObjectId,
|
||||||
|
name: string,
|
||||||
|
entityId: string,
|
||||||
|
requester: User["id"],
|
||||||
|
startDate: number,
|
||||||
|
modules: Module[],
|
||||||
|
examId?: string,
|
||||||
|
status: ApprovalWorkflowStatus,
|
||||||
|
steps: WorkflowStep[],
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface EditableApprovalWorkflow extends Omit<ApprovalWorkflow, "_id" | "steps"> {
|
||||||
|
id: string,
|
||||||
|
steps: EditableWorkflowStep[],
|
||||||
|
}
|
||||||
|
|
||||||
|
export type StepType = "form-intake" | "approval-by";
|
||||||
|
export const StepTypeLabel: Record<StepType, string> = {
|
||||||
|
"form-intake": "Form Intake",
|
||||||
|
"approval-by": "Approval",
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface WorkflowStep {
|
||||||
|
stepType: StepType,
|
||||||
|
stepNumber: number,
|
||||||
|
completed: boolean,
|
||||||
|
rejected?: boolean,
|
||||||
|
completedBy?: User["id"],
|
||||||
|
completedDate?: number,
|
||||||
|
assignees: (User["id"])[];
|
||||||
|
firstStep?: boolean,
|
||||||
|
finalStep?: boolean,
|
||||||
|
selected?: boolean,
|
||||||
|
comments?: string,
|
||||||
|
examChanges?: string[],
|
||||||
|
onClick?: React.MouseEventHandler<HTMLDivElement>
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface EditableWorkflowStep {
|
||||||
|
key: number,
|
||||||
|
stepType: StepType,
|
||||||
|
stepNumber: number,
|
||||||
|
completed: boolean,
|
||||||
|
rejected?: boolean,
|
||||||
|
completedBy?: User["id"],
|
||||||
|
completedDate?: number,
|
||||||
|
assignees: (User["id"] | null | undefined)[]; // bit of an hack, but allowing null or undefined values allows us to match one to one the select input components with the assignees array. And since select inputs allow undefined or null values, it is allowed here too, but must validate required input before form submission
|
||||||
|
firstStep: boolean,
|
||||||
|
finalStep?: boolean,
|
||||||
|
onDelete?: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getUserTypeLabel(type: Type | undefined): string {
|
||||||
|
if (type) return userTypeLabels[type];
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
export function getUserTypeLabelShort(type: Type | undefined): string {
|
||||||
|
if (type) return userTypeLabelsShort[type];
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
export type ApprovalWorkflowStatus = "approved" | "pending" | "rejected";
|
||||||
|
export const ApprovalWorkflowStatusLabel: Record<ApprovalWorkflowStatus, string> = {
|
||||||
|
approved: "Approved",
|
||||||
|
pending: "Pending",
|
||||||
|
rejected: "Rejected",
|
||||||
|
};
|
||||||
@@ -1,4 +1,3 @@
|
|||||||
import instructions from "@/pages/api/exam/media/instructions";
|
|
||||||
import { Module } from ".";
|
import { Module } from ".";
|
||||||
|
|
||||||
export type Exam = ReadingExam | ListeningExam | WritingExam | SpeakingExam | LevelExam;
|
export type Exam = ReadingExam | ListeningExam | WritingExam | SpeakingExam | LevelExam;
|
||||||
@@ -10,6 +9,9 @@ export type Difficulty = BasicDifficulty | CEFRLevels;
|
|||||||
// Left easy, medium and hard to support older exam versions
|
// Left easy, medium and hard to support older exam versions
|
||||||
export type BasicDifficulty = "easy" | "medium" | "hard";
|
export type BasicDifficulty = "easy" | "medium" | "hard";
|
||||||
export type CEFRLevels = "A1" | "A2" | "B1" | "B2" | "C1" | "C2";
|
export type CEFRLevels = "A1" | "A2" | "B1" | "B2" | "C1" | "C2";
|
||||||
|
export const ACCESSTYPE = ["public", "private", "confidential"] as const;
|
||||||
|
export type AccessType = typeof ACCESSTYPE[number];
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
export interface ExamBase {
|
export interface ExamBase {
|
||||||
@@ -24,8 +26,10 @@ export interface ExamBase {
|
|||||||
shuffle?: boolean;
|
shuffle?: boolean;
|
||||||
createdBy?: string; // option as it has been added later
|
createdBy?: string; // option as it has been added later
|
||||||
createdAt?: string; // option as it has been added later
|
createdAt?: string; // option as it has been added later
|
||||||
private?: boolean;
|
access: AccessType;
|
||||||
label?: string;
|
label?: string;
|
||||||
|
requiresApproval?: boolean;
|
||||||
|
approved?: boolean;
|
||||||
}
|
}
|
||||||
export interface ReadingExam extends ExamBase {
|
export interface ReadingExam extends ExamBase {
|
||||||
module: "reading";
|
module: "reading";
|
||||||
@@ -238,6 +242,7 @@ export interface InteractiveSpeakingExercise extends Section {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface FillBlanksMCOption {
|
export interface FillBlanksMCOption {
|
||||||
|
uuid: string; // added later to fulfill the need for an immutable identifier.
|
||||||
id: string;
|
id: string;
|
||||||
options: {
|
options: {
|
||||||
A: string;
|
A: string;
|
||||||
@@ -255,6 +260,7 @@ export interface FillBlanksExercise {
|
|||||||
text: string; // *EXAMPLE: "They tried to {{1}} burning"
|
text: string; // *EXAMPLE: "They tried to {{1}} burning"
|
||||||
allowRepetition?: boolean;
|
allowRepetition?: boolean;
|
||||||
solutions: {
|
solutions: {
|
||||||
|
uuid: string; // added later to fulfill the need for an immutable identifier.
|
||||||
id: string; // *EXAMPLE: "1"
|
id: string; // *EXAMPLE: "1"
|
||||||
solution: string; // *EXAMPLE: "preserve"
|
solution: string; // *EXAMPLE: "preserve"
|
||||||
}[];
|
}[];
|
||||||
@@ -278,6 +284,7 @@ export interface TrueFalseExercise {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface TrueFalseQuestion {
|
export interface TrueFalseQuestion {
|
||||||
|
uuid: string; // added later to fulfill the need for an immutable identifier.
|
||||||
id: string; // *EXAMPLE: "1"
|
id: string; // *EXAMPLE: "1"
|
||||||
prompt: string; // *EXAMPLE: "What does her briefcase look like?"
|
prompt: string; // *EXAMPLE: "What does her briefcase look like?"
|
||||||
solution: "true" | "false" | "not_given" | undefined; // *EXAMPLE: "True"
|
solution: "true" | "false" | "not_given" | undefined; // *EXAMPLE: "True"
|
||||||
@@ -290,6 +297,7 @@ export interface WriteBlanksExercise {
|
|||||||
id: string;
|
id: string;
|
||||||
text: string; // *EXAMPLE: "The Government plans to give ${{14}}"
|
text: string; // *EXAMPLE: "The Government plans to give ${{14}}"
|
||||||
solutions: {
|
solutions: {
|
||||||
|
uuid: string; // added later to fulfill the need for an immutable identifier.
|
||||||
id: string; // *EXAMPLE: "14"
|
id: string; // *EXAMPLE: "14"
|
||||||
solution: string[]; // *EXAMPLE: ["Prescott"] - All possible solutions (case sensitive)
|
solution: string[]; // *EXAMPLE: ["Prescott"] - All possible solutions (case sensitive)
|
||||||
}[];
|
}[];
|
||||||
@@ -316,12 +324,14 @@ export interface MatchSentencesExercise {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface MatchSentenceExerciseSentence {
|
export interface MatchSentenceExerciseSentence {
|
||||||
|
uuid: string; // added later to fulfill the need for an immutable identifier.
|
||||||
id: string;
|
id: string;
|
||||||
sentence: string;
|
sentence: string;
|
||||||
solution: string;
|
solution: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface MatchSentenceExerciseOption {
|
export interface MatchSentenceExerciseOption {
|
||||||
|
uuid: string; // added later to fulfill the need for an immutable identifier.
|
||||||
id: string;
|
id: string;
|
||||||
sentence: string;
|
sentence: string;
|
||||||
}
|
}
|
||||||
@@ -343,6 +353,7 @@ export interface MultipleChoiceExercise {
|
|||||||
|
|
||||||
export interface MultipleChoiceQuestion {
|
export interface MultipleChoiceQuestion {
|
||||||
variant: "image" | "text";
|
variant: "image" | "text";
|
||||||
|
uuid: string; // added later to fulfill the need for an immutable identifier.
|
||||||
id: string; // *EXAMPLE: "1"
|
id: string; // *EXAMPLE: "1"
|
||||||
prompt: string; // *EXAMPLE: "What does her briefcase look like?"
|
prompt: string; // *EXAMPLE: "What does her briefcase look like?"
|
||||||
solution: string; // *EXAMPLE: "A"
|
solution: string; // *EXAMPLE: "A"
|
||||||
|
|||||||
@@ -1,4 +1,11 @@
|
|||||||
export type Module = "reading" | "listening" | "writing" | "speaking" | "level";
|
export type Module = "reading" | "listening" | "writing" | "speaking" | "level";
|
||||||
|
export const ModuleTypeLabels: Record<Module, string> = {
|
||||||
|
reading: "Reading",
|
||||||
|
listening: "Listening",
|
||||||
|
writing: "Writing",
|
||||||
|
speaking: "Speaking",
|
||||||
|
level: "Level",
|
||||||
|
};
|
||||||
|
|
||||||
export interface Step {
|
export interface Step {
|
||||||
min: number;
|
min: number;
|
||||||
|
|||||||
@@ -37,3 +37,5 @@ export interface Assignment {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export type AssignmentWithCorporateId = Assignment & { corporateId: string };
|
export type AssignmentWithCorporateId = Assignment & { corporateId: string };
|
||||||
|
|
||||||
|
export type AssignmentWithHasResults = Assignment & { hasResults: boolean };
|
||||||
|
|||||||
@@ -170,4 +170,24 @@ export interface Code {
|
|||||||
export type Type = "student" | "teacher" | "corporate" | "admin" | "developer" | "agent" | "mastercorporate";
|
export type Type = "student" | "teacher" | "corporate" | "admin" | "developer" | "agent" | "mastercorporate";
|
||||||
export const userTypes: Type[] = ["student", "teacher", "corporate", "admin", "developer", "agent", "mastercorporate"];
|
export const userTypes: Type[] = ["student", "teacher", "corporate", "admin", "developer", "agent", "mastercorporate"];
|
||||||
|
|
||||||
|
export const userTypeLabels: Record<Type, string> = {
|
||||||
|
student: "Student",
|
||||||
|
teacher: "Teacher",
|
||||||
|
corporate: "Corporate",
|
||||||
|
admin: "Admin",
|
||||||
|
developer: "Developer",
|
||||||
|
agent: "Agent",
|
||||||
|
mastercorporate: "Master Corporate",
|
||||||
|
};
|
||||||
|
|
||||||
|
export const userTypeLabelsShort: Record<Type, string> = {
|
||||||
|
student: "",
|
||||||
|
teacher: "Prof.",
|
||||||
|
corporate: "Dir.",
|
||||||
|
admin: "Admin",
|
||||||
|
developer: "Dev.",
|
||||||
|
agent: "Agent",
|
||||||
|
mastercorporate: "Dir.",
|
||||||
|
};
|
||||||
|
|
||||||
export type WithUser<T> = T extends { participants: string[] } ? Omit<T, "participants"> & { participants: User[] } : T;
|
export type WithUser<T> = T extends { participants: string[] } ? Omit<T, "participants"> & { participants: User[] } : T;
|
||||||
|
|||||||
84
src/lib/createWorkflowsOnExamCreation.ts
Normal file
84
src/lib/createWorkflowsOnExamCreation.ts
Normal file
@@ -0,0 +1,84 @@
|
|||||||
|
import { Module } from "@/interfaces";
|
||||||
|
import { getApprovalWorkflowByFormIntaker, createApprovalWorkflow } from "@/utils/approval.workflows.be";
|
||||||
|
import client from "@/lib/mongodb";
|
||||||
|
|
||||||
|
const db = client.db(process.env.MONGODB_DB);
|
||||||
|
|
||||||
|
/* export async function createApprovalWorkflowsOnExamCreation(examAuthor: string, examEntities: string[], examId: string, examModule: string) {
|
||||||
|
const results = await Promise.all(
|
||||||
|
examEntities.map(async (entity) => {
|
||||||
|
const configuredWorkflow = await getApprovalWorkflowByFormIntaker(entity, examAuthor);
|
||||||
|
if (!configuredWorkflow) {
|
||||||
|
return { entity, created: false };
|
||||||
|
}
|
||||||
|
|
||||||
|
configuredWorkflow.modules.push(examModule as Module);
|
||||||
|
configuredWorkflow.name = examId;
|
||||||
|
configuredWorkflow.examId = examId;
|
||||||
|
configuredWorkflow.entityId = entity;
|
||||||
|
configuredWorkflow.startDate = Date.now();
|
||||||
|
|
||||||
|
try {
|
||||||
|
await createApprovalWorkflow("active-workflows", configuredWorkflow);
|
||||||
|
return { entity, created: true };
|
||||||
|
} catch (error: any) {
|
||||||
|
return { entity, created: false };
|
||||||
|
}
|
||||||
|
})
|
||||||
|
);
|
||||||
|
|
||||||
|
const successCount = results.filter((r) => r.created).length;
|
||||||
|
const totalCount = examEntities.length;
|
||||||
|
|
||||||
|
return {
|
||||||
|
successCount,
|
||||||
|
totalCount,
|
||||||
|
};
|
||||||
|
} */
|
||||||
|
|
||||||
|
// TEMPORARY BEHAVIOUR! ONLY THE FIRST CONFIGURED WORKFLOW FOUND IS STARTED
|
||||||
|
export async function createApprovalWorkflowOnExamCreation(examAuthor: string, examEntities: string[], examId: string, examModule: string) {
|
||||||
|
let successCount = 0;
|
||||||
|
let totalCount = 0;
|
||||||
|
|
||||||
|
for (const entity of examEntities) {
|
||||||
|
const configuredWorkflow = await getApprovalWorkflowByFormIntaker(entity, examAuthor);
|
||||||
|
|
||||||
|
if (!configuredWorkflow) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
totalCount = 1; // a workflow was found
|
||||||
|
|
||||||
|
configuredWorkflow.modules.push(examModule as Module);
|
||||||
|
configuredWorkflow.name = examId;
|
||||||
|
configuredWorkflow.examId = examId;
|
||||||
|
configuredWorkflow.entityId = entity;
|
||||||
|
configuredWorkflow.startDate = Date.now();
|
||||||
|
configuredWorkflow.steps[0].completed = true;
|
||||||
|
configuredWorkflow.steps[0].completedBy = examAuthor;
|
||||||
|
configuredWorkflow.steps[0].completedDate = Date.now();
|
||||||
|
|
||||||
|
try {
|
||||||
|
await createApprovalWorkflow("active-workflows", configuredWorkflow);
|
||||||
|
successCount = 1;
|
||||||
|
break; // Stop after the first success
|
||||||
|
} catch (error: any) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// commented because they asked for every exam to stay confidential
|
||||||
|
/* if (totalCount === 0) { // current behaviour: if no workflow was found skip approval process
|
||||||
|
await db.collection(examModule).updateOne(
|
||||||
|
{ id: examId },
|
||||||
|
{ $set: { id: examId, access: "private" }},
|
||||||
|
{ upsert: true }
|
||||||
|
);
|
||||||
|
} */
|
||||||
|
|
||||||
|
return {
|
||||||
|
successCount,
|
||||||
|
totalCount,
|
||||||
|
};
|
||||||
|
}
|
||||||
@@ -5,7 +5,9 @@ if (!process.env.MONGODB_URI) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const uri = process.env.MONGODB_URI || "";
|
const uri = process.env.MONGODB_URI || "";
|
||||||
const options = {};
|
const options = {
|
||||||
|
maxPoolSize: 10,
|
||||||
|
};
|
||||||
|
|
||||||
let client: MongoClient;
|
let client: MongoClient;
|
||||||
|
|
||||||
|
|||||||
@@ -1,7 +1,5 @@
|
|||||||
import Button from "@/components/Low/Button";
|
import Button from "@/components/Low/Button";
|
||||||
import Checkbox from "@/components/Low/Checkbox";
|
import Checkbox from "@/components/Low/Checkbox";
|
||||||
import { PERMISSIONS } from "@/constants/userPermissions";
|
|
||||||
import useUsers from "@/hooks/useUsers";
|
|
||||||
import { Type, User } from "@/interfaces/user";
|
import { Type, User } from "@/interfaces/user";
|
||||||
import { USER_TYPE_LABELS } from "@/resources/user";
|
import { USER_TYPE_LABELS } from "@/resources/user";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
@@ -15,444 +13,587 @@ import ShortUniqueId from "short-unique-id";
|
|||||||
import { useFilePicker } from "use-file-picker";
|
import { useFilePicker } from "use-file-picker";
|
||||||
import readXlsxFile from "read-excel-file";
|
import readXlsxFile from "read-excel-file";
|
||||||
import Modal from "@/components/Modal";
|
import Modal from "@/components/Modal";
|
||||||
import { BsFileEarmarkEaselFill, BsQuestionCircleFill } from "react-icons/bs";
|
|
||||||
import { checkAccess, getTypesOfUser } from "@/utils/permissions";
|
import { checkAccess, getTypesOfUser } from "@/utils/permissions";
|
||||||
import { PermissionType } from "@/interfaces/permissions";
|
import { PermissionType } from "@/interfaces/permissions";
|
||||||
import usePermissions from "@/hooks/usePermissions";
|
|
||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
import Select from "@/components/Low/Select";
|
import Select from "@/components/Low/Select";
|
||||||
import CodeGenImportSummary, { ExcelCodegenDuplicatesMap } from "@/components/ImportSummaries/Codegen";
|
import CodeGenImportSummary, {
|
||||||
|
ExcelCodegenDuplicatesMap,
|
||||||
|
} from "@/components/ImportSummaries/Codegen";
|
||||||
import { FaFileDownload } from "react-icons/fa";
|
import { FaFileDownload } from "react-icons/fa";
|
||||||
import { IoInformationCircleOutline } from "react-icons/io5";
|
import { IoInformationCircleOutline } from "react-icons/io5";
|
||||||
import { HiOutlineDocumentText } from "react-icons/hi";
|
import { HiOutlineDocumentText } from "react-icons/hi";
|
||||||
import CodegenTable from "@/components/Tables/CodeGenTable";
|
import CodegenTable from "@/components/Tables/CodeGenTable";
|
||||||
|
|
||||||
const EMAIL_REGEX = new RegExp(/^[a-zA-Z0-9]+(?:\.[a-zA-Z0-9]+)*@[a-zA-Z0-9]+(?:\.[a-zA-Z0-9]+)*$/);
|
const EMAIL_REGEX = new RegExp(
|
||||||
|
/^[a-zA-Z0-9]+(?:\.[a-zA-Z0-9]+)*@[a-zA-Z0-9]+(?:\.[a-zA-Z0-9]+)*$/
|
||||||
|
);
|
||||||
|
|
||||||
const USER_TYPE_PERMISSIONS: {
|
const USER_TYPE_PERMISSIONS: {
|
||||||
[key in Type]: { perm: PermissionType | undefined; list: Type[] };
|
[key in Type]: { perm: PermissionType | undefined; list: Type[] };
|
||||||
} = {
|
} = {
|
||||||
student: {
|
student: {
|
||||||
perm: "createCodeStudent",
|
perm: "createCodeStudent",
|
||||||
list: [],
|
list: [],
|
||||||
},
|
},
|
||||||
teacher: {
|
teacher: {
|
||||||
perm: "createCodeTeacher",
|
perm: "createCodeTeacher",
|
||||||
list: [],
|
list: [],
|
||||||
},
|
},
|
||||||
agent: {
|
agent: {
|
||||||
perm: "createCodeCountryManager",
|
perm: "createCodeCountryManager",
|
||||||
list: ["student", "teacher", "corporate", "mastercorporate"],
|
list: ["student", "teacher", "corporate", "mastercorporate"],
|
||||||
},
|
},
|
||||||
corporate: {
|
corporate: {
|
||||||
perm: "createCodeCorporate",
|
perm: "createCodeCorporate",
|
||||||
list: ["student", "teacher"],
|
list: ["student", "teacher"],
|
||||||
},
|
},
|
||||||
mastercorporate: {
|
mastercorporate: {
|
||||||
perm: undefined,
|
perm: undefined,
|
||||||
list: ["student", "teacher", "corporate"],
|
list: ["student", "teacher", "corporate"],
|
||||||
},
|
},
|
||||||
admin: {
|
admin: {
|
||||||
perm: "createCodeAdmin",
|
perm: "createCodeAdmin",
|
||||||
list: ["student", "teacher", "agent", "corporate", "admin", "mastercorporate"],
|
list: [
|
||||||
},
|
"student",
|
||||||
developer: {
|
"teacher",
|
||||||
perm: undefined,
|
"agent",
|
||||||
list: ["student", "teacher", "agent", "corporate", "admin", "developer", "mastercorporate"],
|
"corporate",
|
||||||
},
|
"admin",
|
||||||
|
"mastercorporate",
|
||||||
|
],
|
||||||
|
},
|
||||||
|
developer: {
|
||||||
|
perm: undefined,
|
||||||
|
list: [
|
||||||
|
"student",
|
||||||
|
"teacher",
|
||||||
|
"agent",
|
||||||
|
"corporate",
|
||||||
|
"admin",
|
||||||
|
"developer",
|
||||||
|
"mastercorporate",
|
||||||
|
],
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
user: User;
|
user: User;
|
||||||
users: User[];
|
users: User[];
|
||||||
permissions: PermissionType[];
|
permissions: PermissionType[];
|
||||||
entities: EntityWithRoles[]
|
entities: EntityWithRoles[];
|
||||||
onFinish: () => void;
|
onFinish: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function BatchCodeGenerator({ user, users, entities = [], permissions, onFinish }: Props) {
|
export default function BatchCodeGenerator({
|
||||||
const [infos, setInfos] = useState<{ email: string; name: string; passport_id: string }[]>([]);
|
user,
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
users,
|
||||||
const [expiryDate, setExpiryDate] = useState<Date | null>(
|
entities = [],
|
||||||
user?.subscriptionExpirationDate ? moment(user.subscriptionExpirationDate).toDate() : null,
|
permissions,
|
||||||
);
|
onFinish,
|
||||||
const [isExpiryDateEnabled, setIsExpiryDateEnabled] = useState(true);
|
}: Props) {
|
||||||
const [type, setType] = useState<Type>("student");
|
const [infos, setInfos] = useState<
|
||||||
const [showHelp, setShowHelp] = useState(false);
|
{ email: string; name: string; passport_id: string }[]
|
||||||
const [entity, setEntity] = useState((entities || [])[0]?.id || undefined);
|
>([]);
|
||||||
const [parsedExcel, setParsedExcel] = useState<{ rows?: any[]; errors?: any[] }>({ rows: undefined, errors: undefined });
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [duplicatedRows, setDuplicatedRows] = useState<{ duplicates: ExcelCodegenDuplicatesMap, count: number }>();
|
const [expiryDate, setExpiryDate] = useState<Date | null>(
|
||||||
|
user?.subscriptionExpirationDate
|
||||||
|
? moment(user.subscriptionExpirationDate).toDate()
|
||||||
|
: null
|
||||||
|
);
|
||||||
|
const [isExpiryDateEnabled, setIsExpiryDateEnabled] = useState(true);
|
||||||
|
const [type, setType] = useState<Type>("student");
|
||||||
|
const [showHelp, setShowHelp] = useState(false);
|
||||||
|
const [entity, setEntity] = useState((entities || [])[0]?.id || undefined);
|
||||||
|
const [parsedExcel, setParsedExcel] = useState<{
|
||||||
|
rows?: any[];
|
||||||
|
errors?: any[];
|
||||||
|
}>({ rows: undefined, errors: undefined });
|
||||||
|
const [duplicatedRows, setDuplicatedRows] = useState<{
|
||||||
|
duplicates: ExcelCodegenDuplicatesMap;
|
||||||
|
count: number;
|
||||||
|
}>();
|
||||||
|
|
||||||
const { openFilePicker, filesContent, clear } = useFilePicker({
|
const { openFilePicker, filesContent, clear } = useFilePicker({
|
||||||
accept: ".xlsx",
|
accept: ".xlsx",
|
||||||
multiple: false,
|
multiple: false,
|
||||||
readAs: "ArrayBuffer",
|
readAs: "ArrayBuffer",
|
||||||
});
|
});
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!isExpiryDateEnabled) setExpiryDate(null);
|
if (!isExpiryDateEnabled) setExpiryDate(null);
|
||||||
}, [isExpiryDateEnabled]);
|
}, [isExpiryDateEnabled]);
|
||||||
|
|
||||||
const schema = {
|
const schema = {
|
||||||
'First Name': {
|
"First Name": {
|
||||||
prop: 'firstName',
|
prop: "firstName",
|
||||||
type: String,
|
type: String,
|
||||||
required: true,
|
required: true,
|
||||||
validate: (value: string) => {
|
validate: (value: string) => {
|
||||||
if (!value || value.trim() === '') {
|
if (!value || value.trim() === "") {
|
||||||
throw new Error('First Name cannot be empty')
|
throw new Error("First Name cannot be empty");
|
||||||
}
|
}
|
||||||
return true
|
return true;
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
'Last Name': {
|
"Last Name": {
|
||||||
prop: 'lastName',
|
prop: "lastName",
|
||||||
type: String,
|
type: String,
|
||||||
required: true,
|
required: true,
|
||||||
validate: (value: string) => {
|
validate: (value: string) => {
|
||||||
if (!value || value.trim() === '') {
|
if (!value || value.trim() === "") {
|
||||||
throw new Error('Last Name cannot be empty')
|
throw new Error("Last Name cannot be empty");
|
||||||
}
|
}
|
||||||
return true
|
return true;
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
'Passport/National ID': {
|
"Passport/National ID": {
|
||||||
prop: 'passport_id',
|
prop: "passport_id",
|
||||||
type: String,
|
type: String,
|
||||||
required: true,
|
required: true,
|
||||||
validate: (value: string) => {
|
validate: (value: string) => {
|
||||||
if (!value || value.trim() === '') {
|
if (!value || value.trim() === "") {
|
||||||
throw new Error('Passport/National ID cannot be empty')
|
throw new Error("Passport/National ID cannot be empty");
|
||||||
}
|
}
|
||||||
return true
|
return true;
|
||||||
}
|
},
|
||||||
},
|
},
|
||||||
'E-mail': {
|
"E-mail": {
|
||||||
prop: 'email',
|
prop: "email",
|
||||||
required: true,
|
required: true,
|
||||||
type: (value: any) => {
|
type: (value: any) => {
|
||||||
if (!value || value.trim() === '') {
|
if (!value || value.trim() === "") {
|
||||||
throw new Error('Email cannot be empty')
|
throw new Error("Email cannot be empty");
|
||||||
}
|
}
|
||||||
if (!EMAIL_REGEX.test(value.trim())) {
|
if (!EMAIL_REGEX.test(value.trim())) {
|
||||||
throw new Error('Invalid Email')
|
throw new Error("Invalid Email");
|
||||||
}
|
}
|
||||||
return value
|
return value;
|
||||||
}
|
},
|
||||||
}
|
},
|
||||||
}
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (filesContent.length > 0) {
|
if (filesContent.length > 0) {
|
||||||
const file = filesContent[0];
|
const file = filesContent[0];
|
||||||
readXlsxFile(
|
readXlsxFile(file.content, { schema, ignoreEmptyRows: false }).then(
|
||||||
file.content, { schema, ignoreEmptyRows: false })
|
(data) => {
|
||||||
.then((data) => {
|
setParsedExcel(data);
|
||||||
setParsedExcel(data)
|
}
|
||||||
});
|
);
|
||||||
}
|
}
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
}, [filesContent]);
|
}, [filesContent]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (parsedExcel.rows) {
|
if (parsedExcel.rows) {
|
||||||
const duplicates: ExcelCodegenDuplicatesMap = {
|
const duplicates: ExcelCodegenDuplicatesMap = {
|
||||||
email: new Map(),
|
email: new Map(),
|
||||||
passport_id: new Map(),
|
passport_id: new Map(),
|
||||||
};
|
};
|
||||||
const duplicateValues = new Set<string>();
|
const duplicateValues = new Set<string>();
|
||||||
const duplicateRowIndices = new Set<number>();
|
const duplicateRowIndices = new Set<number>();
|
||||||
|
|
||||||
const errorRowIndices = new Set(
|
const errorRowIndices = new Set(
|
||||||
parsedExcel.errors?.map(error => error.row) || []
|
parsedExcel.errors?.map((error) => error.row) || []
|
||||||
);
|
);
|
||||||
|
|
||||||
parsedExcel.rows.forEach((row, index) => {
|
parsedExcel.rows.forEach((row, index) => {
|
||||||
if (!errorRowIndices.has(index + 2)) {
|
if (!errorRowIndices.has(index + 2)) {
|
||||||
(Object.keys(duplicates) as Array<keyof ExcelCodegenDuplicatesMap>).forEach(field => {
|
(
|
||||||
if (row !== null) {
|
Object.keys(duplicates) as Array<keyof ExcelCodegenDuplicatesMap>
|
||||||
const value = row[field];
|
).forEach((field) => {
|
||||||
if (value) {
|
if (row !== null) {
|
||||||
if (!duplicates[field].has(value)) {
|
const value = row[field];
|
||||||
duplicates[field].set(value, [index + 2]);
|
if (value) {
|
||||||
} else {
|
if (!duplicates[field].has(value)) {
|
||||||
const existingRows = duplicates[field].get(value);
|
duplicates[field].set(value, [index + 2]);
|
||||||
if (existingRows) {
|
} else {
|
||||||
existingRows.push(index + 2);
|
const existingRows = duplicates[field].get(value);
|
||||||
duplicateValues.add(value);
|
if (existingRows) {
|
||||||
existingRows.forEach(rowNum => duplicateRowIndices.add(rowNum));
|
existingRows.push(index + 2);
|
||||||
}
|
duplicateValues.add(value);
|
||||||
}
|
existingRows.forEach((rowNum) =>
|
||||||
}
|
duplicateRowIndices.add(rowNum)
|
||||||
}
|
);
|
||||||
});
|
}
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
const info = parsedExcel.rows
|
const info = parsedExcel.rows
|
||||||
.map((row, index) => {
|
.map((row, index) => {
|
||||||
if (errorRowIndices.has(index + 2) || duplicateRowIndices.has(index + 2) || row === null) {
|
if (
|
||||||
return undefined;
|
errorRowIndices.has(index + 2) ||
|
||||||
}
|
duplicateRowIndices.has(index + 2) ||
|
||||||
const { firstName, lastName, studentID, passport_id, email, phone, group, country } = row;
|
row === null
|
||||||
if (!email || !EMAIL_REGEX.test(email.toString().trim())) {
|
) {
|
||||||
return undefined;
|
return undefined;
|
||||||
}
|
}
|
||||||
|
const {
|
||||||
|
firstName,
|
||||||
|
lastName,
|
||||||
|
studentID,
|
||||||
|
passport_id,
|
||||||
|
email,
|
||||||
|
phone,
|
||||||
|
group,
|
||||||
|
country,
|
||||||
|
} = row;
|
||||||
|
if (!email || !EMAIL_REGEX.test(email.toString().trim())) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
email: email.toString().trim().toLowerCase(),
|
email: email.toString().trim().toLowerCase(),
|
||||||
name: `${firstName ?? ""} ${lastName ?? ""}`.trim(),
|
name: `${firstName ?? ""} ${lastName ?? ""}`.trim(),
|
||||||
passport_id: passport_id?.toString().trim() || undefined,
|
passport_id: passport_id?.toString().trim() || undefined,
|
||||||
};
|
};
|
||||||
}).filter((x) => !!x) as typeof infos;
|
})
|
||||||
|
.filter((x) => !!x) as typeof infos;
|
||||||
|
|
||||||
setInfos(info);
|
setInfos(info);
|
||||||
}
|
}
|
||||||
}, [entity, parsedExcel, type]);
|
}, [entity, parsedExcel, type]);
|
||||||
|
|
||||||
const generateAndInvite = async () => {
|
const generateAndInvite = async () => {
|
||||||
const newUsers = infos.filter((x) => !users.map((u) => u.email).includes(x.email));
|
const newUsers = infos.filter(
|
||||||
const existingUsers = infos
|
(x) => !users.map((u) => u.email).includes(x.email)
|
||||||
.filter((x) => users.map((u) => u.email).includes(x.email))
|
);
|
||||||
.map((i) => users.find((u) => u.email === i.email))
|
const existingUsers = infos
|
||||||
.filter((x) => !!x && x.type === "student") as User[];
|
.filter((x) => users.map((u) => u.email).includes(x.email))
|
||||||
|
.map((i) => users.find((u) => u.email === i.email))
|
||||||
|
.filter((x) => !!x && x.type === "student") as User[];
|
||||||
|
|
||||||
const newUsersSentence = newUsers.length > 0 ? `generate ${newUsers.length} code(s)` : undefined;
|
const newUsersSentence =
|
||||||
const existingUsersSentence = existingUsers.length > 0 ? `invite ${existingUsers.length} registered student(s)` : undefined;
|
newUsers.length > 0 ? `generate ${newUsers.length} code(s)` : undefined;
|
||||||
if (
|
const existingUsersSentence =
|
||||||
!confirm(
|
existingUsers.length > 0
|
||||||
`You are about to ${[newUsersSentence, existingUsersSentence].filter((x) => !!x).join(" and ")}, are you sure you want to continue?`,
|
? `invite ${existingUsers.length} registered student(s)`
|
||||||
)
|
: undefined;
|
||||||
)
|
if (
|
||||||
return;
|
!confirm(
|
||||||
|
`You are about to ${[newUsersSentence, existingUsersSentence]
|
||||||
|
.filter((x) => !!x)
|
||||||
|
.join(" and ")}, are you sure you want to continue?`
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return;
|
||||||
|
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
Promise.all(existingUsers.map(async (u) => await axios.post(`/api/invites`, { to: u.id, from: user.id })))
|
Promise.all(
|
||||||
.then(() => toast.success(`Successfully invited ${existingUsers.length} registered student(s)!`))
|
existingUsers.map(
|
||||||
.finally(() => {
|
async (u) =>
|
||||||
if (newUsers.length === 0) setIsLoading(false);
|
await axios.post(`/api/invites`, { to: u.id, from: user.id })
|
||||||
});
|
)
|
||||||
|
)
|
||||||
|
.then(() =>
|
||||||
|
toast.success(
|
||||||
|
`Successfully invited ${existingUsers.length} registered student(s)!`
|
||||||
|
)
|
||||||
|
)
|
||||||
|
.finally(() => {
|
||||||
|
if (newUsers.length === 0) setIsLoading(false);
|
||||||
|
});
|
||||||
|
|
||||||
if (newUsers.length > 0) generateCode(type, newUsers);
|
if (newUsers.length > 0) generateCode(type, newUsers);
|
||||||
setInfos([]);
|
setInfos([]);
|
||||||
};
|
};
|
||||||
|
|
||||||
const generateCode = (type: Type, informations: typeof infos) => {
|
const generateCode = (type: Type, informations: typeof infos) => {
|
||||||
const uid = new ShortUniqueId();
|
const uid = new ShortUniqueId();
|
||||||
const codes = informations.map(() => uid.randomUUID(6));
|
const codes = informations.map(() => uid.randomUUID(6));
|
||||||
|
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
axios
|
axios
|
||||||
.post<{ ok: boolean; valid?: number; reason?: string }>("/api/code", {
|
.post<{ ok: boolean; valid?: number; reason?: string }>("/api/code", {
|
||||||
type,
|
type,
|
||||||
codes,
|
codes,
|
||||||
infos: informations.map((info, index) => ({ ...info, code: codes[index] })),
|
infos: informations.map((info, index) => ({
|
||||||
expiryDate,
|
...info,
|
||||||
entity
|
code: codes[index],
|
||||||
})
|
})),
|
||||||
.then(({ data, status }) => {
|
expiryDate,
|
||||||
if (data.ok) {
|
entity,
|
||||||
toast.success(
|
})
|
||||||
`Successfully generated${data.valid ? ` ${data.valid}/${informations.length}` : ""} ${capitalize(
|
.then(({ data, status }) => {
|
||||||
type,
|
if (data.ok) {
|
||||||
)} codes and they have been notified by e-mail!`,
|
toast.success(
|
||||||
{ toastId: "success" },
|
`Successfully generated${
|
||||||
);
|
data.valid ? ` ${data.valid}/${informations.length}` : ""
|
||||||
|
} ${capitalize(type)} codes and they have been notified by e-mail!`,
|
||||||
|
{ toastId: "success" }
|
||||||
|
);
|
||||||
|
|
||||||
onFinish();
|
onFinish();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (status === 403) {
|
if (status === 403) {
|
||||||
toast.error(data.reason, { toastId: "forbidden" });
|
toast.error(data.reason, { toastId: "forbidden" });
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.catch(({ response: { status, data } }) => {
|
.catch(({ response: { status, data } }) => {
|
||||||
if (status === 403) {
|
if (status === 403) {
|
||||||
toast.error(data.reason, { toastId: "forbidden" });
|
toast.error(data.reason, { toastId: "forbidden" });
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
toast.error(`Something went wrong, please try again later!`, {
|
toast.error(`Something went wrong, please try again later!`, {
|
||||||
toastId: "error",
|
toastId: "error",
|
||||||
});
|
});
|
||||||
})
|
})
|
||||||
.finally(() => {
|
.finally(() => {
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
return clear();
|
return clear();
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleTemplateDownload = () => {
|
const handleTemplateDownload = () => {
|
||||||
const fileName = "BatchCodeTemplate.xlsx";
|
const fileName = "BatchCodeTemplate.xlsx";
|
||||||
const url = `https://firebasestorage.googleapis.com/v0/b/encoach-staging.appspot.com/o/import_templates%2F${fileName}?alt=media&token=b771a535-bf95-4060-889c-a086df65d480`;
|
const url = `https://firebasestorage.googleapis.com/v0/b/encoach-staging.appspot.com/o/import_templates%2F${fileName}?alt=media&token=b771a535-bf95-4060-889c-a086df65d480`;
|
||||||
|
|
||||||
const link = document.createElement('a');
|
const link = document.createElement("a");
|
||||||
link.href = url;
|
link.href = url;
|
||||||
|
|
||||||
link.download = fileName;
|
link.download = fileName;
|
||||||
|
|
||||||
document.body.appendChild(link);
|
document.body.appendChild(link);
|
||||||
link.click();
|
link.click();
|
||||||
document.body.removeChild(link);
|
document.body.removeChild(link);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Modal isOpen={showHelp} onClose={() => setShowHelp(false)}>
|
<Modal isOpen={showHelp} onClose={() => setShowHelp(false)}>
|
||||||
<>
|
<>
|
||||||
<div className="flex font-bold text-xl justify-center text-gray-700"><span>Excel File Format</span></div>
|
<div className="flex font-bold text-xl justify-center text-gray-700">
|
||||||
<div className="mt-4 flex flex-col gap-4">
|
<span>Excel File Format</span>
|
||||||
<div className="flex flex-col gap-3 bg-gray-100 rounded-lg p-4">
|
</div>
|
||||||
<div className="flex items-center gap-2">
|
<div className="mt-4 flex flex-col gap-4">
|
||||||
<HiOutlineDocumentText className={`w-5 h-5 text-mti-purple-light`} />
|
<div className="flex flex-col gap-3 bg-gray-100 rounded-lg p-4">
|
||||||
<h2 className="text-lg font-semibold">
|
<div className="flex items-center gap-2">
|
||||||
The uploaded document must:
|
<HiOutlineDocumentText
|
||||||
</h2>
|
className={`w-5 h-5 text-mti-purple-light`}
|
||||||
</div>
|
/>
|
||||||
<ul className="flex flex-col pl-10 gap-2">
|
<h2 className="text-lg font-semibold">
|
||||||
<li className="text-gray-700 list-disc">
|
The uploaded document must:
|
||||||
be an Excel .xlsx document.
|
</h2>
|
||||||
</li>
|
</div>
|
||||||
<li className="text-gray-700 list-disc">
|
<ul className="flex flex-col pl-10 gap-2">
|
||||||
only have a single spreadsheet with the following <b>exact same name</b> columns:
|
<li className="text-gray-700 list-disc">
|
||||||
<div className="py-4 pr-4">
|
be an Excel .xlsx document.
|
||||||
<table className="w-full bg-white">
|
</li>
|
||||||
<thead>
|
<li className="text-gray-700 list-disc">
|
||||||
<tr>
|
only have a single spreadsheet with the following{" "}
|
||||||
<th className="border border-neutral-200 px-2 py-1">First Name</th>
|
<b>exact same name</b> columns:
|
||||||
<th className="border border-neutral-200 px-2 py-1">Last Name</th>
|
<div className="py-4 pr-4">
|
||||||
<th className="border border-neutral-200 px-2 py-1">Passport/National ID</th>
|
<table className="w-full bg-white">
|
||||||
<th className="border border-neutral-200 px-2 py-1">E-mail</th>
|
<thead>
|
||||||
</tr>
|
<tr>
|
||||||
</thead>
|
<th className="border border-neutral-200 px-2 py-1">
|
||||||
</table>
|
First Name
|
||||||
</div>
|
</th>
|
||||||
</li>
|
<th className="border border-neutral-200 px-2 py-1">
|
||||||
</ul>
|
Last Name
|
||||||
</div>
|
</th>
|
||||||
<div className="flex flex-col gap-3 bg-gray-100 rounded-lg p-4">
|
<th className="border border-neutral-200 px-2 py-1">
|
||||||
<div className="flex items-center gap-2">
|
Passport/National ID
|
||||||
<IoInformationCircleOutline className={`w-5 h-5 text-mti-purple-light`} />
|
</th>
|
||||||
<h2 className="text-lg font-semibold">
|
<th className="border border-neutral-200 px-2 py-1">
|
||||||
Note that:
|
E-mail
|
||||||
</h2>
|
</th>
|
||||||
</div>
|
</tr>
|
||||||
<ul className="flex flex-col pl-10 gap-2">
|
</thead>
|
||||||
<li className="text-gray-700 list-disc">
|
</table>
|
||||||
all incorrect e-mails will be ignored.
|
</div>
|
||||||
</li>
|
</li>
|
||||||
<li className="text-gray-700 list-disc">
|
</ul>
|
||||||
all already registered e-mails will be ignored.
|
</div>
|
||||||
</li>
|
<div className="flex flex-col gap-3 bg-gray-100 rounded-lg p-4">
|
||||||
<li className="text-gray-700 list-disc">
|
<div className="flex items-center gap-2">
|
||||||
all rows which contain duplicate values in the columns: "Passport/National ID", "E-mail", will be ignored.
|
<IoInformationCircleOutline
|
||||||
</li>
|
className={`w-5 h-5 text-mti-purple-light`}
|
||||||
<li className="text-gray-700 list-disc">
|
/>
|
||||||
all of the e-mails in the file will receive an e-mail to join EnCoach with the role selected below.
|
<h2 className="text-lg font-semibold">Note that:</h2>
|
||||||
</li>
|
</div>
|
||||||
</ul>
|
<ul className="flex flex-col pl-10 gap-2">
|
||||||
</div>
|
<li className="text-gray-700 list-disc">
|
||||||
<div className="bg-gray-100 rounded-lg p-4">
|
all incorrect e-mails will be ignored.
|
||||||
<p className="text-gray-600">
|
</li>
|
||||||
{`The downloadable template is an example of a file that can be imported. Your document doesn't need to be a carbon copy of the template - it can have different styling but it must adhere to the previous requirements.`}
|
<li className="text-gray-700 list-disc">
|
||||||
</p>
|
all already registered e-mails will be ignored.
|
||||||
</div>
|
</li>
|
||||||
<div className="w-full flex justify-between mt-6 gap-8">
|
<li className="text-gray-700 list-disc">
|
||||||
<Button color="purple" onClick={() => setShowHelp(false)} variant="outline" className="self-end w-full bg-white">
|
all rows which contain duplicate values in the columns:
|
||||||
Close
|
"Passport/National ID", "E-mail", will be
|
||||||
</Button>
|
ignored.
|
||||||
|
</li>
|
||||||
|
<li className="text-gray-700 list-disc">
|
||||||
|
all of the e-mails in the file will receive an e-mail to join
|
||||||
|
EnCoach with the role selected below.
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
<div className="bg-gray-100 rounded-lg p-4">
|
||||||
|
<p className="text-gray-600">
|
||||||
|
{`The downloadable template is an example of a file that can be imported. Your document doesn't need to be a carbon copy of the template - it can have different styling but it must adhere to the previous requirements.`}
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
<div className="w-full flex justify-between mt-6 gap-8">
|
||||||
|
<Button
|
||||||
|
color="purple"
|
||||||
|
onClick={() => setShowHelp(false)}
|
||||||
|
variant="outline"
|
||||||
|
className="self-end w-full bg-white"
|
||||||
|
>
|
||||||
|
Close
|
||||||
|
</Button>
|
||||||
|
|
||||||
<Button color="purple" onClick={handleTemplateDownload} variant="solid" className="self-end w-full">
|
<Button
|
||||||
<div className="flex items-center gap-2">
|
color="purple"
|
||||||
<FaFileDownload size={24} />
|
onClick={handleTemplateDownload}
|
||||||
Download Template
|
variant="solid"
|
||||||
</div>
|
className="self-end w-full"
|
||||||
</Button>
|
>
|
||||||
</div>
|
<div className="flex items-center gap-2">
|
||||||
</div>
|
<FaFileDownload size={24} />
|
||||||
</>
|
Download Template
|
||||||
</Modal>
|
</div>
|
||||||
<div className="border-mti-gray-platinum flex flex-col gap-4 rounded-xl border p-4">
|
</Button>
|
||||||
<div className="flex items-end justify-between">
|
</div>
|
||||||
<label className="text-mti-gray-dim text-base font-normal">Choose an Excel file</label>
|
</div>
|
||||||
<button
|
</>
|
||||||
onClick={() => setShowHelp(true)}
|
</Modal>
|
||||||
className="tooltip cursor-pointer p-1.5 hover:bg-gray-200 rounded-full transition-colors duration-200"
|
<div className="border-mti-gray-platinum flex flex-col gap-4 rounded-xl border p-4">
|
||||||
data-tip="Excel File Format"
|
<div className="flex items-end justify-between">
|
||||||
>
|
<label className="text-mti-gray-dim text-base font-normal">
|
||||||
<IoInformationCircleOutline size={24} />
|
Choose an Excel file
|
||||||
</button>
|
</label>
|
||||||
</div>
|
<button
|
||||||
<Button onClick={openFilePicker} isLoading={isLoading} disabled={isLoading}>
|
onClick={() => setShowHelp(true)}
|
||||||
{filesContent.length > 0 ? filesContent[0].name : "Choose a file"}
|
className="tooltip cursor-pointer p-1.5 hover:bg-gray-200 rounded-full transition-colors duration-200"
|
||||||
</Button>
|
data-tip="Excel File Format"
|
||||||
{user && checkAccess(user, ["developer", "admin", "corporate", "mastercorporate"]) && (
|
>
|
||||||
<>
|
<IoInformationCircleOutline size={24} />
|
||||||
<div className="-md:flex-row -md:items-center flex justify-between gap-2 md:flex-col 2xl:flex-row 2xl:items-center">
|
</button>
|
||||||
<label className="text-mti-gray-dim text-base font-normal">Expiry Date</label>
|
</div>
|
||||||
<Checkbox isChecked={isExpiryDateEnabled} onChange={setIsExpiryDateEnabled} disabled={!!user.subscriptionExpirationDate}>
|
<Button
|
||||||
Enabled
|
onClick={openFilePicker}
|
||||||
</Checkbox>
|
isLoading={isLoading}
|
||||||
</div>
|
disabled={isLoading}
|
||||||
{isExpiryDateEnabled && (
|
>
|
||||||
<ReactDatePicker
|
{filesContent.length > 0 ? filesContent[0].name : "Choose a file"}
|
||||||
className={clsx(
|
</Button>
|
||||||
"flex min-h-[70px] w-full cursor-pointer justify-center rounded-full border p-6 text-sm font-normal focus:outline-none",
|
{user &&
|
||||||
"hover:border-mti-purple tooltip",
|
checkAccess(user, [
|
||||||
"transition duration-300 ease-in-out",
|
"developer",
|
||||||
)}
|
"admin",
|
||||||
filterDate={(date) =>
|
"corporate",
|
||||||
moment(date).isAfter(new Date()) &&
|
"mastercorporate",
|
||||||
(user.subscriptionExpirationDate ? moment(date).isBefore(user.subscriptionExpirationDate) : true)
|
]) && (
|
||||||
}
|
<>
|
||||||
dateFormat="dd/MM/yyyy"
|
<div className="-md:flex-row -md:items-center flex justify-between gap-2 md:flex-col 2xl:flex-row 2xl:items-center">
|
||||||
selected={expiryDate}
|
<label className="text-mti-gray-dim text-base font-normal">
|
||||||
onChange={(date) => setExpiryDate(date)}
|
Expiry Date
|
||||||
/>
|
</label>
|
||||||
)}
|
<Checkbox
|
||||||
</>
|
isChecked={isExpiryDateEnabled}
|
||||||
)}
|
onChange={setIsExpiryDateEnabled}
|
||||||
<div className={clsx("flex flex-col gap-4")}>
|
disabled={!!user.subscriptionExpirationDate}
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Entity</label>
|
>
|
||||||
<Select
|
Enabled
|
||||||
defaultValue={{ value: (entities || [])[0]?.id, label: (entities || [])[0]?.label }}
|
</Checkbox>
|
||||||
options={entities.map((e) => ({ value: e.id, label: e.label }))}
|
</div>
|
||||||
onChange={(e) => setEntity(e?.value || undefined)}
|
{isExpiryDateEnabled && (
|
||||||
isClearable={checkAccess(user, ["admin", "developer"])}
|
<ReactDatePicker
|
||||||
/>
|
className={clsx(
|
||||||
</div>
|
"flex min-h-[70px] w-full cursor-pointer justify-center rounded-full border p-6 text-sm font-normal focus:outline-none",
|
||||||
<label className="text-mti-gray-dim text-base font-normal">Select the type of user they should be</label>
|
"hover:border-mti-purple tooltip",
|
||||||
{user && (
|
"transition duration-300 ease-in-out"
|
||||||
<select
|
)}
|
||||||
defaultValue="student"
|
filterDate={(date) =>
|
||||||
onChange={(e) => setType(e.target.value as typeof user.type)}
|
moment(date).isAfter(new Date()) &&
|
||||||
className="flex min-h-[70px] w-full min-w-[350px] cursor-pointer justify-center rounded-full border bg-white p-6 text-sm font-normal focus:outline-none">
|
(user.subscriptionExpirationDate
|
||||||
{Object.keys(USER_TYPE_LABELS)
|
? moment(date).isBefore(user.subscriptionExpirationDate)
|
||||||
.filter((x) => {
|
: true)
|
||||||
const { list, perm } = USER_TYPE_PERMISSIONS[x as Type];
|
}
|
||||||
return checkAccess(user, getTypesOfUser(list), permissions, perm);
|
dateFormat="dd/MM/yyyy"
|
||||||
})
|
selected={expiryDate}
|
||||||
.map((type) => (
|
onChange={(date) => setExpiryDate(date)}
|
||||||
<option key={type} value={type}>
|
/>
|
||||||
{USER_TYPE_LABELS[type as keyof typeof USER_TYPE_LABELS]}
|
)}
|
||||||
</option>
|
</>
|
||||||
))}
|
)}
|
||||||
</select>
|
<div className={clsx("flex flex-col gap-4")}>
|
||||||
)}
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
{infos.length > 0 && <CodeGenImportSummary infos={infos} parsedExcel={parsedExcel} duplicateRows={duplicatedRows}/>}
|
Entity
|
||||||
{infos.length !== 0 && (
|
</label>
|
||||||
<div className="flex w-full flex-col gap-4">
|
<Select
|
||||||
<span className="text-mti-gray-dim text-base font-normal">Codes will be sent to:</span>
|
defaultValue={{
|
||||||
<CodegenTable infos={infos} />
|
value: (entities || [])[0]?.id,
|
||||||
</div>
|
label: (entities || [])[0]?.label,
|
||||||
)}
|
}}
|
||||||
{checkAccess(user, ["developer", "admin", "corporate", "mastercorporate"], permissions, "createCodes") && (
|
options={entities.map((e) => ({ value: e.id, label: e.label }))}
|
||||||
<Button onClick={generateAndInvite} disabled={infos.length === 0 || (isExpiryDateEnabled ? !expiryDate : false)}>
|
onChange={(e) => setEntity(e?.value || undefined)}
|
||||||
Generate & Send
|
isClearable={checkAccess(user, ["admin", "developer"])}
|
||||||
</Button>
|
/>
|
||||||
)}
|
</div>
|
||||||
</div>
|
<label className="text-mti-gray-dim text-base font-normal">
|
||||||
</>
|
Select the type of user they should be
|
||||||
);
|
</label>
|
||||||
|
{user && (
|
||||||
|
<select
|
||||||
|
defaultValue="student"
|
||||||
|
onChange={(e) => setType(e.target.value as typeof user.type)}
|
||||||
|
className="flex min-h-[70px] w-full min-w-[350px] cursor-pointer justify-center rounded-full border bg-white p-6 text-sm font-normal focus:outline-none"
|
||||||
|
>
|
||||||
|
{Object.keys(USER_TYPE_LABELS).reduce((acc, x) => {
|
||||||
|
const { list, perm } = USER_TYPE_PERMISSIONS[x as Type];
|
||||||
|
if (checkAccess(user, getTypesOfUser(list), permissions, perm))
|
||||||
|
acc.push(
|
||||||
|
<option key={type} value={type}>
|
||||||
|
{USER_TYPE_LABELS[type as keyof typeof USER_TYPE_LABELS]}
|
||||||
|
</option>
|
||||||
|
);
|
||||||
|
return acc;
|
||||||
|
}, [] as JSX.Element[])}
|
||||||
|
</select>
|
||||||
|
)}
|
||||||
|
{infos.length > 0 && (
|
||||||
|
<CodeGenImportSummary
|
||||||
|
infos={infos}
|
||||||
|
parsedExcel={parsedExcel}
|
||||||
|
duplicateRows={duplicatedRows}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{infos.length !== 0 && (
|
||||||
|
<div className="flex w-full flex-col gap-4">
|
||||||
|
<span className="text-mti-gray-dim text-base font-normal">
|
||||||
|
Codes will be sent to:
|
||||||
|
</span>
|
||||||
|
<CodegenTable infos={infos} />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{checkAccess(
|
||||||
|
user,
|
||||||
|
["developer", "admin", "corporate", "mastercorporate"],
|
||||||
|
permissions,
|
||||||
|
"createCodes"
|
||||||
|
) && (
|
||||||
|
<Button
|
||||||
|
onClick={generateAndInvite}
|
||||||
|
disabled={
|
||||||
|
infos.length === 0 || (isExpiryDateEnabled ? !expiryDate : false)
|
||||||
|
}
|
||||||
|
>
|
||||||
|
Generate & Send
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import Button from "@/components/Low/Button";
|
import Button from "@/components/Low/Button";
|
||||||
import Checkbox from "@/components/Low/Checkbox";
|
import Checkbox from "@/components/Low/Checkbox";
|
||||||
import { PERMISSIONS } from "@/constants/userPermissions";
|
|
||||||
import { Type, User } from "@/interfaces/user";
|
import { Type, User } from "@/interfaces/user";
|
||||||
import { USER_TYPE_LABELS } from "@/resources/user";
|
import { USER_TYPE_LABELS } from "@/resources/user";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
@@ -13,173 +12,225 @@ import { toast } from "react-toastify";
|
|||||||
import ShortUniqueId from "short-unique-id";
|
import ShortUniqueId from "short-unique-id";
|
||||||
import { checkAccess, getTypesOfUser } from "@/utils/permissions";
|
import { checkAccess, getTypesOfUser } from "@/utils/permissions";
|
||||||
import { PermissionType } from "@/interfaces/permissions";
|
import { PermissionType } from "@/interfaces/permissions";
|
||||||
import usePermissions from "@/hooks/usePermissions";
|
|
||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
import Select from "@/components/Low/Select";
|
import Select from "@/components/Low/Select";
|
||||||
import { useAllowedEntities } from "@/hooks/useEntityPermissions";
|
|
||||||
|
|
||||||
const USER_TYPE_PERMISSIONS: {
|
const USER_TYPE_PERMISSIONS: {
|
||||||
[key in Type]: { perm: PermissionType | undefined; list: Type[] };
|
[key in Type]: { perm: PermissionType | undefined; list: Type[] };
|
||||||
} = {
|
} = {
|
||||||
student: {
|
student: {
|
||||||
perm: "createCodeStudent",
|
perm: "createCodeStudent",
|
||||||
list: [],
|
list: [],
|
||||||
},
|
},
|
||||||
teacher: {
|
teacher: {
|
||||||
perm: "createCodeTeacher",
|
perm: "createCodeTeacher",
|
||||||
list: [],
|
list: [],
|
||||||
},
|
},
|
||||||
agent: {
|
agent: {
|
||||||
perm: "createCodeCountryManager",
|
perm: "createCodeCountryManager",
|
||||||
list: ["student", "teacher", "corporate", "mastercorporate"],
|
list: ["student", "teacher", "corporate", "mastercorporate"],
|
||||||
},
|
},
|
||||||
corporate: {
|
corporate: {
|
||||||
perm: "createCodeCorporate",
|
perm: "createCodeCorporate",
|
||||||
list: ["student", "teacher"],
|
list: ["student", "teacher"],
|
||||||
},
|
},
|
||||||
mastercorporate: {
|
mastercorporate: {
|
||||||
perm: undefined,
|
perm: undefined,
|
||||||
list: ["student", "teacher", "corporate"],
|
list: ["student", "teacher", "corporate"],
|
||||||
},
|
},
|
||||||
admin: {
|
admin: {
|
||||||
perm: "createCodeAdmin",
|
perm: "createCodeAdmin",
|
||||||
list: ["student", "teacher", "agent", "corporate", "admin", "mastercorporate"],
|
list: [
|
||||||
},
|
"student",
|
||||||
developer: {
|
"teacher",
|
||||||
perm: undefined,
|
"agent",
|
||||||
list: ["student", "teacher", "agent", "corporate", "admin", "developer", "mastercorporate"],
|
"corporate",
|
||||||
},
|
"admin",
|
||||||
|
"mastercorporate",
|
||||||
|
],
|
||||||
|
},
|
||||||
|
developer: {
|
||||||
|
perm: undefined,
|
||||||
|
list: [
|
||||||
|
"student",
|
||||||
|
"teacher",
|
||||||
|
"agent",
|
||||||
|
"corporate",
|
||||||
|
"admin",
|
||||||
|
"developer",
|
||||||
|
"mastercorporate",
|
||||||
|
],
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
user: User;
|
user: User;
|
||||||
permissions: PermissionType[];
|
permissions: PermissionType[];
|
||||||
entities: EntityWithRoles[]
|
entities: EntityWithRoles[];
|
||||||
onFinish: () => void;
|
onFinish: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function CodeGenerator({ user, entities = [], permissions, onFinish }: Props) {
|
export default function CodeGenerator({
|
||||||
const [generatedCode, setGeneratedCode] = useState<string>();
|
user,
|
||||||
|
entities = [],
|
||||||
|
permissions,
|
||||||
|
onFinish,
|
||||||
|
}: Props) {
|
||||||
|
const [generatedCode, setGeneratedCode] = useState<string>();
|
||||||
|
|
||||||
const [expiryDate, setExpiryDate] = useState<Date | null>(
|
const [expiryDate, setExpiryDate] = useState<Date | null>(
|
||||||
user?.subscriptionExpirationDate ? moment(user.subscriptionExpirationDate).toDate() : null,
|
user?.subscriptionExpirationDate
|
||||||
);
|
? moment(user.subscriptionExpirationDate).toDate()
|
||||||
const [isExpiryDateEnabled, setIsExpiryDateEnabled] = useState(true);
|
: null
|
||||||
const [type, setType] = useState<Type>("student");
|
);
|
||||||
const [entity, setEntity] = useState((entities || [])[0]?.id || undefined)
|
const [isExpiryDateEnabled, setIsExpiryDateEnabled] = useState(true);
|
||||||
|
const [type, setType] = useState<Type>("student");
|
||||||
|
const [entity, setEntity] = useState((entities || [])[0]?.id || undefined);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!isExpiryDateEnabled) setExpiryDate(null);
|
if (!isExpiryDateEnabled) setExpiryDate(null);
|
||||||
}, [isExpiryDateEnabled]);
|
}, [isExpiryDateEnabled]);
|
||||||
|
|
||||||
const generateCode = (type: Type) => {
|
const generateCode = (type: Type) => {
|
||||||
const uid = new ShortUniqueId();
|
const uid = new ShortUniqueId();
|
||||||
const code = uid.randomUUID(6);
|
const code = uid.randomUUID(6);
|
||||||
|
|
||||||
axios
|
axios
|
||||||
.post("/api/code", { type, codes: [code], expiryDate, entity })
|
.post("/api/code", { type, codes: [code], expiryDate, entity })
|
||||||
.then(({ data, status }) => {
|
.then(({ data, status }) => {
|
||||||
if (data.ok) {
|
if (data.ok) {
|
||||||
toast.success(`Successfully generated a ${capitalize(type)} code!`, {
|
toast.success(`Successfully generated a ${capitalize(type)} code!`, {
|
||||||
toastId: "success",
|
toastId: "success",
|
||||||
});
|
});
|
||||||
setGeneratedCode(code);
|
setGeneratedCode(code);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (status === 403) {
|
if (status === 403) {
|
||||||
toast.error(data.reason, { toastId: "forbidden" });
|
toast.error(data.reason, { toastId: "forbidden" });
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.catch(({ response: { status, data } }) => {
|
.catch(({ response: { status, data } }) => {
|
||||||
if (status === 403) {
|
if (status === 403) {
|
||||||
toast.error(data.reason, { toastId: "forbidden" });
|
toast.error(data.reason, { toastId: "forbidden" });
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
toast.error(`Something went wrong, please try again later!`, {
|
toast.error(`Something went wrong, please try again later!`, {
|
||||||
toastId: "error",
|
toastId: "error",
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex flex-col gap-4 border p-4 border-mti-gray-platinum rounded-xl">
|
<div className="flex flex-col gap-4 border p-4 border-mti-gray-platinum rounded-xl">
|
||||||
<label className="font-normal text-base text-mti-gray-dim">User Code Generator</label>
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
<div className={clsx("flex flex-col gap-4")}>
|
User Code Generator
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Entity</label>
|
</label>
|
||||||
<Select
|
<div className={clsx("flex flex-col gap-4")}>
|
||||||
defaultValue={{ value: (entities || [])[0]?.id, label: (entities || [])[0]?.label }}
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
options={entities.map((e) => ({ value: e.id, label: e.label }))}
|
Entity
|
||||||
onChange={(e) => setEntity(e?.value || undefined)}
|
</label>
|
||||||
isClearable={checkAccess(user, ["admin", "developer"])}
|
<Select
|
||||||
/>
|
defaultValue={{
|
||||||
</div>
|
value: (entities || [])[0]?.id,
|
||||||
|
label: (entities || [])[0]?.label,
|
||||||
|
}}
|
||||||
|
options={entities.map((e) => ({ value: e.id, label: e.label }))}
|
||||||
|
onChange={(e) => setEntity(e?.value || undefined)}
|
||||||
|
isClearable={checkAccess(user, ["admin", "developer"])}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div className={clsx("flex flex-col gap-4")}>
|
<div className={clsx("flex flex-col gap-4")}>
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Type</label>
|
<label className="font-normal text-base text-mti-gray-dim">Type</label>
|
||||||
<select
|
<select
|
||||||
defaultValue="student"
|
defaultValue="student"
|
||||||
onChange={(e) => setType(e.target.value as typeof user.type)}
|
onChange={(e) => setType(e.target.value as typeof user.type)}
|
||||||
className="p-6 w-full min-w-[350px] min-h-[70px] flex justify-center text-sm font-normal rounded-full border focus:outline-none cursor-pointer bg-white">
|
className="p-6 w-full min-w-[350px] min-h-[70px] flex justify-center text-sm font-normal rounded-full border focus:outline-none cursor-pointer bg-white"
|
||||||
{Object.keys(USER_TYPE_LABELS)
|
>
|
||||||
.filter((x) => {
|
{Object.keys(USER_TYPE_LABELS).reduce<string[]>((acc, x) => {
|
||||||
const { list, perm } = USER_TYPE_PERMISSIONS[x as Type];
|
const { list, perm } = USER_TYPE_PERMISSIONS[x as Type];
|
||||||
return checkAccess(user, getTypesOfUser(list), permissions, perm);
|
if (checkAccess(user, getTypesOfUser(list), permissions, perm))
|
||||||
})
|
acc.push(x);
|
||||||
.map((type) => (
|
return acc;
|
||||||
<option key={type} value={type}>
|
}, [])}
|
||||||
{USER_TYPE_LABELS[type as keyof typeof USER_TYPE_LABELS]}
|
</select>
|
||||||
</option>
|
</div>
|
||||||
))}
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{checkAccess(user, ["developer", "admin", "corporate", "mastercorporate"]) && (
|
{checkAccess(user, [
|
||||||
<>
|
"developer",
|
||||||
<div className="-md:flex-row -md:items-center flex justify-between gap-2 md:flex-col 2xl:flex-row 2xl:items-center">
|
"admin",
|
||||||
<label className="text-mti-gray-dim text-base font-normal">Expiry Date</label>
|
"corporate",
|
||||||
<Checkbox isChecked={isExpiryDateEnabled} onChange={setIsExpiryDateEnabled} disabled={!!user.subscriptionExpirationDate}>
|
"mastercorporate",
|
||||||
Enabled
|
]) && (
|
||||||
</Checkbox>
|
<>
|
||||||
</div>
|
<div className="-md:flex-row -md:items-center flex justify-between gap-2 md:flex-col 2xl:flex-row 2xl:items-center">
|
||||||
{isExpiryDateEnabled && (
|
<label className="text-mti-gray-dim text-base font-normal">
|
||||||
<ReactDatePicker
|
Expiry Date
|
||||||
className={clsx(
|
</label>
|
||||||
"flex min-h-[70px] w-full cursor-pointer justify-center rounded-full border p-6 text-sm font-normal focus:outline-none",
|
<Checkbox
|
||||||
"hover:border-mti-purple tooltip",
|
isChecked={isExpiryDateEnabled}
|
||||||
"transition duration-300 ease-in-out",
|
onChange={setIsExpiryDateEnabled}
|
||||||
)}
|
disabled={!!user.subscriptionExpirationDate}
|
||||||
filterDate={(date) =>
|
>
|
||||||
moment(date).isAfter(new Date()) &&
|
Enabled
|
||||||
(user.subscriptionExpirationDate ? moment(date).isBefore(user.subscriptionExpirationDate) : true)
|
</Checkbox>
|
||||||
}
|
</div>
|
||||||
dateFormat="dd/MM/yyyy"
|
{isExpiryDateEnabled && (
|
||||||
selected={expiryDate}
|
<ReactDatePicker
|
||||||
onChange={(date) => setExpiryDate(date)}
|
className={clsx(
|
||||||
/>
|
"flex min-h-[70px] w-full cursor-pointer justify-center rounded-full border p-6 text-sm font-normal focus:outline-none",
|
||||||
)}
|
"hover:border-mti-purple tooltip",
|
||||||
</>
|
"transition duration-300 ease-in-out"
|
||||||
)}
|
)}
|
||||||
{checkAccess(user, ["developer", "admin", "corporate", "mastercorporate"], permissions, "createCodes") && (
|
filterDate={(date) =>
|
||||||
<Button onClick={() => generateCode(type)} disabled={isExpiryDateEnabled ? !expiryDate : false}>
|
moment(date).isAfter(new Date()) &&
|
||||||
Generate
|
(user.subscriptionExpirationDate
|
||||||
</Button>
|
? moment(date).isBefore(user.subscriptionExpirationDate)
|
||||||
)}
|
: true)
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Generated Code:</label>
|
}
|
||||||
<div
|
dateFormat="dd/MM/yyyy"
|
||||||
className={clsx(
|
selected={expiryDate}
|
||||||
"p-6 w-full min-h-[70px] flex justify-center text-sm font-normal rounded-full border focus:outline-none cursor-pointer",
|
onChange={(date) => setExpiryDate(date)}
|
||||||
"hover:border-mti-purple tooltip",
|
/>
|
||||||
"transition duration-300 ease-in-out",
|
)}
|
||||||
)}
|
</>
|
||||||
data-tip="Click to copy"
|
)}
|
||||||
onClick={() => {
|
{checkAccess(
|
||||||
if (generatedCode) navigator.clipboard.writeText(generatedCode);
|
user,
|
||||||
}}>
|
["developer", "admin", "corporate", "mastercorporate"],
|
||||||
{generatedCode}
|
permissions,
|
||||||
</div>
|
"createCodes"
|
||||||
{generatedCode && <span className="text-sm text-mti-gray-dim font-light">Give this code to the user to complete their registration</span>}
|
) && (
|
||||||
</div>
|
<Button
|
||||||
);
|
onClick={() => generateCode(type)}
|
||||||
|
disabled={isExpiryDateEnabled ? !expiryDate : false}
|
||||||
|
>
|
||||||
|
Generate
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
|
Generated Code:
|
||||||
|
</label>
|
||||||
|
<div
|
||||||
|
className={clsx(
|
||||||
|
"p-6 w-full min-h-[70px] flex justify-center text-sm font-normal rounded-full border focus:outline-none cursor-pointer",
|
||||||
|
"hover:border-mti-purple tooltip",
|
||||||
|
"transition duration-300 ease-in-out"
|
||||||
|
)}
|
||||||
|
data-tip="Click to copy"
|
||||||
|
onClick={() => {
|
||||||
|
if (generatedCode) navigator.clipboard.writeText(generatedCode);
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{generatedCode}
|
||||||
|
</div>
|
||||||
|
{generatedCode && (
|
||||||
|
<span className="text-sm text-mti-gray-dim font-light">
|
||||||
|
Give this code to the user to complete their registration
|
||||||
|
</span>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,192 +5,329 @@ import Separator from "@/components/Low/Separator";
|
|||||||
import { Grading, Step } from "@/interfaces";
|
import { Grading, Step } from "@/interfaces";
|
||||||
import { Entity } from "@/interfaces/entity";
|
import { Entity } from "@/interfaces/entity";
|
||||||
import { User } from "@/interfaces/user";
|
import { User } from "@/interfaces/user";
|
||||||
import { CEFR_STEPS, GENERAL_STEPS, IELTS_STEPS, TOFEL_STEPS } from "@/resources/grading";
|
import {
|
||||||
import { mapBy } from "@/utils";
|
CEFR_STEPS,
|
||||||
|
GENERAL_STEPS,
|
||||||
|
IELTS_STEPS,
|
||||||
|
TOFEL_STEPS,
|
||||||
|
} from "@/resources/grading";
|
||||||
import { checkAccess } from "@/utils/permissions";
|
import { checkAccess } from "@/utils/permissions";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import { Divider } from "primereact/divider";
|
import {
|
||||||
import { useEffect, useState } from "react";
|
Dispatch,
|
||||||
|
memo,
|
||||||
|
SetStateAction,
|
||||||
|
useCallback,
|
||||||
|
useEffect,
|
||||||
|
useState,
|
||||||
|
} from "react";
|
||||||
import { BsPlusCircle, BsTrash } from "react-icons/bs";
|
import { BsPlusCircle, BsTrash } from "react-icons/bs";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
|
|
||||||
const areStepsOverlapped = (steps: Step[]) => {
|
const areStepsOverlapped = (steps: Step[]) => {
|
||||||
for (let i = 0; i < steps.length; i++) {
|
for (let i = 0; i < steps.length; i++) {
|
||||||
if (i === 0) continue;
|
if (i === 0) continue;
|
||||||
|
|
||||||
const step = steps[i];
|
const step = steps[i];
|
||||||
const previous = steps[i - 1];
|
const previous = steps[i - 1];
|
||||||
|
|
||||||
if (previous.max >= step.min) return true;
|
if (previous.max >= step.min) return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
|
interface RowProps {
|
||||||
|
min: number;
|
||||||
|
max: number;
|
||||||
|
index: number;
|
||||||
|
label: string;
|
||||||
|
isLast: boolean;
|
||||||
|
isLoading: boolean;
|
||||||
|
setSteps: Dispatch<SetStateAction<Step[]>>;
|
||||||
|
addRow: (index: number) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
function GradingRow({
|
||||||
|
min,
|
||||||
|
max,
|
||||||
|
label,
|
||||||
|
index,
|
||||||
|
isLoading,
|
||||||
|
isLast,
|
||||||
|
setSteps,
|
||||||
|
addRow,
|
||||||
|
}: RowProps) {
|
||||||
|
const onChangeMin = useCallback(
|
||||||
|
(e: string) => {
|
||||||
|
setSteps((prev) =>
|
||||||
|
prev.map((x, i) => (i === index ? { ...x, min: parseInt(e) } : x))
|
||||||
|
);
|
||||||
|
},
|
||||||
|
[index, setSteps]
|
||||||
|
);
|
||||||
|
|
||||||
|
const onChangeMax = useCallback(
|
||||||
|
(e: string) => {
|
||||||
|
setSteps((prev) =>
|
||||||
|
prev.map((x, i) => (i === index ? { ...x, max: parseInt(e) } : x))
|
||||||
|
);
|
||||||
|
},
|
||||||
|
[index, setSteps]
|
||||||
|
);
|
||||||
|
|
||||||
|
const onChangeLabel = useCallback(
|
||||||
|
(e: string) => {
|
||||||
|
setSteps((prev) =>
|
||||||
|
prev.map((x, i) => (i === index ? { ...x, label: e } : x))
|
||||||
|
);
|
||||||
|
},
|
||||||
|
[index, setSteps]
|
||||||
|
);
|
||||||
|
|
||||||
|
const onAddRow = useCallback(() => addRow(index), [addRow, index]);
|
||||||
|
|
||||||
|
const removeRow = useCallback(
|
||||||
|
() => setSteps((prev) => prev.filter((_, i) => i !== index)),
|
||||||
|
[index, setSteps]
|
||||||
|
);
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<div className="flex items-center gap-4">
|
||||||
|
<div className="grid grid-cols-3 gap-4 w-full">
|
||||||
|
<Input
|
||||||
|
label="Min. Percentage"
|
||||||
|
value={min}
|
||||||
|
type="number"
|
||||||
|
disabled={index === 0 || isLoading}
|
||||||
|
onChange={onChangeMin}
|
||||||
|
name="min"
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
label="Grade"
|
||||||
|
value={label}
|
||||||
|
type="text"
|
||||||
|
disabled={isLoading}
|
||||||
|
onChange={onChangeLabel}
|
||||||
|
name="min"
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
label="Max. Percentage"
|
||||||
|
value={max}
|
||||||
|
type="number"
|
||||||
|
disabled={isLast || isLoading}
|
||||||
|
onChange={onChangeMax}
|
||||||
|
name="max"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
{index !== 0 && !isLast && (
|
||||||
|
<button
|
||||||
|
disabled={isLoading}
|
||||||
|
className="pt-9 text-xl group"
|
||||||
|
onClick={removeRow}
|
||||||
|
>
|
||||||
|
<div className="w-full h-full flex items-center justify-center group-hover:bg-neutral-200 rounded-full p-3 transition ease-in-out duration-300">
|
||||||
|
<BsTrash />
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{!isLast && (
|
||||||
|
<Button
|
||||||
|
className="w-full flex items-center justify-center"
|
||||||
|
disabled={isLoading}
|
||||||
|
onClick={onAddRow}
|
||||||
|
>
|
||||||
|
<BsPlusCircle />
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
const GradingRowMemo = memo(GradingRow);
|
||||||
interface Props {
|
interface Props {
|
||||||
user: User;
|
user: User;
|
||||||
entitiesGrading: Grading[];
|
entitiesGrading: Grading[];
|
||||||
entities: Entity[]
|
entities: Entity[];
|
||||||
mutate: () => void
|
mutate: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function CorporateGradingSystem({ user, entitiesGrading = [], entities = [], mutate }: Props) {
|
export default function CorporateGradingSystem({
|
||||||
const [entity, setEntity] = useState(entitiesGrading[0]?.entity || undefined)
|
user,
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
entitiesGrading = [],
|
||||||
const [steps, setSteps] = useState<Step[]>([]);
|
entities = [],
|
||||||
const [otherEntities, setOtherEntities] = useState<string[]>([])
|
mutate,
|
||||||
|
}: Props) {
|
||||||
|
const [entity, setEntity] = useState(entitiesGrading[0]?.entity || undefined);
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
const [steps, setSteps] = useState<Step[]>([]);
|
||||||
|
const [otherEntities, setOtherEntities] = useState<string[]>([]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (entity) {
|
if (entity) {
|
||||||
const entitySteps = entitiesGrading.find(e => e.entity === entity)!.steps
|
const entitySteps = entitiesGrading.find(
|
||||||
setSteps(entitySteps || [])
|
(e) => e.entity === entity
|
||||||
}
|
)!.steps;
|
||||||
}, [entitiesGrading, entity])
|
setSteps(entitySteps || []);
|
||||||
|
}
|
||||||
|
}, [entitiesGrading, entity]);
|
||||||
|
|
||||||
const saveGradingSystem = () => {
|
const saveGradingSystem = () => {
|
||||||
if (!steps.every((x) => x.min < x.max)) return toast.error("One of your steps has a minimum threshold inferior to its superior threshold.");
|
if (!steps.every((x) => x.min < x.max))
|
||||||
if (areStepsOverlapped(steps)) return toast.error("There seems to be an overlap in one of your steps.");
|
return toast.error(
|
||||||
if (
|
"One of your steps has a minimum threshold inferior to its superior threshold."
|
||||||
steps.reduce((acc, curr) => {
|
);
|
||||||
return acc - (curr.max - curr.min + 1);
|
if (areStepsOverlapped(steps))
|
||||||
}, 100) > 0
|
return toast.error("There seems to be an overlap in one of your steps.");
|
||||||
)
|
if (
|
||||||
return toast.error("There seems to be an open interval in your steps.");
|
steps.reduce((acc, curr) => {
|
||||||
|
return acc - (curr.max - curr.min + 1);
|
||||||
|
}, 100) > 0
|
||||||
|
)
|
||||||
|
return toast.error("There seems to be an open interval in your steps.");
|
||||||
|
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
axios
|
axios
|
||||||
.post("/api/grading", { user: user.id, entity, steps })
|
.post("/api/grading", { user: user.id, entity, steps })
|
||||||
.then(() => toast.success("Your grading system has been saved!"))
|
.then(() => toast.success("Your grading system has been saved!"))
|
||||||
.then(mutate)
|
.then(mutate)
|
||||||
.catch(() => toast.error("Something went wrong, please try again later"))
|
.catch(() => toast.error("Something went wrong, please try again later"))
|
||||||
.finally(() => setIsLoading(false));
|
.finally(() => setIsLoading(false));
|
||||||
};
|
};
|
||||||
|
|
||||||
const applyToOtherEntities = () => {
|
const applyToOtherEntities = () => {
|
||||||
if (!steps.every((x) => x.min < x.max)) return toast.error("One of your steps has a minimum threshold inferior to its superior threshold.");
|
if (!steps.every((x) => x.min < x.max))
|
||||||
if (areStepsOverlapped(steps)) return toast.error("There seems to be an overlap in one of your steps.");
|
return toast.error(
|
||||||
if (
|
"One of your steps has a minimum threshold inferior to its superior threshold."
|
||||||
steps.reduce((acc, curr) => {
|
);
|
||||||
return acc - (curr.max - curr.min + 1);
|
if (areStepsOverlapped(steps))
|
||||||
}, 100) > 0
|
return toast.error("There seems to be an overlap in one of your steps.");
|
||||||
)
|
if (
|
||||||
return toast.error("There seems to be an open interval in your steps.");
|
steps.reduce((acc, curr) => {
|
||||||
|
return acc - (curr.max - curr.min + 1);
|
||||||
|
}, 100) > 0
|
||||||
|
)
|
||||||
|
return toast.error("There seems to be an open interval in your steps.");
|
||||||
|
|
||||||
if (otherEntities.length === 0) return toast.error("Select at least one entity")
|
if (otherEntities.length === 0)
|
||||||
|
return toast.error("Select at least one entity");
|
||||||
|
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
axios
|
axios
|
||||||
.post("/api/grading/multiple", { user: user.id, entities: otherEntities, steps })
|
.post("/api/grading/multiple", {
|
||||||
.then(() => toast.success("Your grading system has been saved!"))
|
user: user.id,
|
||||||
.then(mutate)
|
entities: otherEntities,
|
||||||
.catch(() => toast.error("Something went wrong, please try again later"))
|
steps,
|
||||||
.finally(() => setIsLoading(false));
|
})
|
||||||
};
|
.then(() => toast.success("Your grading system has been saved!"))
|
||||||
|
.then(mutate)
|
||||||
|
.catch(() => toast.error("Something went wrong, please try again later"))
|
||||||
|
.finally(() => setIsLoading(false));
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
const addRow = useCallback((index: number) => {
|
||||||
<div className="flex flex-col gap-4 border p-4 border-mti-gray-platinum rounded-xl">
|
setSteps((prev) => {
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Grading System</label>
|
const item = {
|
||||||
<div className={clsx("flex flex-col gap-4")}>
|
min: prev[index === 0 ? 0 : index - 1].max + 1,
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Entity</label>
|
max: prev[index + 1].min - 1,
|
||||||
<Select
|
label: "",
|
||||||
defaultValue={{ value: (entities || [])[0]?.id, label: (entities || [])[0]?.label }}
|
};
|
||||||
options={entities.map((e) => ({ value: e.id, label: e.label }))}
|
return [
|
||||||
onChange={(e) => setEntity(e?.value || undefined)}
|
...prev.slice(0, index + 1),
|
||||||
isClearable={checkAccess(user, ["admin", "developer"])}
|
item,
|
||||||
/>
|
...prev.slice(index + 1, prev.length),
|
||||||
</div>
|
];
|
||||||
|
});
|
||||||
|
}, []);
|
||||||
|
|
||||||
{entities.length > 1 && (
|
return (
|
||||||
<>
|
<div className="flex flex-col gap-4 border p-4 border-mti-gray-platinum rounded-xl">
|
||||||
<Separator />
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Apply this grading system to other entities</label>
|
Grading System
|
||||||
<Select
|
</label>
|
||||||
options={entities.map((e) => ({ value: e.id, label: e.label }))}
|
<div className={clsx("flex flex-col gap-4")}>
|
||||||
onChange={(e) => !e ? setOtherEntities([]) : setOtherEntities(e.map(o => o.value!))}
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
isMulti
|
Entity
|
||||||
/>
|
</label>
|
||||||
<Button onClick={applyToOtherEntities} isLoading={isLoading} disabled={isLoading || otherEntities.length === 0} variant="outline">
|
<Select
|
||||||
Apply to {otherEntities.length} other entities
|
defaultValue={{
|
||||||
</Button>
|
value: (entities || [])[0]?.id,
|
||||||
<Separator />
|
label: (entities || [])[0]?.label,
|
||||||
</>
|
}}
|
||||||
)}
|
options={entities.map((e) => ({ value: e.id, label: e.label }))}
|
||||||
|
onChange={(e) => setEntity(e?.value || undefined)}
|
||||||
|
isClearable={checkAccess(user, ["admin", "developer"])}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Preset Systems</label>
|
{entities.length > 1 && (
|
||||||
<div className="grid grid-cols-4 gap-4">
|
<>
|
||||||
<Button variant="outline" onClick={() => setSteps(CEFR_STEPS)}>
|
<Separator />
|
||||||
CEFR
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
</Button>
|
Copy this grading system to other entities
|
||||||
<Button variant="outline" onClick={() => setSteps(GENERAL_STEPS)}>
|
</label>
|
||||||
General English
|
<Select
|
||||||
</Button>
|
options={entities.map((e) => ({ value: e.id, label: e.label }))}
|
||||||
<Button variant="outline" onClick={() => setSteps(IELTS_STEPS)}>
|
onChange={(e) =>
|
||||||
IELTS
|
!e
|
||||||
</Button>
|
? setOtherEntities([])
|
||||||
<Button variant="outline" onClick={() => setSteps(TOFEL_STEPS)}>
|
: setOtherEntities(e.map((o) => o.value!))
|
||||||
TOFEL iBT
|
}
|
||||||
</Button>
|
isMulti
|
||||||
</div>
|
/>
|
||||||
|
<Button
|
||||||
|
onClick={applyToOtherEntities}
|
||||||
|
isLoading={isLoading}
|
||||||
|
disabled={isLoading || otherEntities.length === 0}
|
||||||
|
variant="outline"
|
||||||
|
>
|
||||||
|
Copy to {otherEntities.length} other entities
|
||||||
|
</Button>
|
||||||
|
<Separator />
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
{steps.map((step, index) => (
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
<>
|
Preset Systems
|
||||||
<div className="flex items-center gap-4">
|
</label>
|
||||||
<div className="grid grid-cols-3 gap-4 w-full" key={step.min}>
|
<div className="grid grid-cols-4 gap-4">
|
||||||
<Input
|
<Button variant="outline" onClick={() => setSteps(CEFR_STEPS)}>
|
||||||
label="Min. Percentage"
|
CEFR
|
||||||
value={step.min}
|
</Button>
|
||||||
type="number"
|
<Button variant="outline" onClick={() => setSteps(GENERAL_STEPS)}>
|
||||||
disabled={index === 0 || isLoading}
|
General English
|
||||||
onChange={(e) => setSteps((prev) => prev.map((x, i) => (i === index ? { ...x, min: parseInt(e) } : x)))}
|
</Button>
|
||||||
name="min"
|
<Button variant="outline" onClick={() => setSteps(IELTS_STEPS)}>
|
||||||
/>
|
IELTS
|
||||||
<Input
|
</Button>
|
||||||
label="Grade"
|
<Button variant="outline" onClick={() => setSteps(TOFEL_STEPS)}>
|
||||||
value={step.label}
|
TOFEL iBT
|
||||||
type="text"
|
</Button>
|
||||||
disabled={isLoading}
|
</div>
|
||||||
onChange={(e) => setSteps((prev) => prev.map((x, i) => (i === index ? { ...x, label: e } : x)))}
|
|
||||||
name="min"
|
|
||||||
/>
|
|
||||||
<Input
|
|
||||||
label="Max. Percentage"
|
|
||||||
value={step.max}
|
|
||||||
type="number"
|
|
||||||
disabled={index === steps.length - 1 || isLoading}
|
|
||||||
onChange={(e) => setSteps((prev) => prev.map((x, i) => (i === index ? { ...x, max: parseInt(e) } : x)))}
|
|
||||||
name="max"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
{index !== 0 && index !== steps.length - 1 && (
|
|
||||||
<button
|
|
||||||
disabled={isLoading}
|
|
||||||
className="pt-9 text-xl group"
|
|
||||||
onClick={() => setSteps((prev) => prev.filter((_, i) => i !== index))}>
|
|
||||||
<div className="w-full h-full flex items-center justify-center group-hover:bg-neutral-200 rounded-full p-3 transition ease-in-out duration-300">
|
|
||||||
<BsTrash />
|
|
||||||
</div>
|
|
||||||
</button>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{index < steps.length - 1 && (
|
{steps.map((step, index) => (
|
||||||
<Button
|
<GradingRowMemo
|
||||||
className="w-full flex items-center justify-center"
|
key={index}
|
||||||
disabled={isLoading}
|
min={step.min}
|
||||||
onClick={() => {
|
max={step.max}
|
||||||
const item = { min: steps[index === 0 ? 0 : index - 1].max + 1, max: steps[index + 1].min - 1, label: "" };
|
label={step.label}
|
||||||
setSteps((prev) => [...prev.slice(0, index + 1), item, ...prev.slice(index + 1, steps.length)]);
|
index={index}
|
||||||
}}>
|
isLoading={isLoading}
|
||||||
<BsPlusCircle />
|
isLast={index === steps.length - 1}
|
||||||
</Button>
|
setSteps={setSteps}
|
||||||
)}
|
addRow={addRow}
|
||||||
</>
|
/>
|
||||||
))}
|
))}
|
||||||
|
|
||||||
<Button onClick={saveGradingSystem} isLoading={isLoading} disabled={isLoading} className="mt-8">
|
<Button
|
||||||
Save Grading System
|
onClick={saveGradingSystem}
|
||||||
</Button>
|
isLoading={isLoading}
|
||||||
</div>
|
disabled={isLoading}
|
||||||
);
|
className="mt-8"
|
||||||
|
>
|
||||||
|
Save Changes to entities
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,5 @@
|
|||||||
import Button from "@/components/Low/Button";
|
import Button from "@/components/Low/Button";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import { capitalize, uniqBy } from "lodash";
|
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
import { useFilePicker } from "use-file-picker";
|
import { useFilePicker } from "use-file-picker";
|
||||||
|
|||||||
@@ -15,170 +15,210 @@ import { findBy, mapBy } from "@/utils";
|
|||||||
import useEntitiesCodes from "@/hooks/useEntitiesCodes";
|
import useEntitiesCodes from "@/hooks/useEntitiesCodes";
|
||||||
import Table from "@/components/High/Table";
|
import Table from "@/components/High/Table";
|
||||||
|
|
||||||
type TableData = Code & { entity?: EntityWithRoles, creator?: User }
|
type TableData = Code & { entity?: EntityWithRoles; creator?: User };
|
||||||
const columnHelper = createColumnHelper<TableData>();
|
const columnHelper = createColumnHelper<TableData>();
|
||||||
|
|
||||||
export default function CodeList({ user, entities, canDeleteCodes }
|
export default function CodeList({
|
||||||
: { user: User, entities: EntityWithRoles[], canDeleteCodes?: boolean }) {
|
user,
|
||||||
const [selectedCodes, setSelectedCodes] = useState<string[]>([]);
|
entities,
|
||||||
|
canDeleteCodes,
|
||||||
|
}: {
|
||||||
|
user: User;
|
||||||
|
entities: EntityWithRoles[];
|
||||||
|
canDeleteCodes?: boolean;
|
||||||
|
}) {
|
||||||
|
const [selectedCodes, setSelectedCodes] = useState<string[]>([]);
|
||||||
|
|
||||||
const entityIDs = useMemo(() => mapBy(entities, 'id'), [entities])
|
const entityIDs = useMemo(() => mapBy(entities, "id"), [entities]);
|
||||||
|
|
||||||
const { users } = useUsers();
|
const { users } = useUsers();
|
||||||
const { codes, reload } = useEntitiesCodes(isAdmin(user) ? undefined : entityIDs)
|
const { codes, reload, isLoading } = useEntitiesCodes(
|
||||||
|
isAdmin(user) ? undefined : entityIDs
|
||||||
|
);
|
||||||
|
|
||||||
const data: TableData[] = useMemo(() => codes.map((code) => ({
|
const data: TableData[] = useMemo(
|
||||||
...code,
|
() =>
|
||||||
entity: findBy(entities, 'id', code.entity),
|
codes.map((code) => ({
|
||||||
creator: findBy(users, 'id', code.creator)
|
...code,
|
||||||
})) as TableData[], [codes, entities, users])
|
entity: findBy(entities, "id", code.entity),
|
||||||
|
creator: findBy(users, "id", code.creator),
|
||||||
|
})) as TableData[],
|
||||||
|
[codes, entities, users]
|
||||||
|
);
|
||||||
|
|
||||||
const toggleCode = (id: string) => {
|
const toggleCode = (id: string) => {
|
||||||
setSelectedCodes((prev) => (prev.includes(id) ? prev.filter((x) => x !== id) : [...prev, id]));
|
setSelectedCodes((prev) =>
|
||||||
};
|
prev.includes(id) ? prev.filter((x) => x !== id) : [...prev, id]
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
// const toggleAllCodes = (checked: boolean) => {
|
// const toggleAllCodes = (checked: boolean) => {
|
||||||
// if (checked) return setSelectedCodes(visibleRows.filter((x) => !x.userId).map((x) => x.code));
|
// if (checked) return setSelectedCodes(visibleRows.filter((x) => !x.userId).map((x) => x.code));
|
||||||
|
|
||||||
// return setSelectedCodes([]);
|
// return setSelectedCodes([]);
|
||||||
// };
|
// };
|
||||||
|
|
||||||
const deleteCodes = async (codes: string[]) => {
|
const deleteCodes = async (codes: string[]) => {
|
||||||
if (!canDeleteCodes) return
|
if (!canDeleteCodes) return;
|
||||||
if (!confirm(`Are you sure you want to delete these ${codes.length} code(s)?`)) return;
|
if (
|
||||||
|
!confirm(`Are you sure you want to delete these ${codes.length} code(s)?`)
|
||||||
|
)
|
||||||
|
return;
|
||||||
|
|
||||||
const params = new URLSearchParams();
|
const params = new URLSearchParams();
|
||||||
codes.forEach((code) => params.append("code", code));
|
codes.forEach((code) => params.append("code", code));
|
||||||
|
|
||||||
axios
|
axios
|
||||||
.delete(`/api/code?${params.toString()}`)
|
.delete(`/api/code?${params.toString()}`)
|
||||||
.then(() => {
|
.then(() => {
|
||||||
toast.success(`Deleted the codes!`);
|
toast.success(`Deleted the codes!`);
|
||||||
setSelectedCodes([]);
|
setSelectedCodes([]);
|
||||||
})
|
})
|
||||||
.catch((reason) => {
|
.catch((reason) => {
|
||||||
if (reason.response.status === 404) {
|
if (reason.response.status === 404) {
|
||||||
toast.error("Code not found!");
|
toast.error("Code not found!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (reason.response.status === 403) {
|
if (reason.response.status === 403) {
|
||||||
toast.error("You do not have permission to delete this code!");
|
toast.error("You do not have permission to delete this code!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
toast.error("Something went wrong, please try again later.");
|
toast.error("Something went wrong, please try again later.");
|
||||||
})
|
})
|
||||||
.finally(reload);
|
.finally(reload);
|
||||||
};
|
};
|
||||||
|
|
||||||
const deleteCode = async (code: Code) => {
|
const deleteCode = async (code: Code) => {
|
||||||
if (!canDeleteCodes) return
|
if (!canDeleteCodes) return;
|
||||||
if (!confirm(`Are you sure you want to delete this "${code.code}" code?`)) return;
|
if (!confirm(`Are you sure you want to delete this "${code.code}" code?`))
|
||||||
|
return;
|
||||||
|
|
||||||
axios
|
axios
|
||||||
.delete(`/api/code/${code.code}`)
|
.delete(`/api/code/${code.code}`)
|
||||||
.then(() => toast.success(`Deleted the "${code.code}" exam`))
|
.then(() => toast.success(`Deleted the "${code.code}" exam`))
|
||||||
.catch((reason) => {
|
.catch((reason) => {
|
||||||
if (reason.response.status === 404) {
|
if (reason.response.status === 404) {
|
||||||
toast.error("Code not found!");
|
toast.error("Code not found!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (reason.response.status === 403) {
|
if (reason.response.status === 403) {
|
||||||
toast.error("You do not have permission to delete this code!");
|
toast.error("You do not have permission to delete this code!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
toast.error("Something went wrong, please try again later.");
|
toast.error("Something went wrong, please try again later.");
|
||||||
})
|
})
|
||||||
.finally(reload);
|
.finally(reload);
|
||||||
};
|
};
|
||||||
|
|
||||||
const defaultColumns = [
|
const defaultColumns = [
|
||||||
columnHelper.accessor("code", {
|
columnHelper.accessor("code", {
|
||||||
id: "codeCheckbox",
|
id: "codeCheckbox",
|
||||||
enableSorting: false,
|
enableSorting: false,
|
||||||
header: () => (""),
|
header: () => "",
|
||||||
cell: (info) =>
|
cell: (info) =>
|
||||||
!info.row.original.userId ? (
|
!info.row.original.userId ? (
|
||||||
<Checkbox isChecked={selectedCodes.includes(info.getValue())} onChange={() => toggleCode(info.getValue())}>
|
<Checkbox
|
||||||
{""}
|
isChecked={selectedCodes.includes(info.getValue())}
|
||||||
</Checkbox>
|
onChange={() => toggleCode(info.getValue())}
|
||||||
) : null,
|
>
|
||||||
}),
|
{""}
|
||||||
columnHelper.accessor("code", {
|
</Checkbox>
|
||||||
header: "Code",
|
) : null,
|
||||||
cell: (info) => info.getValue(),
|
}),
|
||||||
}),
|
columnHelper.accessor("code", {
|
||||||
columnHelper.accessor("creationDate", {
|
header: "Code",
|
||||||
header: "Creation Date",
|
cell: (info) => info.getValue(),
|
||||||
cell: (info) => (info.getValue() ? moment(info.getValue()).format("DD/MM/YYYY") : "N/A"),
|
}),
|
||||||
}),
|
columnHelper.accessor("creationDate", {
|
||||||
columnHelper.accessor("email", {
|
header: "Creation Date",
|
||||||
header: "E-mail",
|
cell: (info) =>
|
||||||
cell: (info) => info.getValue() || "N/A",
|
info.getValue() ? moment(info.getValue()).format("DD/MM/YYYY") : "N/A",
|
||||||
}),
|
}),
|
||||||
columnHelper.accessor("creator", {
|
columnHelper.accessor("email", {
|
||||||
header: "Creator",
|
header: "E-mail",
|
||||||
cell: (info) => info.getValue() ? `${info.getValue().name} (${USER_TYPE_LABELS[info.getValue().type]})` : "N/A",
|
cell: (info) => info.getValue() || "N/A",
|
||||||
}),
|
}),
|
||||||
columnHelper.accessor("entity", {
|
columnHelper.accessor("creator", {
|
||||||
header: "Entity",
|
header: "Creator",
|
||||||
cell: (info) => info.getValue()?.label || "N/A",
|
cell: (info) =>
|
||||||
}),
|
info.getValue()
|
||||||
columnHelper.accessor("userId", {
|
? `${info.getValue().name} (${
|
||||||
header: "Availability",
|
USER_TYPE_LABELS[info.getValue().type]
|
||||||
cell: (info) =>
|
})`
|
||||||
info.getValue() ? (
|
: "N/A",
|
||||||
<span className="flex gap-1 items-center text-mti-green">
|
}),
|
||||||
<div className="w-2 h-2 rounded-full bg-mti-green" /> In Use
|
columnHelper.accessor("entity", {
|
||||||
</span>
|
header: "Entity",
|
||||||
) : (
|
cell: (info) => info.getValue()?.label || "N/A",
|
||||||
<span className="flex gap-1 items-center text-mti-red">
|
}),
|
||||||
<div className="w-2 h-2 rounded-full bg-mti-red" /> Unused
|
columnHelper.accessor("userId", {
|
||||||
</span>
|
header: "Availability",
|
||||||
),
|
cell: (info) =>
|
||||||
}),
|
info.getValue() ? (
|
||||||
{
|
<span className="flex gap-1 items-center text-mti-green">
|
||||||
header: "",
|
<div className="w-2 h-2 rounded-full bg-mti-green" /> In Use
|
||||||
id: "actions",
|
</span>
|
||||||
cell: ({ row }: { row: { original: Code } }) => {
|
) : (
|
||||||
return (
|
<span className="flex gap-1 items-center text-mti-red">
|
||||||
<div className="flex gap-4">
|
<div className="w-2 h-2 rounded-full bg-mti-red" /> Unused
|
||||||
{canDeleteCodes && !row.original.userId && (
|
</span>
|
||||||
<div data-tip="Delete" className="cursor-pointer tooltip" onClick={() => deleteCode(row.original)}>
|
),
|
||||||
<BsTrash className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
}),
|
||||||
</div>
|
{
|
||||||
)}
|
header: "",
|
||||||
</div>
|
id: "actions",
|
||||||
);
|
cell: ({ row }: { row: { original: Code } }) => {
|
||||||
},
|
return (
|
||||||
},
|
<div className="flex gap-4">
|
||||||
];
|
{canDeleteCodes && !row.original.userId && (
|
||||||
|
<div
|
||||||
|
data-tip="Delete"
|
||||||
|
className="cursor-pointer tooltip"
|
||||||
|
onClick={() => deleteCode(row.original)}
|
||||||
|
>
|
||||||
|
<BsTrash className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className="flex items-center justify-between pb-4 pt-1">
|
<div className="flex items-center justify-between pb-4 pt-1">
|
||||||
{canDeleteCodes && (
|
{canDeleteCodes && (
|
||||||
<div className="flex gap-4 items-center w-full justify-end">
|
<div className="flex gap-4 items-center w-full justify-end">
|
||||||
<span>{selectedCodes.length} code(s) selected</span>
|
<span>{selectedCodes.length} code(s) selected</span>
|
||||||
<Button
|
<Button
|
||||||
disabled={selectedCodes.length === 0}
|
disabled={selectedCodes.length === 0}
|
||||||
variant="outline"
|
variant="outline"
|
||||||
color="red"
|
color="red"
|
||||||
className="!py-1 px-10"
|
className="!py-1 px-10"
|
||||||
onClick={() => deleteCodes(selectedCodes)}>
|
onClick={() => deleteCodes(selectedCodes)}
|
||||||
Delete
|
>
|
||||||
</Button>
|
Delete
|
||||||
</div>
|
</Button>
|
||||||
)}
|
</div>
|
||||||
</div>
|
)}
|
||||||
<Table<TableData>
|
</div>
|
||||||
data={data}
|
<Table<TableData>
|
||||||
columns={defaultColumns}
|
data={data}
|
||||||
searchFields={[["code"], ["email"], ["entity", "label"], ["creator", "name"], ['creator', 'type']]}
|
columns={defaultColumns}
|
||||||
/>
|
isLoading={isLoading}
|
||||||
</>
|
searchFields={[
|
||||||
);
|
["code"],
|
||||||
|
["email"],
|
||||||
|
["entity", "label"],
|
||||||
|
["creator", "name"],
|
||||||
|
["creator", "type"],
|
||||||
|
]}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,346 +4,391 @@ import useExams from "@/hooks/useExams";
|
|||||||
import useUsers from "@/hooks/useUsers";
|
import useUsers from "@/hooks/useUsers";
|
||||||
import { Module } from "@/interfaces";
|
import { Module } from "@/interfaces";
|
||||||
import { Exam } from "@/interfaces/exam";
|
import { Exam } from "@/interfaces/exam";
|
||||||
import { Type, User } from "@/interfaces/user";
|
import { User } from "@/interfaces/user";
|
||||||
import useExamStore from "@/stores/exam";
|
import useExamStore from "@/stores/exam";
|
||||||
import { getExamById } from "@/utils/exams";
|
import { getExamById } from "@/utils/exams";
|
||||||
import { countExercises } from "@/utils/moduleUtils";
|
import { countExercises } from "@/utils/moduleUtils";
|
||||||
import { createColumnHelper, flexRender, getCoreRowModel, useReactTable } from "@tanstack/react-table";
|
import {
|
||||||
|
createColumnHelper,
|
||||||
|
flexRender,
|
||||||
|
getCoreRowModel,
|
||||||
|
useReactTable,
|
||||||
|
} from "@tanstack/react-table";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import clsx from "clsx";
|
import { capitalize } from "lodash";
|
||||||
import { capitalize, uniq } from "lodash";
|
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { BsBan, BsBanFill, BsCheck, BsCircle, BsPencil, BsStop, BsTrash, BsUpload, BsX } from "react-icons/bs";
|
import { BsPencil, BsTrash, BsUpload } from "react-icons/bs";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
import { useListSearch } from "@/hooks/useListSearch";
|
import { useListSearch } from "@/hooks/useListSearch";
|
||||||
import Modal from "@/components/Modal";
|
import Modal from "@/components/Modal";
|
||||||
import { checkAccess } from "@/utils/permissions";
|
import { checkAccess, findAllowedEntities } from "@/utils/permissions";
|
||||||
import useGroups from "@/hooks/useGroups";
|
|
||||||
import Button from "@/components/Low/Button";
|
import Button from "@/components/Low/Button";
|
||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
import { FiEdit, FiArrowRight } from 'react-icons/fi';
|
|
||||||
import { HiArrowRight } from "react-icons/hi";
|
|
||||||
import { BiEdit } from "react-icons/bi";
|
import { BiEdit } from "react-icons/bi";
|
||||||
|
import { findBy, mapBy } from "@/utils";
|
||||||
|
|
||||||
const searchFields = [["module"], ["id"], ["createdBy"]];
|
const searchFields = [["module"], ["id"], ["createdBy"]];
|
||||||
|
|
||||||
const CLASSES: { [key in Module]: string } = {
|
const CLASSES: { [key in Module]: string } = {
|
||||||
reading: "text-ielts-reading",
|
reading: "text-ielts-reading",
|
||||||
listening: "text-ielts-listening",
|
listening: "text-ielts-listening",
|
||||||
speaking: "text-ielts-speaking",
|
speaking: "text-ielts-speaking",
|
||||||
writing: "text-ielts-writing",
|
writing: "text-ielts-writing",
|
||||||
level: "text-ielts-level",
|
level: "text-ielts-level",
|
||||||
};
|
};
|
||||||
|
|
||||||
const columnHelper = createColumnHelper<Exam>();
|
const columnHelper = createColumnHelper<Exam>();
|
||||||
|
|
||||||
const ExamOwnerSelector = ({ options, exam, onSave }: { options: User[]; exam: Exam; onSave: (owners: string[]) => void }) => {
|
export default function ExamList({
|
||||||
const [owners, setOwners] = useState(exam.owners || []);
|
user,
|
||||||
|
entities,
|
||||||
|
}: {
|
||||||
|
user: User;
|
||||||
|
entities: EntityWithRoles[];
|
||||||
|
}) {
|
||||||
|
const [selectedExam, setSelectedExam] = useState<Exam>();
|
||||||
|
|
||||||
return (
|
const canViewConfidentialEntities = useMemo(
|
||||||
<div className="w-full flex flex-col gap-4">
|
() =>
|
||||||
<div className="grid grid-cols-4 mt-4">
|
mapBy(
|
||||||
{options.map((c) => (
|
findAllowedEntities(user, entities, "view_confidential_exams"),
|
||||||
<Button
|
"id"
|
||||||
variant={owners.includes(c.id) ? "solid" : "outline"}
|
),
|
||||||
onClick={() => setOwners((prev) => (prev.includes(c.id) ? prev.filter((x) => x !== c.id) : [...prev, c.id]))}
|
[user, entities]
|
||||||
className="max-w-[200px] w-full"
|
);
|
||||||
key={c.id}>
|
|
||||||
{c.name}
|
|
||||||
</Button>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
<Button onClick={() => onSave(owners)} className="w-full max-w-[200px] self-end">
|
|
||||||
Save
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
export default function ExamList({ user, entities }: { user: User; entities: EntityWithRoles[]; }) {
|
const { exams, reload, isLoading } = useExams();
|
||||||
const [selectedExam, setSelectedExam] = useState<Exam>();
|
const { users } = useUsers();
|
||||||
|
// Pass this permission filter to the backend later
|
||||||
|
const filteredExams = useMemo(
|
||||||
|
() =>
|
||||||
|
["admin", "developer"].includes(user?.type)
|
||||||
|
? exams
|
||||||
|
: exams.filter((item) => {
|
||||||
|
if (
|
||||||
|
item.access === "confidential" &&
|
||||||
|
!canViewConfidentialEntities.find((x) =>
|
||||||
|
(item.entities ?? []).includes(x)
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return false;
|
||||||
|
return true;
|
||||||
|
}),
|
||||||
|
[canViewConfidentialEntities, exams, user?.type]
|
||||||
|
);
|
||||||
|
|
||||||
const { exams, reload } = useExams();
|
const parsedExams = useMemo(() => {
|
||||||
const { users } = useUsers();
|
return filteredExams.map((exam) => {
|
||||||
const { groups } = useGroups({ admin: user?.id, userType: user?.type });
|
if (exam.createdBy) {
|
||||||
|
const user = users.find((u) => u.id === exam.createdBy);
|
||||||
|
if (!user) return exam;
|
||||||
|
|
||||||
const filteredExams = useMemo(() => exams.filter((e) => {
|
return {
|
||||||
if (!e.private) return true
|
...exam,
|
||||||
return (e.owners || []).includes(user?.id || "")
|
createdBy: user.type === "developer" ? "system" : user.name,
|
||||||
}), [exams, user?.id])
|
};
|
||||||
|
}
|
||||||
|
|
||||||
const filteredCorporates = useMemo(() => {
|
return exam;
|
||||||
const participantsAndAdmins = uniq(groups.flatMap((x) => [...x.participants, x.admin])).filter((x) => x !== user?.id);
|
});
|
||||||
return users.filter((x) => participantsAndAdmins.includes(x.id) && x.type === "corporate");
|
}, [filteredExams, users]);
|
||||||
}, [users, groups, user]);
|
|
||||||
|
|
||||||
const parsedExams = useMemo(() => {
|
const { rows: filteredRows, renderSearch } = useListSearch<Exam>(
|
||||||
return filteredExams.map((exam) => {
|
searchFields,
|
||||||
if (exam.createdBy) {
|
parsedExams
|
||||||
const user = users.find((u) => u.id === exam.createdBy);
|
);
|
||||||
if (!user) return exam;
|
|
||||||
|
|
||||||
return {
|
const dispatch = useExamStore((state) => state.dispatch);
|
||||||
...exam,
|
|
||||||
createdBy: user.type === "developer" ? "system" : user.name,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
return exam;
|
const router = useRouter();
|
||||||
});
|
|
||||||
}, [filteredExams, users]);
|
|
||||||
|
|
||||||
const { rows: filteredRows, renderSearch } = useListSearch<Exam>(searchFields, parsedExams);
|
const loadExam = async (module: Module, examId: string) => {
|
||||||
|
const exam = await getExamById(module, examId.trim());
|
||||||
|
if (!exam) {
|
||||||
|
toast.error(
|
||||||
|
"Unknown Exam ID! Please make sure you selected the right module and entered the right exam ID",
|
||||||
|
{
|
||||||
|
toastId: "invalid-exam-id",
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
const dispatch = useExamStore((state) => state.dispatch);
|
return;
|
||||||
|
}
|
||||||
|
dispatch({
|
||||||
|
type: "INIT_EXAM",
|
||||||
|
payload: { exams: [exam], modules: [module] },
|
||||||
|
});
|
||||||
|
|
||||||
const router = useRouter();
|
router.push("/exam");
|
||||||
|
};
|
||||||
|
|
||||||
const loadExam = async (module: Module, examId: string) => {
|
/*
|
||||||
const exam = await getExamById(module, examId.trim());
|
const privatizeExam = async (exam: Exam) => {
|
||||||
if (!exam) {
|
if (
|
||||||
toast.error("Unknown Exam ID! Please make sure you selected the right module and entered the right exam ID", {
|
!confirm(
|
||||||
toastId: "invalid-exam-id",
|
`Are you sure you want to make this ${capitalize(exam.module)} exam ${
|
||||||
});
|
exam.access
|
||||||
|
}?`
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return;
|
||||||
|
|
||||||
return;
|
axios
|
||||||
}
|
.patch(`/api/exam/${exam.module}/${exam.id}`, { private: !exam.private })
|
||||||
dispatch({ type: "INIT_EXAM", payload: { exams: [exam], modules: [module] } })
|
.then(() => toast.success(`Updated the "${exam.id}" exam`))
|
||||||
|
.catch((reason) => {
|
||||||
|
if (reason.response.status === 404) {
|
||||||
|
toast.error("Exam not found!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
router.push("/exam");
|
if (reason.response.status === 403) {
|
||||||
};
|
toast.error("You do not have permission to update this exam!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const privatizeExam = async (exam: Exam) => {
|
toast.error("Something went wrong, please try again later.");
|
||||||
if (!confirm(`Are you sure you want to make this ${capitalize(exam.module)} exam ${exam.private ? "public" : "private"}?`)) return;
|
})
|
||||||
|
.finally(reload);
|
||||||
|
};
|
||||||
|
*/
|
||||||
|
|
||||||
axios
|
const deleteExam = async (exam: Exam) => {
|
||||||
.patch(`/api/exam/${exam.module}/${exam.id}`, { private: !exam.private })
|
if (
|
||||||
.then(() => toast.success(`Updated the "${exam.id}" exam`))
|
!confirm(
|
||||||
.catch((reason) => {
|
`Are you sure you want to delete this ${capitalize(exam.module)} exam?`
|
||||||
if (reason.response.status === 404) {
|
)
|
||||||
toast.error("Exam not found!");
|
)
|
||||||
return;
|
return;
|
||||||
}
|
|
||||||
|
|
||||||
if (reason.response.status === 403) {
|
axios
|
||||||
toast.error("You do not have permission to update this exam!");
|
.delete(`/api/exam/${exam.module}/${exam.id}`)
|
||||||
return;
|
.then(() => toast.success(`Deleted the "${exam.id}" exam`))
|
||||||
}
|
.catch((reason) => {
|
||||||
|
if (reason.response.status === 404) {
|
||||||
|
toast.error("Exam not found!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
toast.error("Something went wrong, please try again later.");
|
if (reason.response.status === 403) {
|
||||||
})
|
toast.error("You do not have permission to delete this exam!");
|
||||||
.finally(reload);
|
return;
|
||||||
};
|
}
|
||||||
|
|
||||||
const updateExam = async (exam: Exam, body: object) => {
|
toast.error("Something went wrong, please try again later.");
|
||||||
if (!confirm(`Are you sure you want to update this ${capitalize(exam.module)} exam?`)) return;
|
})
|
||||||
|
.finally(reload);
|
||||||
|
};
|
||||||
|
|
||||||
axios
|
const getTotalExercises = (exam: Exam) => {
|
||||||
.patch(`/api/exam/${exam.module}/${exam.id}`, body)
|
if (
|
||||||
.then(() => toast.success(`Updated the "${exam.id}" exam`))
|
exam.module === "reading" ||
|
||||||
.catch((reason) => {
|
exam.module === "listening" ||
|
||||||
if (reason.response.status === 404) {
|
exam.module === "level"
|
||||||
toast.error("Exam not found!");
|
) {
|
||||||
return;
|
return countExercises((exam.parts ?? []).flatMap((x) => x.exercises));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (reason.response.status === 403) {
|
return countExercises(exam.exercises);
|
||||||
toast.error("You do not have permission to update this exam!");
|
};
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
toast.error("Something went wrong, please try again later.");
|
const defaultColumns = [
|
||||||
})
|
columnHelper.accessor("id", {
|
||||||
.finally(reload)
|
header: "ID",
|
||||||
.finally(() => setSelectedExam(undefined));
|
cell: (info) => info.getValue(),
|
||||||
};
|
}),
|
||||||
|
columnHelper.accessor("module", {
|
||||||
|
header: "Module",
|
||||||
|
cell: (info) => (
|
||||||
|
<span className={CLASSES[info.getValue()]}>
|
||||||
|
{capitalize(info.getValue())}
|
||||||
|
</span>
|
||||||
|
),
|
||||||
|
}),
|
||||||
|
columnHelper.accessor((x) => getTotalExercises(x), {
|
||||||
|
header: "Exercises",
|
||||||
|
cell: (info) => info.getValue(),
|
||||||
|
}),
|
||||||
|
columnHelper.accessor("minTimer", {
|
||||||
|
header: "Timer",
|
||||||
|
cell: (info) => <>{info.getValue()} minute(s)</>,
|
||||||
|
}),
|
||||||
|
columnHelper.accessor("access", {
|
||||||
|
header: "Access",
|
||||||
|
cell: (info) => <span>{capitalize(info.getValue())}</span>,
|
||||||
|
}),
|
||||||
|
columnHelper.accessor("createdAt", {
|
||||||
|
header: "Created At",
|
||||||
|
cell: (info) => {
|
||||||
|
const value = info.getValue();
|
||||||
|
if (value) {
|
||||||
|
return new Date(value).toLocaleDateString();
|
||||||
|
}
|
||||||
|
|
||||||
const deleteExam = async (exam: Exam) => {
|
return null;
|
||||||
if (!confirm(`Are you sure you want to delete this ${capitalize(exam.module)} exam?`)) return;
|
},
|
||||||
|
}),
|
||||||
|
columnHelper.accessor("createdBy", {
|
||||||
|
header: "Created By",
|
||||||
|
cell: (info) =>
|
||||||
|
!info.getValue()
|
||||||
|
? "System"
|
||||||
|
: findBy(users, "id", info.getValue())?.name || "N/A",
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
header: "",
|
||||||
|
id: "actions",
|
||||||
|
cell: ({ row }: { row: { original: Exam } }) => {
|
||||||
|
return (
|
||||||
|
<div className="flex gap-4">
|
||||||
|
{(row.original.owners?.includes(user.id) ||
|
||||||
|
checkAccess(user, ["admin", "developer"])) && (
|
||||||
|
<>
|
||||||
|
{checkAccess(user, [
|
||||||
|
"admin",
|
||||||
|
"developer",
|
||||||
|
"mastercorporate",
|
||||||
|
]) && (
|
||||||
|
<button
|
||||||
|
data-tip="Edit exam"
|
||||||
|
onClick={() => setSelectedExam(row.original)}
|
||||||
|
className="cursor-pointer tooltip"
|
||||||
|
>
|
||||||
|
<BsPencil />
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
<button
|
||||||
|
data-tip="Load exam"
|
||||||
|
className="cursor-pointer tooltip"
|
||||||
|
onClick={async () =>
|
||||||
|
await loadExam(row.original.module, row.original.id)
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<BsUpload className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
|
</button>
|
||||||
|
{PERMISSIONS.examManagement.delete.includes(user.type) && (
|
||||||
|
<div
|
||||||
|
data-tip="Delete"
|
||||||
|
className="cursor-pointer tooltip"
|
||||||
|
onClick={() => deleteExam(row.original)}
|
||||||
|
>
|
||||||
|
<BsTrash className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
axios
|
const table = useReactTable({
|
||||||
.delete(`/api/exam/${exam.module}/${exam.id}`)
|
data: filteredRows,
|
||||||
.then(() => toast.success(`Deleted the "${exam.id}" exam`))
|
columns: defaultColumns,
|
||||||
.catch((reason) => {
|
getCoreRowModel: getCoreRowModel(),
|
||||||
if (reason.response.status === 404) {
|
});
|
||||||
toast.error("Exam not found!");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (reason.response.status === 403) {
|
const handleExamEdit = () => {
|
||||||
toast.error("You do not have permission to delete this exam!");
|
router.push(
|
||||||
return;
|
`/generation?id=${selectedExam!.id}&module=${selectedExam!.module}`
|
||||||
}
|
);
|
||||||
|
};
|
||||||
|
|
||||||
toast.error("Something went wrong, please try again later.");
|
return (
|
||||||
})
|
<div className="flex flex-col gap-4 w-full h-full">
|
||||||
.finally(reload);
|
{renderSearch()}
|
||||||
};
|
<Modal
|
||||||
|
isOpen={!!selectedExam}
|
||||||
|
onClose={() => setSelectedExam(undefined)}
|
||||||
|
maxWidth="max-w-xl"
|
||||||
|
>
|
||||||
|
{!!selectedExam ? (
|
||||||
|
<>
|
||||||
|
<div className="p-6">
|
||||||
|
<div className="mb-6">
|
||||||
|
<div className="flex items-center gap-2 mb-4">
|
||||||
|
<BiEdit className="w-5 h-5 text-gray-600" />
|
||||||
|
<span className="text-gray-600 font-medium">
|
||||||
|
Ready to Edit
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
|
||||||
const getTotalExercises = (exam: Exam) => {
|
<div className="bg-gray-50 rounded-lg p-4 mb-3">
|
||||||
if (exam.module === "reading" || exam.module === "listening" || exam.module === "level") {
|
<p className="font-medium mb-1">Exam ID: {selectedExam.id}</p>
|
||||||
return countExercises(exam.parts.flatMap((x) => x.exercises));
|
</div>
|
||||||
}
|
|
||||||
|
|
||||||
return countExercises(exam.exercises);
|
<p className="text-gray-500 text-sm">
|
||||||
};
|
Click 'Next' to proceed to the exam editor.
|
||||||
|
</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
const defaultColumns = [
|
<div className="flex justify-between gap-4 mt-8">
|
||||||
columnHelper.accessor("id", {
|
<Button
|
||||||
header: "ID",
|
color="purple"
|
||||||
cell: (info) => info.getValue(),
|
variant="outline"
|
||||||
}),
|
onClick={() => setSelectedExam(undefined)}
|
||||||
columnHelper.accessor("module", {
|
className="w-32"
|
||||||
header: "Module",
|
>
|
||||||
cell: (info) => <span className={CLASSES[info.getValue()]}>{capitalize(info.getValue())}</span>,
|
Cancel
|
||||||
}),
|
</Button>
|
||||||
columnHelper.accessor((x) => getTotalExercises(x), {
|
<Button
|
||||||
header: "Exercises",
|
color="purple"
|
||||||
cell: (info) => info.getValue(),
|
onClick={handleExamEdit}
|
||||||
}),
|
className="w-32 text-white flex items-center justify-center gap-2"
|
||||||
columnHelper.accessor("minTimer", {
|
>
|
||||||
header: "Timer",
|
Proceed
|
||||||
cell: (info) => <>{info.getValue()} minute(s)</>,
|
</Button>
|
||||||
}),
|
</div>
|
||||||
columnHelper.accessor("private", {
|
</div>
|
||||||
header: "Private",
|
{/*<ExamOwnerSelector options={filteredCorporates} exam={selectedExam} onSave={(owners) => updateExam(selectedExam, { owners })} />*/}
|
||||||
cell: (info) => <span className="w-full flex items-center justify-center">{!info.getValue() ? <BsX /> : <BsCheck />}</span>,
|
</>
|
||||||
}),
|
) : (
|
||||||
columnHelper.accessor("createdAt", {
|
<div />
|
||||||
header: "Created At",
|
)}
|
||||||
cell: (info) => {
|
</Modal>
|
||||||
const value = info.getValue();
|
<table className="rounded-xl bg-mti-purple-ultralight/40 w-full">
|
||||||
if (value) {
|
<thead>
|
||||||
return new Date(value).toLocaleDateString();
|
{table.getHeaderGroups().map((headerGroup) => (
|
||||||
}
|
<tr key={headerGroup.id}>
|
||||||
|
{headerGroup.headers.map((header) => (
|
||||||
return null;
|
<th className="p-4 text-left" key={header.id}>
|
||||||
},
|
{header.isPlaceholder
|
||||||
}),
|
? null
|
||||||
columnHelper.accessor("createdBy", {
|
: flexRender(
|
||||||
header: "Created By",
|
header.column.columnDef.header,
|
||||||
cell: (info) => info.getValue(),
|
header.getContext()
|
||||||
}),
|
)}
|
||||||
{
|
</th>
|
||||||
header: "",
|
))}
|
||||||
id: "actions",
|
</tr>
|
||||||
cell: ({ row }: { row: { original: Exam } }) => {
|
))}
|
||||||
return (
|
</thead>
|
||||||
<div className="flex gap-4">
|
<tbody className="px-2">
|
||||||
{(row.original.owners?.includes(user.id) || checkAccess(user, ["admin", "developer"])) && (
|
{table.getRowModel().rows.map((row) => (
|
||||||
<>
|
<tr
|
||||||
<button
|
className="odd:bg-white even:bg-mti-purple-ultralight/40 rounded-lg py-2"
|
||||||
data-tip={row.original.private ? "Set as public" : "Set as private"}
|
key={row.id}
|
||||||
onClick={async () => await privatizeExam(row.original)}
|
>
|
||||||
className="cursor-pointer tooltip">
|
{row.getVisibleCells().map((cell) => (
|
||||||
{row.original.private ? <BsCircle /> : <BsBan />}
|
<td className="px-4 py-2" key={cell.id}>
|
||||||
</button>
|
{flexRender(cell.column.columnDef.cell, cell.getContext())}
|
||||||
{checkAccess(user, ["admin", "developer", "mastercorporate"]) && (
|
</td>
|
||||||
<button data-tip="Edit exam" onClick={() => setSelectedExam(row.original)} className="cursor-pointer tooltip">
|
))}
|
||||||
<BsPencil />
|
</tr>
|
||||||
</button>
|
))}
|
||||||
)}
|
</tbody>
|
||||||
</>
|
</table>
|
||||||
)}
|
{isLoading ? (
|
||||||
<button
|
<div className="min-h-screen flex justify-center items-start">
|
||||||
data-tip="Load exam"
|
<span className="loading loading-infinity w-32" />
|
||||||
className="cursor-pointer tooltip"
|
</div>
|
||||||
onClick={async () => await loadExam(row.original.module, row.original.id)}>
|
) : (
|
||||||
<BsUpload className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
filteredRows.length === 0 && (
|
||||||
</button>
|
<div className="w-full flex justify-center items-start">
|
||||||
{PERMISSIONS.examManagement.delete.includes(user.type) && (
|
<span className="text-xl text-gray-500">No data found...</span>
|
||||||
<div data-tip="Delete" className="cursor-pointer tooltip" onClick={() => deleteExam(row.original)}>
|
</div>
|
||||||
<BsTrash className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
)
|
||||||
</div>
|
)}
|
||||||
)}
|
</div>
|
||||||
</div>
|
);
|
||||||
);
|
|
||||||
},
|
|
||||||
},
|
|
||||||
];
|
|
||||||
|
|
||||||
const table = useReactTable({
|
|
||||||
data: filteredRows,
|
|
||||||
columns: defaultColumns,
|
|
||||||
getCoreRowModel: getCoreRowModel(),
|
|
||||||
});
|
|
||||||
|
|
||||||
const handleExamEdit = () => {
|
|
||||||
router.push(`/generation?id=${selectedExam!.id}&module=${selectedExam!.module}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div className="flex flex-col gap-4 w-full h-full">
|
|
||||||
{renderSearch()}
|
|
||||||
<Modal isOpen={!!selectedExam} onClose={() => setSelectedExam(undefined)} maxWidth="max-w-xl">
|
|
||||||
{!!selectedExam ? (
|
|
||||||
<>
|
|
||||||
<div className="p-6">
|
|
||||||
<div className="mb-6">
|
|
||||||
<div className="flex items-center gap-2 mb-4">
|
|
||||||
<BiEdit className="w-5 h-5 text-gray-600" />
|
|
||||||
<span className="text-gray-600 font-medium">Ready to Edit</span>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="bg-gray-50 rounded-lg p-4 mb-3">
|
|
||||||
<p className="font-medium mb-1">
|
|
||||||
Exam ID: {selectedExam.id}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<p className="text-gray-500 text-sm">
|
|
||||||
Click 'Next' to proceed to the exam editor.
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div className="flex justify-between gap-4 mt-8">
|
|
||||||
<Button
|
|
||||||
color="purple"
|
|
||||||
variant="outline"
|
|
||||||
onClick={() => setSelectedExam(undefined)}
|
|
||||||
className="w-32"
|
|
||||||
>
|
|
||||||
Cancel
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
color="purple"
|
|
||||||
onClick={handleExamEdit}
|
|
||||||
className="w-32 text-white flex items-center justify-center gap-2"
|
|
||||||
>
|
|
||||||
Proceed
|
|
||||||
</Button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
{/*<ExamOwnerSelector options={filteredCorporates} exam={selectedExam} onSave={(owners) => updateExam(selectedExam, { owners })} />*/}
|
|
||||||
</>
|
|
||||||
) : (
|
|
||||||
<div />
|
|
||||||
)}
|
|
||||||
</Modal>
|
|
||||||
<table className="rounded-xl bg-mti-purple-ultralight/40 w-full">
|
|
||||||
<thead>
|
|
||||||
{table.getHeaderGroups().map((headerGroup) => (
|
|
||||||
<tr key={headerGroup.id}>
|
|
||||||
{headerGroup.headers.map((header) => (
|
|
||||||
<th className="p-4 text-left" key={header.id}>
|
|
||||||
{header.isPlaceholder ? null : flexRender(header.column.columnDef.header, header.getContext())}
|
|
||||||
</th>
|
|
||||||
))}
|
|
||||||
</tr>
|
|
||||||
))}
|
|
||||||
</thead>
|
|
||||||
<tbody className="px-2">
|
|
||||||
{table.getRowModel().rows.map((row) => (
|
|
||||||
<tr className="odd:bg-white even:bg-mti-purple-ultralight/40 rounded-lg py-2" key={row.id}>
|
|
||||||
{row.getVisibleCells().map((cell) => (
|
|
||||||
<td className="px-4 py-2" key={cell.id}>
|
|
||||||
{flexRender(cell.column.columnDef.cell, cell.getContext())}
|
|
||||||
</td>
|
|
||||||
))}
|
|
||||||
</tr>
|
|
||||||
))}
|
|
||||||
</tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,31 +1,30 @@
|
|||||||
import Button from "@/components/Low/Button";
|
import Button from "@/components/Low/Button";
|
||||||
import Input from "@/components/Low/Input";
|
import Input from "@/components/Low/Input";
|
||||||
import Modal from "@/components/Modal";
|
import Modal from "@/components/Modal";
|
||||||
import useGroups from "@/hooks/useGroups";
|
import { Group, User } from "@/interfaces/user";
|
||||||
import useUsers from "@/hooks/useUsers";
|
import { createColumnHelper } from "@tanstack/react-table";
|
||||||
import { CorporateUser, Group, User } from "@/interfaces/user";
|
|
||||||
import { createColumnHelper, flexRender, getCoreRowModel, useReactTable } from "@tanstack/react-table";
|
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import { capitalize, uniq } from "lodash";
|
import { uniq } from "lodash";
|
||||||
import { useEffect, useMemo, useState } from "react";
|
import { useEffect, useMemo, useState } from "react";
|
||||||
import { BsPencil, BsQuestionCircleFill, BsTrash } from "react-icons/bs";
|
import { BsPencil, BsQuestionCircleFill, BsTrash } from "react-icons/bs";
|
||||||
import Select from "react-select";
|
import Select from "react-select";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
import readXlsxFile from "read-excel-file";
|
import readXlsxFile from "read-excel-file";
|
||||||
import { useFilePicker } from "use-file-picker";
|
import { useFilePicker } from "use-file-picker";
|
||||||
import { getUserCorporate } from "@/utils/groups";
|
import { USER_TYPE_LABELS } from "@/resources/user";
|
||||||
import { isAgentUser, isCorporateUser, USER_TYPE_LABELS } from "@/resources/user";
|
|
||||||
import { checkAccess } from "@/utils/permissions";
|
import { checkAccess } from "@/utils/permissions";
|
||||||
import usePermissions from "@/hooks/usePermissions";
|
import usePermissions from "@/hooks/usePermissions";
|
||||||
import { useListSearch } from "@/hooks/useListSearch";
|
|
||||||
import Table from "@/components/High/Table";
|
import Table from "@/components/High/Table";
|
||||||
import useEntitiesGroups from "@/hooks/useEntitiesGroups";
|
import useEntitiesGroups from "@/hooks/useEntitiesGroups";
|
||||||
import useEntitiesUsers from "@/hooks/useEntitiesUsers";
|
import useEntitiesUsers from "@/hooks/useEntitiesUsers";
|
||||||
import { WithEntity } from "@/interfaces/entity";
|
import { WithEntity } from "@/interfaces/entity";
|
||||||
|
|
||||||
const searchFields = [["name"]];
|
const searchFields = [["name"]];
|
||||||
|
|
||||||
const columnHelper = createColumnHelper<WithEntity<Group>>();
|
const columnHelper = createColumnHelper<WithEntity<Group>>();
|
||||||
const EMAIL_REGEX = new RegExp(/^[a-zA-Z0-9]+(?:\.[a-zA-Z0-9]+)*@[a-zA-Z0-9]+(?:\.[a-zA-Z0-9]+)*$/);
|
const EMAIL_REGEX = new RegExp(
|
||||||
|
/^[a-zA-Z0-9]+(?:\.[a-zA-Z0-9]+)*@[a-zA-Z0-9]+(?:\.[a-zA-Z0-9]+)*$/
|
||||||
|
);
|
||||||
|
|
||||||
interface CreateDialogProps {
|
interface CreateDialogProps {
|
||||||
user: User;
|
user: User;
|
||||||
@@ -35,9 +34,13 @@ interface CreateDialogProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const CreatePanel = ({ user, users, group, onClose }: CreateDialogProps) => {
|
const CreatePanel = ({ user, users, group, onClose }: CreateDialogProps) => {
|
||||||
const [name, setName] = useState<string | undefined>(group?.name || undefined);
|
const [name, setName] = useState<string | undefined>(
|
||||||
|
group?.name || undefined
|
||||||
|
);
|
||||||
const [admin, setAdmin] = useState<string>(group?.admin || user.id);
|
const [admin, setAdmin] = useState<string>(group?.admin || user.id);
|
||||||
const [participants, setParticipants] = useState<string[]>(group?.participants || []);
|
const [participants, setParticipants] = useState<string[]>(
|
||||||
|
group?.participants || []
|
||||||
|
);
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
|
||||||
const { openFilePicker, filesContent, clear } = useFilePicker({
|
const { openFilePicker, filesContent, clear } = useFilePicker({
|
||||||
@@ -47,9 +50,14 @@ const CreatePanel = ({ user, users, group, onClose }: CreateDialogProps) => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
const availableUsers = useMemo(() => {
|
const availableUsers = useMemo(() => {
|
||||||
if (user?.type === "teacher") return users.filter((x) => ["student"].includes(x.type));
|
if (user?.type === "teacher")
|
||||||
if (user?.type === "corporate") return users.filter((x) => ["teacher", "student"].includes(x.type));
|
return users.filter((x) => ["student"].includes(x.type));
|
||||||
if (user?.type === "mastercorporate") return users.filter((x) => ["corporate", "teacher", "student"].includes(x.type));
|
if (user?.type === "corporate")
|
||||||
|
return users.filter((x) => ["teacher", "student"].includes(x.type));
|
||||||
|
if (user?.type === "mastercorporate")
|
||||||
|
return users.filter((x) =>
|
||||||
|
["corporate", "teacher", "student"].includes(x.type)
|
||||||
|
);
|
||||||
|
|
||||||
return users;
|
return users;
|
||||||
}, [user, users]);
|
}, [user, users]);
|
||||||
@@ -64,9 +72,12 @@ const CreatePanel = ({ user, users, group, onClose }: CreateDialogProps) => {
|
|||||||
rows
|
rows
|
||||||
.map((row) => {
|
.map((row) => {
|
||||||
const [email] = row as string[];
|
const [email] = row as string[];
|
||||||
return EMAIL_REGEX.test(email) && !users.map((u) => u.email).includes(email) ? email.toString().trim() : undefined;
|
return EMAIL_REGEX.test(email) &&
|
||||||
|
!users.map((u) => u.email).includes(email)
|
||||||
|
? email.toString().trim()
|
||||||
|
: undefined;
|
||||||
})
|
})
|
||||||
.filter((x) => !!x),
|
.filter((x) => !!x)
|
||||||
);
|
);
|
||||||
|
|
||||||
if (emails.length === 0) {
|
if (emails.length === 0) {
|
||||||
@@ -76,12 +87,17 @@ const CreatePanel = ({ user, users, group, onClose }: CreateDialogProps) => {
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const emailUsers = [...new Set(emails)].map((x) => users.find((y) => y.email.toLowerCase() === x)).filter((x) => x !== undefined);
|
const emailUsers = [...new Set(emails)]
|
||||||
|
.map((x) => users.find((y) => y.email.toLowerCase() === x))
|
||||||
|
.filter((x) => x !== undefined);
|
||||||
const filteredUsers = emailUsers.filter(
|
const filteredUsers = emailUsers.filter(
|
||||||
(x) =>
|
(x) =>
|
||||||
((user.type === "developer" || user.type === "admin" || user.type === "corporate" || user.type === "mastercorporate") &&
|
((user.type === "developer" ||
|
||||||
|
user.type === "admin" ||
|
||||||
|
user.type === "corporate" ||
|
||||||
|
user.type === "mastercorporate") &&
|
||||||
(x?.type === "student" || x?.type === "teacher")) ||
|
(x?.type === "student" || x?.type === "teacher")) ||
|
||||||
(user.type === "teacher" && x?.type === "student"),
|
(user.type === "teacher" && x?.type === "student")
|
||||||
);
|
);
|
||||||
|
|
||||||
setParticipants(filteredUsers.filter((x) => !!x).map((x) => x!.id));
|
setParticipants(filteredUsers.filter((x) => !!x).map((x) => x!.id));
|
||||||
@@ -89,7 +105,7 @@ const CreatePanel = ({ user, users, group, onClose }: CreateDialogProps) => {
|
|||||||
user.type !== "teacher"
|
user.type !== "teacher"
|
||||||
? "Added all teachers and students found in the file you've provided!"
|
? "Added all teachers and students found in the file you've provided!"
|
||||||
: "Added all students found in the file you've provided!",
|
: "Added all students found in the file you've provided!",
|
||||||
{ toastId: "upload-success" },
|
{ toastId: "upload-success" }
|
||||||
);
|
);
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
});
|
});
|
||||||
@@ -100,15 +116,27 @@ const CreatePanel = ({ user, users, group, onClose }: CreateDialogProps) => {
|
|||||||
const submit = () => {
|
const submit = () => {
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
|
|
||||||
if (name !== group?.name && (name?.trim() === "Students" || name?.trim() === "Teachers" || name?.trim() === "Corporate")) {
|
if (
|
||||||
toast.error("That group name is reserved and cannot be used, please enter another one.");
|
name !== group?.name &&
|
||||||
|
(name?.trim() === "Students" ||
|
||||||
|
name?.trim() === "Teachers" ||
|
||||||
|
name?.trim() === "Corporate")
|
||||||
|
) {
|
||||||
|
toast.error(
|
||||||
|
"That group name is reserved and cannot be used, please enter another one."
|
||||||
|
);
|
||||||
setIsLoading(false);
|
setIsLoading(false);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
(group ? axios.patch : axios.post)(group ? `/api/groups/${group.id}` : "/api/groups", { name, admin, participants })
|
(group ? axios.patch : axios.post)(
|
||||||
|
group ? `/api/groups/${group.id}` : "/api/groups",
|
||||||
|
{ name, admin, participants }
|
||||||
|
)
|
||||||
.then(() => {
|
.then(() => {
|
||||||
toast.success(`Group "${name}" ${group ? "edited" : "created"} successfully`);
|
toast.success(
|
||||||
|
`Group "${name}" ${group ? "edited" : "created"} successfully`
|
||||||
|
);
|
||||||
return true;
|
return true;
|
||||||
})
|
})
|
||||||
.catch(() => {
|
.catch(() => {
|
||||||
@@ -121,30 +149,58 @@ const CreatePanel = ({ user, users, group, onClose }: CreateDialogProps) => {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const userOptions = useMemo(
|
||||||
|
() =>
|
||||||
|
availableUsers.map((x) => ({
|
||||||
|
value: x.id,
|
||||||
|
label: `${x.email} - ${x.name}`,
|
||||||
|
})),
|
||||||
|
[availableUsers]
|
||||||
|
);
|
||||||
|
|
||||||
|
const value = useMemo(
|
||||||
|
() =>
|
||||||
|
participants.map((x) => ({
|
||||||
|
value: x,
|
||||||
|
label: `${users.find((y) => y.id === x)?.email} - ${
|
||||||
|
users.find((y) => y.id === x)?.name
|
||||||
|
}`,
|
||||||
|
})),
|
||||||
|
[participants, users]
|
||||||
|
);
|
||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="mt-4 flex w-full flex-col gap-12 px-4 py-2">
|
<div className="mt-4 flex w-full flex-col gap-12 px-4 py-2">
|
||||||
<div className="flex flex-col gap-8">
|
<div className="flex flex-col gap-8">
|
||||||
<Input name="name" type="text" label="Name" defaultValue={name} onChange={setName} required disabled={group?.disableEditing} />
|
<Input
|
||||||
|
name="name"
|
||||||
|
type="text"
|
||||||
|
label="Name"
|
||||||
|
defaultValue={name}
|
||||||
|
onChange={setName}
|
||||||
|
required
|
||||||
|
disabled={group?.disableEditing}
|
||||||
|
/>
|
||||||
<div className="flex w-full flex-col gap-3">
|
<div className="flex w-full flex-col gap-3">
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-2">
|
||||||
<label className="text-mti-gray-dim text-base font-normal">Participants</label>
|
<label className="text-mti-gray-dim text-base font-normal">
|
||||||
<div className="tooltip" data-tip="The Excel file should only include a column with the desired e-mails.">
|
Participants
|
||||||
|
</label>
|
||||||
|
<div
|
||||||
|
className="tooltip"
|
||||||
|
data-tip="The Excel file should only include a column with the desired e-mails."
|
||||||
|
>
|
||||||
<BsQuestionCircleFill />
|
<BsQuestionCircleFill />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex w-full gap-8">
|
<div className="flex w-full gap-8">
|
||||||
<Select
|
<Select
|
||||||
className="w-full"
|
className="w-full"
|
||||||
value={participants.map((x) => ({
|
value={value}
|
||||||
value: x,
|
|
||||||
label: `${users.find((y) => y.id === x)?.email} - ${users.find((y) => y.id === x)?.name}`,
|
|
||||||
}))}
|
|
||||||
placeholder="Participants..."
|
placeholder="Participants..."
|
||||||
defaultValue={participants.map((x) => ({
|
defaultValue={value}
|
||||||
value: x,
|
options={userOptions}
|
||||||
label: `${users.find((y) => y.id === x)?.email} - ${users.find((y) => y.id === x)?.name}`,
|
|
||||||
}))}
|
|
||||||
options={availableUsers.map((x) => ({ value: x.id, label: `${x.email} - ${x.name}` }))}
|
|
||||||
onChange={(value) => setParticipants(value.map((x) => x.value))}
|
onChange={(value) => setParticipants(value.map((x) => x.value))}
|
||||||
isMulti
|
isMulti
|
||||||
isSearchable
|
isSearchable
|
||||||
@@ -160,18 +216,36 @@ const CreatePanel = ({ user, users, group, onClose }: CreateDialogProps) => {
|
|||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
{user.type !== "teacher" && (
|
{user.type !== "teacher" && (
|
||||||
<Button className="w-full max-w-[300px] h-fit" onClick={openFilePicker} isLoading={isLoading} variant="outline">
|
<Button
|
||||||
{filesContent.length === 0 ? "Upload participants Excel file" : filesContent[0].name}
|
className="w-full max-w-[300px] h-fit"
|
||||||
|
onClick={openFilePicker}
|
||||||
|
isLoading={isLoading}
|
||||||
|
variant="outline"
|
||||||
|
>
|
||||||
|
{filesContent.length === 0
|
||||||
|
? "Upload participants Excel file"
|
||||||
|
: filesContent[0].name}
|
||||||
</Button>
|
</Button>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="mt-8 flex w-full items-center justify-end gap-8">
|
<div className="mt-8 flex w-full items-center justify-end gap-8">
|
||||||
<Button variant="outline" color="red" className="w-full max-w-[200px]" isLoading={isLoading} onClick={onClose}>
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
color="red"
|
||||||
|
className="w-full max-w-[200px]"
|
||||||
|
isLoading={isLoading}
|
||||||
|
onClick={onClose}
|
||||||
|
>
|
||||||
Cancel
|
Cancel
|
||||||
</Button>
|
</Button>
|
||||||
<Button className="w-full max-w-[200px]" onClick={submit} isLoading={isLoading} disabled={!name}>
|
<Button
|
||||||
|
className="w-full max-w-[200px]"
|
||||||
|
onClick={submit}
|
||||||
|
isLoading={isLoading}
|
||||||
|
disabled={!name}
|
||||||
|
>
|
||||||
Submit
|
Submit
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
@@ -182,7 +256,8 @@ const CreatePanel = ({ user, users, group, onClose }: CreateDialogProps) => {
|
|||||||
export default function GroupList({ user }: { user: User }) {
|
export default function GroupList({ user }: { user: User }) {
|
||||||
const [isCreating, setIsCreating] = useState(false);
|
const [isCreating, setIsCreating] = useState(false);
|
||||||
const [editingGroup, setEditingGroup] = useState<Group>();
|
const [editingGroup, setEditingGroup] = useState<Group>();
|
||||||
const [viewingAllParticipants, setViewingAllParticipants] = useState<string>();
|
const [viewingAllParticipants, setViewingAllParticipants] =
|
||||||
|
useState<string>();
|
||||||
|
|
||||||
const { permissions } = usePermissions(user?.id || "");
|
const { permissions } = usePermissions(user?.id || "");
|
||||||
|
|
||||||
@@ -211,7 +286,14 @@ export default function GroupList({ user }: { user: User }) {
|
|||||||
columnHelper.accessor("admin", {
|
columnHelper.accessor("admin", {
|
||||||
header: "Admin",
|
header: "Admin",
|
||||||
cell: (info) => (
|
cell: (info) => (
|
||||||
<div className="tooltip" data-tip={USER_TYPE_LABELS[users.find((x) => x.id === info.getValue())?.type || "student"]}>
|
<div
|
||||||
|
className="tooltip"
|
||||||
|
data-tip={
|
||||||
|
USER_TYPE_LABELS[
|
||||||
|
users.find((x) => x.id === info.getValue())?.type || "student"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
>
|
||||||
{users.find((x) => x.id === info.getValue())?.name}
|
{users.find((x) => x.id === info.getValue())?.name}
|
||||||
</div>
|
</div>
|
||||||
),
|
),
|
||||||
@@ -226,23 +308,30 @@ export default function GroupList({ user }: { user: User }) {
|
|||||||
<span>
|
<span>
|
||||||
{info
|
{info
|
||||||
.getValue()
|
.getValue()
|
||||||
.slice(0, viewingAllParticipants === info.row.original.id ? undefined : 5)
|
.slice(
|
||||||
|
0,
|
||||||
|
viewingAllParticipants === info.row.original.id ? undefined : 5
|
||||||
|
)
|
||||||
.map((x) => users.find((y) => y.id === x)?.name)
|
.map((x) => users.find((y) => y.id === x)?.name)
|
||||||
.join(", ")}
|
.join(", ")}
|
||||||
{info.getValue().length > 5 && viewingAllParticipants !== info.row.original.id && (
|
{info.getValue().length > 5 &&
|
||||||
<button
|
viewingAllParticipants !== info.row.original.id && (
|
||||||
className="text-mti-purple-light font-bold hover:text-mti-purple-dark transition ease-in-out duration-300"
|
<button
|
||||||
onClick={() => setViewingAllParticipants(info.row.original.id)}>
|
className="text-mti-purple-light font-bold hover:text-mti-purple-dark transition ease-in-out duration-300"
|
||||||
, View More
|
onClick={() => setViewingAllParticipants(info.row.original.id)}
|
||||||
</button>
|
>
|
||||||
)}
|
, View More
|
||||||
{info.getValue().length > 5 && viewingAllParticipants === info.row.original.id && (
|
</button>
|
||||||
<button
|
)}
|
||||||
className="text-mti-purple-light font-bold hover:text-mti-purple-dark transition ease-in-out duration-300"
|
{info.getValue().length > 5 &&
|
||||||
onClick={() => setViewingAllParticipants(undefined)}>
|
viewingAllParticipants === info.row.original.id && (
|
||||||
, View Less
|
<button
|
||||||
</button>
|
className="text-mti-purple-light font-bold hover:text-mti-purple-dark transition ease-in-out duration-300"
|
||||||
)}
|
onClick={() => setViewingAllParticipants(undefined)}
|
||||||
|
>
|
||||||
|
, View Less
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
</span>
|
</span>
|
||||||
),
|
),
|
||||||
}),
|
}),
|
||||||
@@ -252,20 +341,34 @@ export default function GroupList({ user }: { user: User }) {
|
|||||||
cell: ({ row }: { row: { original: Group } }) => {
|
cell: ({ row }: { row: { original: Group } }) => {
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{user && (checkAccess(user, ["developer", "admin"]) || user.id === row.original.admin) && (
|
{user &&
|
||||||
<div className="flex gap-2">
|
(checkAccess(user, ["developer", "admin"]) ||
|
||||||
{(!row.original.disableEditing || checkAccess(user, ["developer", "admin"]), "editGroup") && (
|
user.id === row.original.admin) && (
|
||||||
<div data-tip="Edit" className="tooltip cursor-pointer" onClick={() => setEditingGroup(row.original)}>
|
<div className="flex gap-2">
|
||||||
<BsPencil className="hover:text-mti-purple-light transition duration-300 ease-in-out" />
|
{(!row.original.disableEditing ||
|
||||||
</div>
|
checkAccess(user, ["developer", "admin"]),
|
||||||
)}
|
"editGroup") && (
|
||||||
{(!row.original.disableEditing || checkAccess(user, ["developer", "admin"]), "deleteGroup") && (
|
<div
|
||||||
<div data-tip="Delete" className="tooltip cursor-pointer" onClick={() => deleteGroup(row.original)}>
|
data-tip="Edit"
|
||||||
<BsTrash className="hover:text-mti-purple-light transition duration-300 ease-in-out" />
|
className="tooltip cursor-pointer"
|
||||||
</div>
|
onClick={() => setEditingGroup(row.original)}
|
||||||
)}
|
>
|
||||||
</div>
|
<BsPencil className="hover:text-mti-purple-light transition duration-300 ease-in-out" />
|
||||||
)}
|
</div>
|
||||||
|
)}
|
||||||
|
{(!row.original.disableEditing ||
|
||||||
|
checkAccess(user, ["developer", "admin"]),
|
||||||
|
"deleteGroup") && (
|
||||||
|
<div
|
||||||
|
data-tip="Delete"
|
||||||
|
className="tooltip cursor-pointer"
|
||||||
|
onClick={() => deleteGroup(row.original)}
|
||||||
|
>
|
||||||
|
<BsTrash className="hover:text-mti-purple-light transition duration-300 ease-in-out" />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@@ -280,7 +383,11 @@ export default function GroupList({ user }: { user: User }) {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="h-full w-full rounded-xl flex flex-col gap-4">
|
<div className="h-full w-full rounded-xl flex flex-col gap-4">
|
||||||
<Modal isOpen={isCreating || !!editingGroup} onClose={closeModal} title={editingGroup ? `Editing ${editingGroup.name}` : "New Group"}>
|
<Modal
|
||||||
|
isOpen={isCreating || !!editingGroup}
|
||||||
|
onClose={closeModal}
|
||||||
|
title={editingGroup ? `Editing ${editingGroup.name}` : "New Group"}
|
||||||
|
>
|
||||||
<CreatePanel
|
<CreatePanel
|
||||||
group={editingGroup}
|
group={editingGroup}
|
||||||
user={user}
|
user={user}
|
||||||
@@ -288,12 +395,22 @@ export default function GroupList({ user }: { user: User }) {
|
|||||||
users={users}
|
users={users}
|
||||||
/>
|
/>
|
||||||
</Modal>
|
</Modal>
|
||||||
<Table data={groups} columns={defaultColumns} searchFields={searchFields} />
|
<Table
|
||||||
|
data={groups}
|
||||||
|
columns={defaultColumns}
|
||||||
|
searchFields={searchFields}
|
||||||
|
/>
|
||||||
|
|
||||||
{checkAccess(user, ["teacher", "corporate", "mastercorporate", "admin", "developer"], permissions, "createGroup") && (
|
{checkAccess(
|
||||||
|
user,
|
||||||
|
["teacher", "corporate", "mastercorporate", "admin", "developer"],
|
||||||
|
permissions,
|
||||||
|
"createGroup"
|
||||||
|
) && (
|
||||||
<button
|
<button
|
||||||
onClick={() => setIsCreating(true)}
|
onClick={() => setIsCreating(true)}
|
||||||
className="bg-mti-purple-light hover:bg-mti-purple w-full py-2 text-white transition duration-300 ease-in-out">
|
className="bg-mti-purple-light hover:bg-mti-purple w-full py-2 text-white transition duration-300 ease-in-out"
|
||||||
|
>
|
||||||
New Group
|
New Group
|
||||||
</button>
|
</button>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@@ -1,256 +1,341 @@
|
|||||||
import Input from "@/components/Low/Input";
|
import Input from "@/components/Low/Input";
|
||||||
import Modal from "@/components/Modal";
|
import Modal from "@/components/Modal";
|
||||||
import usePackages from "@/hooks/usePackages";
|
import usePackages from "@/hooks/usePackages";
|
||||||
import {Module} from "@/interfaces";
|
import { Module } from "@/interfaces";
|
||||||
import {Package} from "@/interfaces/paypal";
|
import { Package } from "@/interfaces/paypal";
|
||||||
import {User} from "@/interfaces/user";
|
import { User } from "@/interfaces/user";
|
||||||
import {createColumnHelper, flexRender, getCoreRowModel, useReactTable} from "@tanstack/react-table";
|
import {
|
||||||
|
createColumnHelper,
|
||||||
|
flexRender,
|
||||||
|
getCoreRowModel,
|
||||||
|
useReactTable,
|
||||||
|
} from "@tanstack/react-table";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import {capitalize} from "lodash";
|
import { capitalize } from "lodash";
|
||||||
import {useState} from "react";
|
import { useCallback, useMemo, useState } from "react";
|
||||||
import {BsPencil, BsTrash} from "react-icons/bs";
|
import { BsPencil, BsTrash } from "react-icons/bs";
|
||||||
import {toast} from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
import Select from "react-select";
|
import Select from "react-select";
|
||||||
import {CURRENCIES} from "@/resources/paypal";
|
import { CURRENCIES } from "@/resources/paypal";
|
||||||
import Button from "@/components/Low/Button";
|
import Button from "@/components/Low/Button";
|
||||||
|
|
||||||
const CLASSES: {[key in Module]: string} = {
|
const CLASSES: { [key in Module]: string } = {
|
||||||
reading: "text-ielts-reading",
|
reading: "text-ielts-reading",
|
||||||
listening: "text-ielts-listening",
|
listening: "text-ielts-listening",
|
||||||
speaking: "text-ielts-speaking",
|
speaking: "text-ielts-speaking",
|
||||||
writing: "text-ielts-writing",
|
writing: "text-ielts-writing",
|
||||||
level: "text-ielts-level",
|
level: "text-ielts-level",
|
||||||
};
|
};
|
||||||
|
|
||||||
const columnHelper = createColumnHelper<Package>();
|
const columnHelper = createColumnHelper<Package>();
|
||||||
|
|
||||||
type DurationUnit = "days" | "weeks" | "months" | "years";
|
type DurationUnit = "days" | "weeks" | "months" | "years";
|
||||||
|
|
||||||
function PackageCreator({pack, onClose}: {pack?: Package; onClose: () => void}) {
|
const currencyOptions = CURRENCIES.map(({ label, currency }) => ({
|
||||||
const [duration, setDuration] = useState(pack?.duration || 1);
|
value: currency,
|
||||||
const [unit, setUnit] = useState<DurationUnit>(pack?.duration_unit || "months");
|
label,
|
||||||
|
}));
|
||||||
|
|
||||||
const [price, setPrice] = useState(pack?.price || 0);
|
function PackageCreator({
|
||||||
const [currency, setCurrency] = useState<string>(pack?.currency || "OMR");
|
pack,
|
||||||
|
onClose,
|
||||||
|
}: {
|
||||||
|
pack?: Package;
|
||||||
|
onClose: () => void;
|
||||||
|
}) {
|
||||||
|
const [duration, setDuration] = useState(pack?.duration || 1);
|
||||||
|
const [unit, setUnit] = useState<DurationUnit>(
|
||||||
|
pack?.duration_unit || "months"
|
||||||
|
);
|
||||||
|
|
||||||
const submit = () => {
|
const [price, setPrice] = useState(pack?.price || 0);
|
||||||
(pack ? axios.patch : axios.post)(pack ? `/api/packages/${pack.id}` : "/api/packages", {
|
const [currency, setCurrency] = useState<string>(pack?.currency || "OMR");
|
||||||
duration,
|
|
||||||
duration_unit: unit,
|
|
||||||
price,
|
|
||||||
currency,
|
|
||||||
})
|
|
||||||
.then(() => {
|
|
||||||
toast.success("New payment has been created successfully!");
|
|
||||||
onClose();
|
|
||||||
})
|
|
||||||
.catch(() => {
|
|
||||||
toast.error("Something went wrong, please try again later!");
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
const submit = useCallback(() => {
|
||||||
<div className="flex flex-col gap-8 py-8">
|
(pack ? axios.patch : axios.post)(
|
||||||
<div className="flex flex-col gap-3">
|
pack ? `/api/packages/${pack.id}` : "/api/packages",
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Price *</label>
|
{
|
||||||
<div className="flex gap-4 items-center">
|
duration,
|
||||||
<Input defaultValue={price} name="price" type="number" onChange={(e) => setPrice(parseInt(e))} />
|
duration_unit: unit,
|
||||||
|
price,
|
||||||
|
currency,
|
||||||
|
}
|
||||||
|
)
|
||||||
|
.then(() => {
|
||||||
|
toast.success("New payment has been created successfully!");
|
||||||
|
onClose();
|
||||||
|
})
|
||||||
|
.catch(() => {
|
||||||
|
toast.error("Something went wrong, please try again later!");
|
||||||
|
});
|
||||||
|
}, [duration, unit, price, currency, pack, onClose]);
|
||||||
|
|
||||||
<Select
|
const currencyDefaultValue = useMemo(() => {
|
||||||
className="px-4 col-span-2 py-4 w-full text-sm font-normal placeholder:text-mti-gray-cool disabled:bg-mti-gray-platinum/40 disabled:text-mti-gray-dim disabled:cursor-not-allowed bg-white rounded-full border border-mti-gray-platinum focus:outline-none"
|
return {
|
||||||
options={CURRENCIES.map(({label, currency}) => ({value: currency, label}))}
|
value: currency || "EUR",
|
||||||
defaultValue={{value: currency || "EUR", label: CURRENCIES.find((c) => c.currency === currency)?.label || "Euro"}}
|
label: CURRENCIES.find((c) => c.currency === currency)?.label || "Euro",
|
||||||
onChange={(value) => setCurrency(value?.value || "EUR")}
|
};
|
||||||
value={{value: currency || "EUR", label: CURRENCIES.find((c) => c.currency === currency)?.label || "Euro"}}
|
}, [currency]);
|
||||||
menuPortalTarget={document?.body}
|
|
||||||
styles={{
|
return (
|
||||||
menuPortal: (base) => ({...base, zIndex: 9999}),
|
<div className="flex flex-col gap-8 py-8">
|
||||||
control: (styles) => ({
|
<div className="flex flex-col gap-3">
|
||||||
...styles,
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
paddingLeft: "4px",
|
Price *
|
||||||
border: "none",
|
</label>
|
||||||
outline: "none",
|
<div className="flex gap-4 items-center">
|
||||||
":focus": {
|
<Input
|
||||||
outline: "none",
|
defaultValue={price}
|
||||||
},
|
name="price"
|
||||||
}),
|
type="number"
|
||||||
option: (styles, state) => ({
|
onChange={(e) => setPrice(parseInt(e))}
|
||||||
...styles,
|
/>
|
||||||
backgroundColor: state.isFocused ? "#D5D9F0" : state.isSelected ? "#7872BF" : "white",
|
|
||||||
color: state.isFocused ? "black" : styles.color,
|
<Select
|
||||||
}),
|
className="px-4 col-span-2 py-4 w-full text-sm font-normal placeholder:text-mti-gray-cool disabled:bg-mti-gray-platinum/40 disabled:text-mti-gray-dim disabled:cursor-not-allowed bg-white rounded-full border border-mti-gray-platinum focus:outline-none"
|
||||||
}}
|
options={currencyOptions}
|
||||||
/>
|
defaultValue={currencyDefaultValue}
|
||||||
</div>
|
onChange={(value) => setCurrency(value?.value || "EUR")}
|
||||||
</div>
|
value={currencyDefaultValue}
|
||||||
<div className="flex flex-col gap-3">
|
menuPortalTarget={document?.body}
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Duration *</label>
|
styles={{
|
||||||
<div className="flex gap-4 items-center">
|
menuPortal: (base) => ({ ...base, zIndex: 9999 }),
|
||||||
<Input defaultValue={duration} name="duration" type="number" onChange={(e) => setDuration(parseInt(e))} />
|
control: (styles) => ({
|
||||||
<Select
|
...styles,
|
||||||
className="px-4 col-span-2 py-4 w-full text-sm font-normal placeholder:text-mti-gray-cool disabled:bg-mti-gray-platinum/40 disabled:text-mti-gray-dim disabled:cursor-not-allowed bg-white rounded-full border border-mti-gray-platinum focus:outline-none"
|
paddingLeft: "4px",
|
||||||
options={[
|
border: "none",
|
||||||
{value: "days", label: "Days"},
|
outline: "none",
|
||||||
{value: "weeks", label: "Weeks"},
|
":focus": {
|
||||||
{value: "months", label: "Months"},
|
outline: "none",
|
||||||
{value: "years", label: "Years"},
|
},
|
||||||
]}
|
}),
|
||||||
defaultValue={{value: "months", label: "Months"}}
|
option: (styles, state) => ({
|
||||||
onChange={(value) => setUnit((value?.value as DurationUnit) || "months")}
|
...styles,
|
||||||
value={{value: unit, label: capitalize(unit)}}
|
backgroundColor: state.isFocused
|
||||||
menuPortalTarget={document?.body}
|
? "#D5D9F0"
|
||||||
styles={{
|
: state.isSelected
|
||||||
menuPortal: (base) => ({...base, zIndex: 9999}),
|
? "#7872BF"
|
||||||
control: (styles) => ({
|
: "white",
|
||||||
...styles,
|
color: state.isFocused ? "black" : styles.color,
|
||||||
paddingLeft: "4px",
|
}),
|
||||||
border: "none",
|
}}
|
||||||
outline: "none",
|
/>
|
||||||
":focus": {
|
</div>
|
||||||
outline: "none",
|
</div>
|
||||||
},
|
<div className="flex flex-col gap-3">
|
||||||
}),
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
option: (styles, state) => ({
|
Duration *
|
||||||
...styles,
|
</label>
|
||||||
backgroundColor: state.isFocused ? "#D5D9F0" : state.isSelected ? "#7872BF" : "white",
|
<div className="flex gap-4 items-center">
|
||||||
color: state.isFocused ? "black" : styles.color,
|
<Input
|
||||||
}),
|
defaultValue={duration}
|
||||||
}}
|
name="duration"
|
||||||
/>
|
type="number"
|
||||||
</div>
|
onChange={(e) => setDuration(parseInt(e))}
|
||||||
</div>
|
/>
|
||||||
<div className="flex w-full justify-end items-center gap-8 mt-8">
|
<Select
|
||||||
<Button variant="outline" color="red" className="w-full max-w-[200px]" onClick={onClose}>
|
className="px-4 col-span-2 py-4 w-full text-sm font-normal placeholder:text-mti-gray-cool disabled:bg-mti-gray-platinum/40 disabled:text-mti-gray-dim disabled:cursor-not-allowed bg-white rounded-full border border-mti-gray-platinum focus:outline-none"
|
||||||
Cancel
|
options={[
|
||||||
</Button>
|
{ value: "days", label: "Days" },
|
||||||
<Button className="w-full max-w-[200px]" onClick={submit} disabled={!duration || !price}>
|
{ value: "weeks", label: "Weeks" },
|
||||||
Submit
|
{ value: "months", label: "Months" },
|
||||||
</Button>
|
{ value: "years", label: "Years" },
|
||||||
</div>
|
]}
|
||||||
</div>
|
defaultValue={{ value: "months", label: "Months" }}
|
||||||
);
|
onChange={(value) =>
|
||||||
|
setUnit((value?.value as DurationUnit) || "months")
|
||||||
|
}
|
||||||
|
value={{ value: unit, label: capitalize(unit) }}
|
||||||
|
menuPortalTarget={document?.body}
|
||||||
|
styles={{
|
||||||
|
menuPortal: (base) => ({ ...base, zIndex: 9999 }),
|
||||||
|
control: (styles) => ({
|
||||||
|
...styles,
|
||||||
|
paddingLeft: "4px",
|
||||||
|
border: "none",
|
||||||
|
outline: "none",
|
||||||
|
":focus": {
|
||||||
|
outline: "none",
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
option: (styles, state) => ({
|
||||||
|
...styles,
|
||||||
|
backgroundColor: state.isFocused
|
||||||
|
? "#D5D9F0"
|
||||||
|
: state.isSelected
|
||||||
|
? "#7872BF"
|
||||||
|
: "white",
|
||||||
|
color: state.isFocused ? "black" : styles.color,
|
||||||
|
}),
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="flex w-full justify-end items-center gap-8 mt-8">
|
||||||
|
<Button
|
||||||
|
variant="outline"
|
||||||
|
color="red"
|
||||||
|
className="w-full max-w-[200px]"
|
||||||
|
onClick={onClose}
|
||||||
|
>
|
||||||
|
Cancel
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
className="w-full max-w-[200px]"
|
||||||
|
onClick={submit}
|
||||||
|
disabled={!duration || !price}
|
||||||
|
>
|
||||||
|
Submit
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function PackageList({user}: {user: User}) {
|
export default function PackageList({ user }: { user: User }) {
|
||||||
const [isCreating, setIsCreating] = useState(false);
|
const [isCreating, setIsCreating] = useState(false);
|
||||||
const [editingPackage, setEditingPackage] = useState<Package>();
|
const [editingPackage, setEditingPackage] = useState<Package>();
|
||||||
|
|
||||||
const {packages, reload} = usePackages();
|
const { packages, reload } = usePackages();
|
||||||
|
|
||||||
const deletePackage = async (pack: Package) => {
|
const deletePackage = useCallback(
|
||||||
if (!confirm(`Are you sure you want to delete this package?`)) return;
|
async (pack: Package) => {
|
||||||
|
if (!confirm(`Are you sure you want to delete this package?`)) return;
|
||||||
|
|
||||||
axios
|
axios
|
||||||
.delete(`/api/packages/${pack.id}`)
|
.delete(`/api/packages/${pack.id}`)
|
||||||
.then(() => toast.success(`Deleted the "${pack.id}" exam`))
|
.then(() => toast.success(`Deleted the "${pack.id}" exam`))
|
||||||
.catch((reason) => {
|
.catch((reason) => {
|
||||||
if (reason.response.status === 404) {
|
if (reason.response.status === 404) {
|
||||||
toast.error("Package not found!");
|
toast.error("Package not found!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (reason.response.status === 403) {
|
if (reason.response.status === 403) {
|
||||||
toast.error("You do not have permission to delete this exam!");
|
toast.error("You do not have permission to delete this exam!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
toast.error("Something went wrong, please try again later.");
|
toast.error("Something went wrong, please try again later.");
|
||||||
})
|
})
|
||||||
.finally(reload);
|
.finally(reload);
|
||||||
};
|
},
|
||||||
|
[reload]
|
||||||
|
);
|
||||||
|
|
||||||
const defaultColumns = [
|
const defaultColumns = useMemo(
|
||||||
columnHelper.accessor("id", {
|
() => [
|
||||||
header: "ID",
|
columnHelper.accessor("id", {
|
||||||
cell: (info) => info.getValue(),
|
header: "ID",
|
||||||
}),
|
cell: (info) => info.getValue(),
|
||||||
columnHelper.accessor("duration", {
|
}),
|
||||||
header: "Duration",
|
columnHelper.accessor("duration", {
|
||||||
cell: (info) => (
|
header: "Duration",
|
||||||
<span>
|
cell: (info) => (
|
||||||
{info.getValue()} {info.row.original.duration_unit}
|
<span>
|
||||||
</span>
|
{info.getValue()} {info.row.original.duration_unit}
|
||||||
),
|
</span>
|
||||||
}),
|
),
|
||||||
columnHelper.accessor("price", {
|
}),
|
||||||
header: "Price",
|
columnHelper.accessor("price", {
|
||||||
cell: (info) => (
|
header: "Price",
|
||||||
<span>
|
cell: (info) => (
|
||||||
{info.getValue()} {info.row.original.currency}
|
<span>
|
||||||
</span>
|
{info.getValue()} {info.row.original.currency}
|
||||||
),
|
</span>
|
||||||
}),
|
),
|
||||||
{
|
}),
|
||||||
header: "",
|
{
|
||||||
id: "actions",
|
header: "",
|
||||||
cell: ({row}: {row: {original: Package}}) => {
|
id: "actions",
|
||||||
return (
|
cell: ({ row }: { row: { original: Package } }) => {
|
||||||
<div className="flex gap-4">
|
return (
|
||||||
{["developer", "admin"].includes(user.type) && (
|
<div className="flex gap-4">
|
||||||
<div data-tip="Edit" className="cursor-pointer tooltip" onClick={() => setEditingPackage(row.original)}>
|
{["developer", "admin"].includes(user?.type) && (
|
||||||
<BsPencil className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
<div
|
||||||
</div>
|
data-tip="Edit"
|
||||||
)}
|
className="cursor-pointer tooltip"
|
||||||
{["developer", "admin"].includes(user.type) && (
|
onClick={() => setEditingPackage(row.original)}
|
||||||
<div data-tip="Delete" className="cursor-pointer tooltip" onClick={() => deletePackage(row.original)}>
|
>
|
||||||
<BsTrash className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
<BsPencil className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</div>
|
{["developer", "admin"].includes(user?.type) && (
|
||||||
);
|
<div
|
||||||
},
|
data-tip="Delete"
|
||||||
},
|
className="cursor-pointer tooltip"
|
||||||
];
|
onClick={() => deletePackage(row.original)}
|
||||||
|
>
|
||||||
|
<BsTrash className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
[deletePackage, user]
|
||||||
|
);
|
||||||
|
|
||||||
const table = useReactTable({
|
const table = useReactTable({
|
||||||
data: packages,
|
data: packages,
|
||||||
columns: defaultColumns,
|
columns: defaultColumns,
|
||||||
getCoreRowModel: getCoreRowModel(),
|
getCoreRowModel: getCoreRowModel(),
|
||||||
});
|
});
|
||||||
|
|
||||||
const closeModal = () => {
|
const closeModal = useCallback(() => {
|
||||||
setIsCreating(false);
|
setIsCreating(false);
|
||||||
setEditingPackage(undefined);
|
setEditingPackage(undefined);
|
||||||
reload();
|
reload();
|
||||||
};
|
}, [reload]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="w-full h-full rounded-xl">
|
<div className="w-full h-full rounded-xl">
|
||||||
<Modal
|
<Modal
|
||||||
isOpen={isCreating || !!editingPackage}
|
isOpen={isCreating || !!editingPackage}
|
||||||
onClose={closeModal}
|
onClose={closeModal}
|
||||||
title={editingPackage ? `Editing ${editingPackage.id}` : "New Package"}>
|
title={editingPackage ? `Editing ${editingPackage.id}` : "New Package"}
|
||||||
<PackageCreator onClose={closeModal} pack={editingPackage} />
|
>
|
||||||
</Modal>
|
<PackageCreator onClose={closeModal} pack={editingPackage} />
|
||||||
<table className="bg-mti-purple-ultralight/40 w-full">
|
</Modal>
|
||||||
<thead>
|
<table className="bg-mti-purple-ultralight/40 w-full">
|
||||||
{table.getHeaderGroups().map((headerGroup) => (
|
<thead>
|
||||||
<tr key={headerGroup.id}>
|
{table.getHeaderGroups().map((headerGroup) => (
|
||||||
{headerGroup.headers.map((header) => (
|
<tr key={headerGroup.id}>
|
||||||
<th className="p-4 text-left" key={header.id}>
|
{headerGroup.headers.map((header) => (
|
||||||
{header.isPlaceholder ? null : flexRender(header.column.columnDef.header, header.getContext())}
|
<th className="p-4 text-left" key={header.id}>
|
||||||
</th>
|
{header.isPlaceholder
|
||||||
))}
|
? null
|
||||||
</tr>
|
: flexRender(
|
||||||
))}
|
header.column.columnDef.header,
|
||||||
</thead>
|
header.getContext()
|
||||||
<tbody className="px-2">
|
)}
|
||||||
{table.getRowModel().rows.map((row) => (
|
</th>
|
||||||
<tr className="odd:bg-white even:bg-mti-purple-ultralight/40 rounded-lg py-2" key={row.id}>
|
))}
|
||||||
{row.getVisibleCells().map((cell) => (
|
</tr>
|
||||||
<td className="px-4 py-2" key={cell.id}>
|
))}
|
||||||
{flexRender(cell.column.columnDef.cell, cell.getContext())}
|
</thead>
|
||||||
</td>
|
<tbody className="px-2">
|
||||||
))}
|
{table.getRowModel().rows.map((row) => (
|
||||||
</tr>
|
<tr
|
||||||
))}
|
className="odd:bg-white even:bg-mti-purple-ultralight/40 rounded-lg py-2"
|
||||||
</tbody>
|
key={row.id}
|
||||||
</table>
|
>
|
||||||
<button
|
{row.getVisibleCells().map((cell) => (
|
||||||
onClick={() => setIsCreating(true)}
|
<td className="px-4 py-2" key={cell.id}>
|
||||||
className="w-full py-2 bg-mti-purple-light hover:bg-mti-purple transition ease-in-out duration-300 text-white">
|
{flexRender(cell.column.columnDef.cell, cell.getContext())}
|
||||||
New Package
|
</td>
|
||||||
</button>
|
))}
|
||||||
</div>
|
</tr>
|
||||||
);
|
))}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<button
|
||||||
|
onClick={() => setIsCreating(true)}
|
||||||
|
className="w-full py-2 bg-mti-purple-light hover:bg-mti-purple transition ease-in-out duration-300 text-white"
|
||||||
|
>
|
||||||
|
New Package
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,7 +5,6 @@ import {averageLevelCalculator} from "@/utils/score";
|
|||||||
import {groupByExam} from "@/utils/stats";
|
import {groupByExam} from "@/utils/stats";
|
||||||
import {createColumnHelper} from "@tanstack/react-table";
|
import {createColumnHelper} from "@tanstack/react-table";
|
||||||
import Checkbox from "@/components/Low/Checkbox";
|
import Checkbox from "@/components/Low/Checkbox";
|
||||||
import List from "@/components/List";
|
|
||||||
import Table from "@/components/High/Table";
|
import Table from "@/components/High/Table";
|
||||||
|
|
||||||
type StudentPerformanceItem = StudentUser & {entitiesLabel: string; group: string};
|
type StudentPerformanceItem = StudentUser & {entitiesLabel: string; group: string};
|
||||||
|
|||||||
@@ -5,8 +5,13 @@ import axios from "axios";
|
|||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import { capitalize } from "lodash";
|
import { capitalize } from "lodash";
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
import { useEffect, useMemo, useState } from "react";
|
import { useMemo, useState } from "react";
|
||||||
import { BsCheck, BsCheckCircle, BsFillExclamationOctagonFill, BsTrash } from "react-icons/bs";
|
import {
|
||||||
|
BsCheck,
|
||||||
|
BsCheckCircle,
|
||||||
|
BsFillExclamationOctagonFill,
|
||||||
|
BsTrash,
|
||||||
|
} from "react-icons/bs";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
import { countries, TCountries } from "countries-list";
|
import { countries, TCountries } from "countries-list";
|
||||||
import countryCodes from "country-codes-list";
|
import countryCodes from "country-codes-list";
|
||||||
@@ -17,433 +22,600 @@ import useFilterStore from "@/stores/listFilterStore";
|
|||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { mapBy } from "@/utils";
|
import { mapBy } from "@/utils";
|
||||||
import { exportListToExcel } from "@/utils/users";
|
import { exportListToExcel } from "@/utils/users";
|
||||||
import usePermissions from "@/hooks/usePermissions";
|
|
||||||
import useUserBalance from "@/hooks/useUserBalance";
|
|
||||||
import useEntitiesUsers from "@/hooks/useEntitiesUsers";
|
import useEntitiesUsers from "@/hooks/useEntitiesUsers";
|
||||||
import { WithLabeledEntities } from "@/interfaces/entity";
|
import { WithLabeledEntities } from "@/interfaces/entity";
|
||||||
import Table from "@/components/High/Table";
|
import Table from "@/components/High/Table";
|
||||||
import useEntities from "@/hooks/useEntities";
|
import useEntities from "@/hooks/useEntities";
|
||||||
import { useAllowedEntities } from "@/hooks/useEntityPermissions";
|
import { useAllowedEntities } from "@/hooks/useEntityPermissions";
|
||||||
import { findAllowedEntities } from "@/utils/permissions";
|
|
||||||
|
|
||||||
const columnHelper = createColumnHelper<WithLabeledEntities<User>>();
|
const columnHelper = createColumnHelper<WithLabeledEntities<User>>();
|
||||||
const searchFields = [["name"], ["email"], ["entities", ""]];
|
const searchFields = [["name"], ["email"], ["entities", ""]];
|
||||||
|
|
||||||
export default function UserList({
|
export default function UserList({
|
||||||
user,
|
user,
|
||||||
filters = [],
|
filters = [],
|
||||||
type,
|
type,
|
||||||
renderHeader,
|
renderHeader,
|
||||||
}: {
|
}: {
|
||||||
user: User;
|
user: User;
|
||||||
filters?: ((user: User) => boolean)[];
|
filters?: ((user: User) => boolean)[];
|
||||||
type?: Type;
|
type?: Type;
|
||||||
renderHeader?: (total: number) => JSX.Element;
|
renderHeader?: (total: number) => JSX.Element;
|
||||||
}) {
|
}) {
|
||||||
const [showDemographicInformation, setShowDemographicInformation] = useState(false);
|
const [showDemographicInformation, setShowDemographicInformation] =
|
||||||
const [selectedUser, setSelectedUser] = useState<User>();
|
useState(false);
|
||||||
|
const [selectedUser, setSelectedUser] = useState<User>();
|
||||||
|
|
||||||
const { users, reload } = useEntitiesUsers(type)
|
const { users, isLoading, reload } = useEntitiesUsers(type);
|
||||||
const { entities } = useEntities()
|
const { entities } = useEntities();
|
||||||
|
|
||||||
const isAdmin = useMemo(() => ["admin", "developer"].includes(user?.type), [user?.type])
|
const isAdmin = useMemo(
|
||||||
|
() => ["admin", "developer"].includes(user?.type),
|
||||||
|
[user?.type]
|
||||||
|
);
|
||||||
|
|
||||||
const entitiesViewStudents = useAllowedEntities(user, entities, "view_students")
|
const entitiesViewStudents = useAllowedEntities(
|
||||||
const entitiesEditStudents = useAllowedEntities(user, entities, "edit_students")
|
user,
|
||||||
const entitiesDeleteStudents = useAllowedEntities(user, entities, "delete_students")
|
entities,
|
||||||
|
"view_students"
|
||||||
|
);
|
||||||
|
const entitiesEditStudents = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"edit_students"
|
||||||
|
);
|
||||||
|
const entitiesDeleteStudents = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"delete_students"
|
||||||
|
);
|
||||||
|
|
||||||
const entitiesViewTeachers = useAllowedEntities(user, entities, "view_teachers")
|
const entitiesViewTeachers = useAllowedEntities(
|
||||||
const entitiesEditTeachers = useAllowedEntities(user, entities, "edit_teachers")
|
user,
|
||||||
const entitiesDeleteTeachers = useAllowedEntities(user, entities, "delete_teachers")
|
entities,
|
||||||
|
"view_teachers"
|
||||||
|
);
|
||||||
|
const entitiesEditTeachers = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"edit_teachers"
|
||||||
|
);
|
||||||
|
const entitiesDeleteTeachers = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"delete_teachers"
|
||||||
|
);
|
||||||
|
|
||||||
const entitiesViewCorporates = useAllowedEntities(user, entities, "view_corporates")
|
const entitiesViewCorporates = useAllowedEntities(
|
||||||
const entitiesEditCorporates = useAllowedEntities(user, entities, "edit_corporates")
|
user,
|
||||||
const entitiesDeleteCorporates = useAllowedEntities(user, entities, "delete_corporates")
|
entities,
|
||||||
|
"view_corporates"
|
||||||
|
);
|
||||||
|
const entitiesEditCorporates = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"edit_corporates"
|
||||||
|
);
|
||||||
|
const entitiesDeleteCorporates = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"delete_corporates"
|
||||||
|
);
|
||||||
|
|
||||||
const entitiesViewMasterCorporates = useAllowedEntities(user, entities, "view_mastercorporates")
|
const entitiesViewMasterCorporates = useAllowedEntities(
|
||||||
const entitiesEditMasterCorporates = useAllowedEntities(user, entities, "edit_mastercorporates")
|
user,
|
||||||
const entitiesDeleteMasterCorporates = useAllowedEntities(user, entities, "delete_mastercorporates")
|
entities,
|
||||||
|
"view_mastercorporates"
|
||||||
|
);
|
||||||
|
const entitiesEditMasterCorporates = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"edit_mastercorporates"
|
||||||
|
);
|
||||||
|
const entitiesDeleteMasterCorporates = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"delete_mastercorporates"
|
||||||
|
);
|
||||||
|
|
||||||
const entitiesDownloadUsers = useAllowedEntities(user, entities, "download_user_list")
|
const entitiesDownloadUsers = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"download_user_list"
|
||||||
|
);
|
||||||
|
|
||||||
const appendUserFilters = useFilterStore((state) => state.appendUserFilter);
|
const appendUserFilters = useFilterStore((state) => state.appendUserFilter);
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
const expirationDateColor = (date: Date) => {
|
const expirationDateColor = (date: Date) => {
|
||||||
const momentDate = moment(date);
|
const momentDate = moment(date);
|
||||||
const today = moment(new Date());
|
const today = moment(new Date());
|
||||||
|
|
||||||
if (today.isAfter(momentDate)) return "!text-mti-red-light font-bold line-through";
|
if (today.isAfter(momentDate))
|
||||||
if (today.add(1, "weeks").isAfter(momentDate)) return "!text-mti-red-light";
|
return "!text-mti-red-light font-bold line-through";
|
||||||
if (today.add(2, "weeks").isAfter(momentDate)) return "!text-mti-rose-light";
|
if (today.add(1, "weeks").isAfter(momentDate)) return "!text-mti-red-light";
|
||||||
if (today.add(1, "months").isAfter(momentDate)) return "!text-mti-orange-light";
|
if (today.add(2, "weeks").isAfter(momentDate))
|
||||||
};
|
return "!text-mti-rose-light";
|
||||||
|
if (today.add(1, "months").isAfter(momentDate))
|
||||||
|
return "!text-mti-orange-light";
|
||||||
|
};
|
||||||
|
|
||||||
const allowedUsers = useMemo(() => users.filter((u) => {
|
const allowedUsers = useMemo(
|
||||||
if (isAdmin) return true
|
() =>
|
||||||
if (u.id === user?.id) return false
|
users.filter((u) => {
|
||||||
|
if (isAdmin) return true;
|
||||||
|
if (u.id === user?.id) return false;
|
||||||
|
|
||||||
switch (u.type) {
|
switch (u.type) {
|
||||||
case "student": return mapBy((u.entities || []), 'id').some((id) => mapBy(entitiesViewStudents, 'id').includes(id))
|
case "student":
|
||||||
case "teacher": return mapBy((u.entities || []), 'id').some((id) => mapBy(entitiesViewTeachers, 'id').includes(id))
|
return mapBy(u.entities || [], "id").some((id) =>
|
||||||
case 'corporate': return mapBy((u.entities || []), 'id').some((id) => mapBy(entitiesViewCorporates, 'id').includes(id))
|
mapBy(entitiesViewStudents, "id").includes(id)
|
||||||
case 'mastercorporate': return mapBy((u.entities || []), 'id').some((id) => mapBy(entitiesViewMasterCorporates, 'id').includes(id))
|
);
|
||||||
default: return false
|
case "teacher":
|
||||||
}
|
return mapBy(u.entities || [], "id").some((id) =>
|
||||||
})
|
mapBy(entitiesViewTeachers, "id").includes(id)
|
||||||
, [entitiesViewCorporates, entitiesViewMasterCorporates, entitiesViewStudents, entitiesViewTeachers, isAdmin, user?.id, users])
|
);
|
||||||
|
case "corporate":
|
||||||
|
return mapBy(u.entities || [], "id").some((id) =>
|
||||||
|
mapBy(entitiesViewCorporates, "id").includes(id)
|
||||||
|
);
|
||||||
|
case "mastercorporate":
|
||||||
|
return mapBy(u.entities || [], "id").some((id) =>
|
||||||
|
mapBy(entitiesViewMasterCorporates, "id").includes(id)
|
||||||
|
);
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}),
|
||||||
|
[
|
||||||
|
entitiesViewCorporates,
|
||||||
|
entitiesViewMasterCorporates,
|
||||||
|
entitiesViewStudents,
|
||||||
|
entitiesViewTeachers,
|
||||||
|
isAdmin,
|
||||||
|
user?.id,
|
||||||
|
users,
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
const displayUsers = useMemo(() =>
|
const displayUsers = useMemo(
|
||||||
filters.length > 0 ? filters.reduce((d, f) => d.filter(f), allowedUsers) : allowedUsers,
|
() =>
|
||||||
[filters, allowedUsers])
|
filters.length > 0
|
||||||
|
? filters.reduce((d, f) => d.filter(f), allowedUsers)
|
||||||
|
: allowedUsers,
|
||||||
|
[filters, allowedUsers]
|
||||||
|
);
|
||||||
|
|
||||||
const deleteAccount = (user: User) => {
|
const deleteAccount = (user: User) => {
|
||||||
if (!confirm(`Are you sure you want to delete ${user.name}'s account?`)) return;
|
if (!confirm(`Are you sure you want to delete ${user.name}'s account?`))
|
||||||
|
return;
|
||||||
|
|
||||||
axios
|
axios
|
||||||
.delete<{ ok: boolean }>(`/api/user?id=${user.id}`)
|
.delete<{ ok: boolean }>(`/api/user?id=${user.id}`)
|
||||||
.then(() => {
|
.then(() => {
|
||||||
toast.success("User deleted successfully!");
|
toast.success("User deleted successfully!");
|
||||||
reload()
|
reload();
|
||||||
})
|
})
|
||||||
.catch(() => {
|
.catch(() => {
|
||||||
toast.error("Something went wrong!", { toastId: "delete-error" });
|
toast.error("Something went wrong!", { toastId: "delete-error" });
|
||||||
})
|
})
|
||||||
.finally(reload);
|
.finally(reload);
|
||||||
};
|
};
|
||||||
|
|
||||||
const verifyAccount = (user: User) => {
|
const verifyAccount = (user: User) => {
|
||||||
axios
|
axios
|
||||||
.post<{ user?: User; ok?: boolean }>(`/api/users/update?id=${user.id}`, {
|
.post<{ user?: User; ok?: boolean }>(`/api/users/update?id=${user.id}`, {
|
||||||
...user,
|
...user,
|
||||||
isVerified: true,
|
isVerified: true,
|
||||||
})
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
toast.success("User verified successfully!");
|
toast.success("User verified successfully!");
|
||||||
reload();
|
reload();
|
||||||
})
|
})
|
||||||
.catch(() => {
|
.catch(() => {
|
||||||
toast.error("Something went wrong!", { toastId: "update-error" });
|
toast.error("Something went wrong!", { toastId: "update-error" });
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const toggleDisableAccount = (user: User) => {
|
const toggleDisableAccount = (user: User) => {
|
||||||
if (
|
if (
|
||||||
!confirm(
|
!confirm(
|
||||||
`Are you sure you want to ${user.status === "disabled" ? "enable" : "disable"} ${user.name
|
`Are you sure you want to ${
|
||||||
}'s account? This change is usually related to their payment state.`,
|
user.status === "disabled" ? "enable" : "disable"
|
||||||
)
|
} ${
|
||||||
)
|
user.name
|
||||||
return;
|
}'s account? This change is usually related to their payment state.`
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return;
|
||||||
|
|
||||||
axios
|
axios
|
||||||
.post<{ user?: User; ok?: boolean }>(`/api/users/update?id=${user.id}`, {
|
.post<{ user?: User; ok?: boolean }>(`/api/users/update?id=${user.id}`, {
|
||||||
...user,
|
...user,
|
||||||
status: user.status === "disabled" ? "active" : "disabled",
|
status: user.status === "disabled" ? "active" : "disabled",
|
||||||
})
|
})
|
||||||
.then(() => {
|
.then(() => {
|
||||||
toast.success(`User ${user.status === "disabled" ? "enabled" : "disabled"} successfully!`);
|
toast.success(
|
||||||
reload();
|
`User ${
|
||||||
})
|
user.status === "disabled" ? "enabled" : "disabled"
|
||||||
.catch(() => {
|
} successfully!`
|
||||||
toast.error("Something went wrong!", { toastId: "update-error" });
|
);
|
||||||
});
|
reload();
|
||||||
};
|
})
|
||||||
|
.catch(() => {
|
||||||
|
toast.error("Something went wrong!", { toastId: "update-error" });
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
const getEditPermission = (type: Type) => {
|
const getEditPermission = (type: Type) => {
|
||||||
if (type === "student") return entitiesEditStudents
|
if (type === "student") return entitiesEditStudents;
|
||||||
if (type === "teacher") return entitiesEditTeachers
|
if (type === "teacher") return entitiesEditTeachers;
|
||||||
if (type === "corporate") return entitiesEditCorporates
|
if (type === "corporate") return entitiesEditCorporates;
|
||||||
if (type === "mastercorporate") return entitiesEditMasterCorporates
|
if (type === "mastercorporate") return entitiesEditMasterCorporates;
|
||||||
|
|
||||||
return []
|
return [];
|
||||||
}
|
};
|
||||||
|
|
||||||
const getDeletePermission = (type: Type) => {
|
const getDeletePermission = (type: Type) => {
|
||||||
if (type === "student") return entitiesDeleteStudents
|
if (type === "student") return entitiesDeleteStudents;
|
||||||
if (type === "teacher") return entitiesDeleteTeachers
|
if (type === "teacher") return entitiesDeleteTeachers;
|
||||||
if (type === "corporate") return entitiesDeleteCorporates
|
if (type === "corporate") return entitiesDeleteCorporates;
|
||||||
if (type === "mastercorporate") return entitiesDeleteMasterCorporates
|
if (type === "mastercorporate") return entitiesDeleteMasterCorporates;
|
||||||
|
|
||||||
return []
|
return [];
|
||||||
}
|
};
|
||||||
|
|
||||||
const canEditUser = (u: User) =>
|
const canEditUser = (u: User) =>
|
||||||
isAdmin || u.entities.some(e => mapBy(getEditPermission(u.type), 'id').includes(e.id))
|
isAdmin ||
|
||||||
|
u.entities.some((e) =>
|
||||||
|
mapBy(getEditPermission(u.type), "id").includes(e.id)
|
||||||
|
);
|
||||||
|
|
||||||
const canDeleteUser = (u: User) =>
|
const canDeleteUser = (u: User) =>
|
||||||
isAdmin || u.entities.some(e => mapBy(getDeletePermission(u.type), 'id').includes(e.id))
|
isAdmin ||
|
||||||
|
u.entities.some((e) =>
|
||||||
|
mapBy(getDeletePermission(u.type), "id").includes(e.id)
|
||||||
|
);
|
||||||
|
|
||||||
const actionColumn = ({ row }: { row: { original: User } }) => {
|
const actionColumn = ({ row }: { row: { original: User } }) => {
|
||||||
const canEdit = canEditUser(row.original)
|
const canEdit = canEditUser(row.original);
|
||||||
const canDelete = canDeleteUser(row.original)
|
const canDelete = canDeleteUser(row.original);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="flex gap-4">
|
<div className="flex gap-4">
|
||||||
{!row.original.isVerified && canEdit && (
|
{!row.original.isVerified && canEdit && (
|
||||||
<div data-tip="Verify User" className="cursor-pointer tooltip" onClick={() => verifyAccount(row.original)}>
|
<div
|
||||||
<BsCheck className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
data-tip="Verify User"
|
||||||
</div>
|
className="cursor-pointer tooltip"
|
||||||
)}
|
onClick={() => verifyAccount(row.original)}
|
||||||
{canEdit && (
|
>
|
||||||
<div
|
<BsCheck className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
data-tip={row.original.status === "disabled" ? "Enable User" : "Disable User"}
|
</div>
|
||||||
className="cursor-pointer tooltip"
|
)}
|
||||||
onClick={() => toggleDisableAccount(row.original)}>
|
{canEdit && (
|
||||||
{row.original.status === "disabled" ? (
|
<div
|
||||||
<BsCheckCircle className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
data-tip={
|
||||||
) : (
|
row.original.status === "disabled"
|
||||||
<BsFillExclamationOctagonFill className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
? "Enable User"
|
||||||
)}
|
: "Disable User"
|
||||||
</div>
|
}
|
||||||
)}
|
className="cursor-pointer tooltip"
|
||||||
{canDelete && (
|
onClick={() => toggleDisableAccount(row.original)}
|
||||||
<div data-tip="Delete" className="cursor-pointer tooltip" onClick={() => deleteAccount(row.original)}>
|
>
|
||||||
<BsTrash className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
{row.original.status === "disabled" ? (
|
||||||
</div>
|
<BsCheckCircle className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
)}
|
) : (
|
||||||
</div>
|
<BsFillExclamationOctagonFill className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
);
|
)}
|
||||||
};
|
</div>
|
||||||
|
)}
|
||||||
|
{canDelete && (
|
||||||
|
<div
|
||||||
|
data-tip="Delete"
|
||||||
|
className="cursor-pointer tooltip"
|
||||||
|
onClick={() => deleteAccount(row.original)}
|
||||||
|
>
|
||||||
|
<BsTrash className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
const demographicColumns = [
|
const demographicColumns = [
|
||||||
columnHelper.accessor("name", {
|
columnHelper.accessor("name", {
|
||||||
header: "Name",
|
header: "Name",
|
||||||
cell: ({ row, getValue }) => (
|
cell: ({ row, getValue }) => (
|
||||||
<div
|
<div
|
||||||
className={clsx(
|
className={clsx(
|
||||||
canEditUser(row.original) &&
|
canEditUser(row.original) &&
|
||||||
"underline text-mti-purple-light hover:text-mti-purple-dark transition ease-in-out duration-300 cursor-pointer",
|
"underline text-mti-purple-light hover:text-mti-purple-dark transition ease-in-out duration-300 cursor-pointer"
|
||||||
)}
|
)}
|
||||||
onClick={() =>
|
onClick={() =>
|
||||||
canEditUser(row.original) ? setSelectedUser(row.original) : null
|
canEditUser(row.original) ? setSelectedUser(row.original) : null
|
||||||
}>
|
}
|
||||||
{getValue()}
|
>
|
||||||
</div>
|
{getValue()}
|
||||||
),
|
</div>
|
||||||
}),
|
),
|
||||||
columnHelper.accessor("demographicInformation.country", {
|
}),
|
||||||
header: "Country",
|
columnHelper.accessor("demographicInformation.country", {
|
||||||
cell: (info) =>
|
header: "Country",
|
||||||
info.getValue()
|
cell: (info) =>
|
||||||
? `${countryCodes.findOne("countryCode" as any, info.getValue())?.flag} ${countries[info.getValue() as unknown as keyof TCountries]?.name
|
info.getValue()
|
||||||
} (+${countryCodes.findOne("countryCode" as any, info.getValue())?.countryCallingCode})`
|
? `${
|
||||||
: "N/A",
|
countryCodes.findOne("countryCode" as any, info.getValue())?.flag
|
||||||
}),
|
} ${
|
||||||
columnHelper.accessor("demographicInformation.phone", {
|
countries[info.getValue() as unknown as keyof TCountries]?.name
|
||||||
header: "Phone",
|
} (+${
|
||||||
cell: (info) => info.getValue() || "N/A",
|
countryCodes.findOne("countryCode" as any, info.getValue())
|
||||||
enableSorting: true,
|
?.countryCallingCode
|
||||||
}),
|
})`
|
||||||
columnHelper.accessor(
|
: "N/A",
|
||||||
(x) =>
|
}),
|
||||||
x.type === "corporate" || x.type === "mastercorporate" ? x.demographicInformation?.position : x.demographicInformation?.employment,
|
columnHelper.accessor("demographicInformation.phone", {
|
||||||
{
|
header: "Phone",
|
||||||
id: "employment",
|
cell: (info) => info.getValue() || "N/A",
|
||||||
header: "Employment",
|
enableSorting: true,
|
||||||
cell: (info) => (info.row.original.type === "corporate" ? info.getValue() : capitalize(info.getValue())) || "N/A",
|
}),
|
||||||
enableSorting: true,
|
columnHelper.accessor(
|
||||||
},
|
(x) =>
|
||||||
),
|
x.type === "corporate" || x.type === "mastercorporate"
|
||||||
columnHelper.accessor("lastLogin", {
|
? x.demographicInformation?.position
|
||||||
header: "Last Login",
|
: x.demographicInformation?.employment,
|
||||||
cell: (info) => (!!info.getValue() ? moment(info.getValue()).format("YYYY-MM-DD HH:mm") : "N/A"),
|
{
|
||||||
}),
|
id: "employment",
|
||||||
columnHelper.accessor("demographicInformation.gender", {
|
header: "Employment",
|
||||||
header: "Gender",
|
cell: (info) =>
|
||||||
cell: (info) => capitalize(info.getValue()) || "N/A",
|
(info.row.original.type === "corporate"
|
||||||
enableSorting: true,
|
? info.getValue()
|
||||||
}),
|
: capitalize(info.getValue())) || "N/A",
|
||||||
{
|
enableSorting: true,
|
||||||
header: (
|
}
|
||||||
<span className="cursor-pointer" onClick={() => setShowDemographicInformation((prev) => !prev)}>
|
),
|
||||||
Switch
|
columnHelper.accessor("lastLogin", {
|
||||||
</span>
|
header: "Last Login",
|
||||||
),
|
cell: (info) =>
|
||||||
id: "actions",
|
!!info.getValue()
|
||||||
cell: actionColumn,
|
? moment(info.getValue()).format("YYYY-MM-DD HH:mm")
|
||||||
sortable: false
|
: "N/A",
|
||||||
},
|
}),
|
||||||
];
|
columnHelper.accessor("demographicInformation.gender", {
|
||||||
|
header: "Gender",
|
||||||
|
cell: (info) => capitalize(info.getValue()) || "N/A",
|
||||||
|
enableSorting: true,
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
header: (
|
||||||
|
<span
|
||||||
|
className="cursor-pointer"
|
||||||
|
onClick={() => setShowDemographicInformation((prev) => !prev)}
|
||||||
|
>
|
||||||
|
Switch
|
||||||
|
</span>
|
||||||
|
),
|
||||||
|
id: "actions",
|
||||||
|
cell: actionColumn,
|
||||||
|
sortable: false,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
const defaultColumns = [
|
const defaultColumns = [
|
||||||
columnHelper.accessor("name", {
|
columnHelper.accessor("name", {
|
||||||
header: "Name",
|
header: "Name",
|
||||||
cell: ({ row, getValue }) => (
|
cell: ({ row, getValue }) => (
|
||||||
<div
|
<div
|
||||||
className={clsx(
|
className={clsx(
|
||||||
canEditUser(row.original) &&
|
canEditUser(row.original) &&
|
||||||
"underline text-mti-purple-light hover:text-mti-purple-dark transition ease-in-out duration-300 cursor-pointer",
|
"underline text-mti-purple-light hover:text-mti-purple-dark transition ease-in-out duration-300 cursor-pointer"
|
||||||
)}
|
)}
|
||||||
onClick={() =>
|
onClick={() =>
|
||||||
canEditUser(row.original) ? setSelectedUser(row.original) : null
|
canEditUser(row.original) ? setSelectedUser(row.original) : null
|
||||||
}>
|
}
|
||||||
{getValue()}
|
>
|
||||||
</div>
|
{getValue()}
|
||||||
),
|
</div>
|
||||||
}),
|
),
|
||||||
columnHelper.accessor("email", {
|
}),
|
||||||
header: "E-mail",
|
columnHelper.accessor("email", {
|
||||||
cell: ({ row, getValue }) => (
|
header: "E-mail",
|
||||||
<div
|
cell: ({ row, getValue }) => (
|
||||||
className={clsx(
|
<div
|
||||||
canEditUser(row.original) &&
|
className={clsx(
|
||||||
"underline text-mti-purple-light hover:text-mti-purple-dark transition ease-in-out duration-300 cursor-pointer",
|
canEditUser(row.original) &&
|
||||||
)}
|
"underline text-mti-purple-light hover:text-mti-purple-dark transition ease-in-out duration-300 cursor-pointer"
|
||||||
onClick={() => (canEditUser(row.original) ? setSelectedUser(row.original) : null)}>
|
)}
|
||||||
{getValue()}
|
onClick={() =>
|
||||||
</div>
|
canEditUser(row.original) ? setSelectedUser(row.original) : null
|
||||||
),
|
}
|
||||||
}),
|
>
|
||||||
columnHelper.accessor("type", {
|
{getValue()}
|
||||||
header: "Type",
|
</div>
|
||||||
cell: (info) => USER_TYPE_LABELS[info.getValue()],
|
),
|
||||||
}),
|
}),
|
||||||
columnHelper.accessor("studentID", {
|
columnHelper.accessor("type", {
|
||||||
header: "Student ID",
|
header: "Type",
|
||||||
cell: (info) => info.getValue() || "N/A",
|
cell: (info) => USER_TYPE_LABELS[info.getValue()],
|
||||||
}),
|
}),
|
||||||
columnHelper.accessor("entities", {
|
columnHelper.accessor("studentID", {
|
||||||
header: "Entities",
|
header: "Student ID",
|
||||||
cell: ({ getValue }) => mapBy(getValue(), 'label').join(', '),
|
cell: (info) => info.getValue() || "N/A",
|
||||||
}),
|
}),
|
||||||
columnHelper.accessor("subscriptionExpirationDate", {
|
columnHelper.accessor("entities", {
|
||||||
header: "Expiration",
|
header: "Entities",
|
||||||
cell: (info) => (
|
cell: ({ getValue }) => mapBy(getValue(), "label").join(", "),
|
||||||
<span className={clsx(info.getValue() ? expirationDateColor(moment(info.getValue()).toDate()) : "")}>
|
}),
|
||||||
{!info.getValue() ? "No expiry date" : moment(info.getValue()).format("DD/MM/YYYY")}
|
columnHelper.accessor("subscriptionExpirationDate", {
|
||||||
</span>
|
header: "Expiration",
|
||||||
),
|
cell: (info) => (
|
||||||
}),
|
<span
|
||||||
columnHelper.accessor("isVerified", {
|
className={clsx(
|
||||||
header: "Verified",
|
info.getValue()
|
||||||
cell: (info) => (
|
? expirationDateColor(moment(info.getValue()).toDate())
|
||||||
<div className="flex gap-3 items-center text-mti-gray-dim text-sm self-center">
|
: ""
|
||||||
<div
|
)}
|
||||||
className={clsx(
|
>
|
||||||
"w-6 h-6 rounded-md flex items-center justify-center border border-mti-purple-light bg-white",
|
{!info.getValue()
|
||||||
"transition duration-300 ease-in-out",
|
? "No expiry date"
|
||||||
info.getValue() && "!bg-mti-purple-light ",
|
: moment(info.getValue()).format("DD/MM/YYYY")}
|
||||||
)}>
|
</span>
|
||||||
<BsCheck color="white" className="w-full h-full" />
|
),
|
||||||
</div>
|
}),
|
||||||
</div>
|
columnHelper.accessor("isVerified", {
|
||||||
),
|
header: "Verified",
|
||||||
}),
|
cell: (info) => (
|
||||||
{
|
<div className="flex gap-3 items-center text-mti-gray-dim text-sm self-center">
|
||||||
header: (
|
<div
|
||||||
<span className="cursor-pointer" onClick={() => setShowDemographicInformation((prev) => !prev)}>
|
className={clsx(
|
||||||
Switch
|
"w-6 h-6 rounded-md flex items-center justify-center border border-mti-purple-light bg-white",
|
||||||
</span>
|
"transition duration-300 ease-in-out",
|
||||||
),
|
info.getValue() && "!bg-mti-purple-light "
|
||||||
id: "actions",
|
)}
|
||||||
cell: actionColumn,
|
>
|
||||||
sortable: false
|
<BsCheck color="white" className="w-full h-full" />
|
||||||
},
|
</div>
|
||||||
];
|
</div>
|
||||||
|
),
|
||||||
|
}),
|
||||||
|
{
|
||||||
|
header: (
|
||||||
|
<span
|
||||||
|
className="cursor-pointer"
|
||||||
|
onClick={() => setShowDemographicInformation((prev) => !prev)}
|
||||||
|
>
|
||||||
|
Switch
|
||||||
|
</span>
|
||||||
|
),
|
||||||
|
id: "actions",
|
||||||
|
cell: actionColumn,
|
||||||
|
sortable: false,
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
const downloadExcel = (rows: WithLabeledEntities<User>[]) => {
|
const downloadExcel = async (rows: WithLabeledEntities<User>[]) => {
|
||||||
if (entitiesDownloadUsers.length === 0) return toast.error("You are not allowed to download the user list.")
|
if (entitiesDownloadUsers.length === 0)
|
||||||
|
return toast.error("You are not allowed to download the user list.");
|
||||||
|
|
||||||
const allowedRows = rows.filter(r => mapBy(r.entities, 'id').some(e => mapBy(entitiesDownloadUsers, 'id').includes(e)))
|
const allowedRows = rows;
|
||||||
const csv = exportListToExcel(allowedRows);
|
const csv = await exportListToExcel(allowedRows);
|
||||||
|
|
||||||
const element = document.createElement("a");
|
const element = document.createElement("a");
|
||||||
const file = new Blob([csv], { type: "text/csv" });
|
const file = new Blob([csv], {
|
||||||
element.href = URL.createObjectURL(file);
|
type: "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet",
|
||||||
element.download = "users.csv";
|
});
|
||||||
document.body.appendChild(element);
|
element.href = URL.createObjectURL(file);
|
||||||
element.click();
|
element.download = "users.xlsx";
|
||||||
document.body.removeChild(element);
|
document.body.appendChild(element);
|
||||||
};
|
element.click();
|
||||||
|
document.body.removeChild(element);
|
||||||
|
};
|
||||||
|
|
||||||
const viewStudentFilter = (x: User) => x.type === "student";
|
const viewStudentFilter = (x: User) => x.type === "student";
|
||||||
const viewTeacherFilter = (x: User) => x.type === "teacher";
|
const viewTeacherFilter = (x: User) => x.type === "teacher";
|
||||||
const belongsToAdminFilter = (x: User) => x.entities.some(({ id }) => mapBy(selectedUser?.entities || [], 'id').includes(id));
|
const belongsToAdminFilter = (x: User) =>
|
||||||
|
x.entities.some(({ id }) =>
|
||||||
|
mapBy(selectedUser?.entities || [], "id").includes(id)
|
||||||
|
);
|
||||||
|
|
||||||
const viewStudentFilterBelongsToAdmin = (x: User) => viewStudentFilter(x) && belongsToAdminFilter(x);
|
const viewStudentFilterBelongsToAdmin = (x: User) =>
|
||||||
const viewTeacherFilterBelongsToAdmin = (x: User) => viewTeacherFilter(x) && belongsToAdminFilter(x);
|
viewStudentFilter(x) && belongsToAdminFilter(x);
|
||||||
|
const viewTeacherFilterBelongsToAdmin = (x: User) =>
|
||||||
|
viewTeacherFilter(x) && belongsToAdminFilter(x);
|
||||||
|
|
||||||
const renderUserCard = (selectedUser: User) => {
|
const renderUserCard = (selectedUser: User) => {
|
||||||
const studentsFromAdmin = users.filter(viewStudentFilterBelongsToAdmin);
|
const studentsFromAdmin = users.filter(viewStudentFilterBelongsToAdmin);
|
||||||
const teachersFromAdmin = users.filter(viewTeacherFilterBelongsToAdmin);
|
const teachersFromAdmin = users.filter(viewTeacherFilterBelongsToAdmin);
|
||||||
return (
|
return (
|
||||||
<div className="w-full flex flex-col gap-8">
|
<div className="w-full flex flex-col gap-8">
|
||||||
<UserCard
|
<UserCard
|
||||||
maxUserAmount={0}
|
maxUserAmount={0}
|
||||||
loggedInUser={user}
|
loggedInUser={user}
|
||||||
onViewStudents={
|
onViewStudents={
|
||||||
(selectedUser.type === "corporate" || selectedUser.type === "teacher") && studentsFromAdmin.length > 0
|
(selectedUser.type === "corporate" ||
|
||||||
? () => {
|
selectedUser.type === "teacher") &&
|
||||||
appendUserFilters({
|
studentsFromAdmin.length > 0
|
||||||
id: "view-students",
|
? () => {
|
||||||
filter: viewStudentFilter,
|
appendUserFilters({
|
||||||
});
|
id: "view-students",
|
||||||
appendUserFilters({
|
filter: viewStudentFilter,
|
||||||
id: "belongs-to-admin",
|
});
|
||||||
filter: belongsToAdminFilter,
|
appendUserFilters({
|
||||||
});
|
id: "belongs-to-admin",
|
||||||
|
filter: belongsToAdminFilter,
|
||||||
|
});
|
||||||
|
|
||||||
router.push("/users");
|
router.push("/users");
|
||||||
}
|
}
|
||||||
: undefined
|
: undefined
|
||||||
}
|
}
|
||||||
onViewTeachers={
|
onViewTeachers={
|
||||||
(selectedUser.type === "corporate" || selectedUser.type === "student") && teachersFromAdmin.length > 0
|
(selectedUser.type === "corporate" ||
|
||||||
? () => {
|
selectedUser.type === "student") &&
|
||||||
appendUserFilters({
|
teachersFromAdmin.length > 0
|
||||||
id: "view-teachers",
|
? () => {
|
||||||
filter: viewTeacherFilter,
|
appendUserFilters({
|
||||||
});
|
id: "view-teachers",
|
||||||
appendUserFilters({
|
filter: viewTeacherFilter,
|
||||||
id: "belongs-to-admin",
|
});
|
||||||
filter: belongsToAdminFilter,
|
appendUserFilters({
|
||||||
});
|
id: "belongs-to-admin",
|
||||||
|
filter: belongsToAdminFilter,
|
||||||
|
});
|
||||||
|
|
||||||
router.push("/users");
|
router.push("/users");
|
||||||
}
|
}
|
||||||
: undefined
|
: undefined
|
||||||
}
|
}
|
||||||
onViewCorporate={
|
onViewCorporate={
|
||||||
selectedUser.type === "teacher" || selectedUser.type === "student"
|
selectedUser.type === "teacher" || selectedUser.type === "student"
|
||||||
? () => {
|
? () => {
|
||||||
appendUserFilters({
|
appendUserFilters({
|
||||||
id: "view-corporate",
|
id: "view-corporate",
|
||||||
filter: (x: User) => x.type === "corporate",
|
filter: (x: User) => x.type === "corporate",
|
||||||
});
|
});
|
||||||
appendUserFilters({
|
appendUserFilters({
|
||||||
id: "belongs-to-admin",
|
id: "belongs-to-admin",
|
||||||
filter: belongsToAdminFilter
|
filter: belongsToAdminFilter,
|
||||||
});
|
});
|
||||||
|
|
||||||
router.push("/users");
|
router.push("/users");
|
||||||
}
|
}
|
||||||
: undefined
|
: undefined
|
||||||
}
|
}
|
||||||
onClose={(shouldReload) => {
|
onClose={(shouldReload) => {
|
||||||
setSelectedUser(undefined);
|
setSelectedUser(undefined);
|
||||||
if (shouldReload) reload();
|
if (shouldReload) reload();
|
||||||
}}
|
}}
|
||||||
user={selectedUser}
|
user={selectedUser}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{renderHeader && renderHeader(displayUsers.length)}
|
{renderHeader && renderHeader(displayUsers.length)}
|
||||||
<div className="w-full">
|
<div className="w-full">
|
||||||
<Modal isOpen={!!selectedUser} onClose={() => setSelectedUser(undefined)}>
|
<Modal
|
||||||
{selectedUser && renderUserCard(selectedUser)}
|
isOpen={!!selectedUser}
|
||||||
</Modal>
|
onClose={() => setSelectedUser(undefined)}
|
||||||
<Table<WithLabeledEntities<User>>
|
>
|
||||||
data={displayUsers}
|
{selectedUser && renderUserCard(selectedUser)}
|
||||||
columns={(!showDemographicInformation ? defaultColumns : demographicColumns) as any}
|
</Modal>
|
||||||
searchFields={searchFields}
|
<Table<WithLabeledEntities<User>>
|
||||||
onDownload={entitiesDownloadUsers.length > 0 ? downloadExcel : undefined}
|
data={displayUsers}
|
||||||
/>
|
columns={
|
||||||
</div>
|
(!showDemographicInformation
|
||||||
</>
|
? defaultColumns
|
||||||
);
|
: demographicColumns) as any
|
||||||
|
}
|
||||||
|
searchFields={searchFields}
|
||||||
|
onDownload={
|
||||||
|
entitiesDownloadUsers.length > 0 ? downloadExcel : undefined
|
||||||
|
}
|
||||||
|
isLoading={isLoading}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ import clsx from "clsx";
|
|||||||
import CodeList from "./CodeList";
|
import CodeList from "./CodeList";
|
||||||
import DiscountList from "./DiscountList";
|
import DiscountList from "./DiscountList";
|
||||||
import ExamList from "./ExamList";
|
import ExamList from "./ExamList";
|
||||||
import GroupList from "./GroupList";
|
|
||||||
import PackageList from "./PackageList";
|
import PackageList from "./PackageList";
|
||||||
import UserList from "./UserList";
|
import UserList from "./UserList";
|
||||||
import { checkAccess } from "@/utils/permissions";
|
import { checkAccess } from "@/utils/permissions";
|
||||||
|
|||||||
@@ -1,24 +1,17 @@
|
|||||||
import Button from "@/components/Low/Button";
|
import Button from "@/components/Low/Button";
|
||||||
import Checkbox from "@/components/Low/Checkbox";
|
import Checkbox from "@/components/Low/Checkbox";
|
||||||
import { PERMISSIONS } from "@/constants/userPermissions";
|
import { Type, User } from "@/interfaces/user";
|
||||||
import { CorporateUser, TeacherUser, Type, User } from "@/interfaces/user";
|
|
||||||
import { USER_TYPE_LABELS } from "@/resources/user";
|
import { USER_TYPE_LABELS } from "@/resources/user";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import { capitalize, uniqBy } from "lodash";
|
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
import { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import ReactDatePicker from "react-datepicker";
|
import ReactDatePicker from "react-datepicker";
|
||||||
import { toast } from "react-toastify";
|
import { toast } from "react-toastify";
|
||||||
import ShortUniqueId from "short-unique-id";
|
|
||||||
import { checkAccess, getTypesOfUser } from "@/utils/permissions";
|
import { checkAccess, getTypesOfUser } from "@/utils/permissions";
|
||||||
import { PermissionType } from "@/interfaces/permissions";
|
import { PermissionType } from "@/interfaces/permissions";
|
||||||
import usePermissions from "@/hooks/usePermissions";
|
|
||||||
import Input from "@/components/Low/Input";
|
import Input from "@/components/Low/Input";
|
||||||
import CountrySelect from "@/components/Low/CountrySelect";
|
import CountrySelect from "@/components/Low/CountrySelect";
|
||||||
import useGroups from "@/hooks/useGroups";
|
|
||||||
import useUsers from "@/hooks/useUsers";
|
|
||||||
import { getUserName } from "@/utils/users";
|
|
||||||
import Select from "@/components/Low/Select";
|
import Select from "@/components/Low/Select";
|
||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
import useEntitiesGroups from "@/hooks/useEntitiesGroups";
|
import useEntitiesGroups from "@/hooks/useEntitiesGroups";
|
||||||
@@ -48,23 +41,44 @@ const USER_TYPE_PERMISSIONS: {
|
|||||||
},
|
},
|
||||||
admin: {
|
admin: {
|
||||||
perm: "createCodeAdmin",
|
perm: "createCodeAdmin",
|
||||||
list: ["student", "teacher", "agent", "corporate", "admin", "mastercorporate"],
|
list: [
|
||||||
|
"student",
|
||||||
|
"teacher",
|
||||||
|
"agent",
|
||||||
|
"corporate",
|
||||||
|
"admin",
|
||||||
|
"mastercorporate",
|
||||||
|
],
|
||||||
},
|
},
|
||||||
developer: {
|
developer: {
|
||||||
perm: undefined,
|
perm: undefined,
|
||||||
list: ["student", "teacher", "agent", "corporate", "admin", "developer", "mastercorporate"],
|
list: [
|
||||||
|
"student",
|
||||||
|
"teacher",
|
||||||
|
"agent",
|
||||||
|
"corporate",
|
||||||
|
"admin",
|
||||||
|
"developer",
|
||||||
|
"mastercorporate",
|
||||||
|
],
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
user: User;
|
user: User;
|
||||||
users: User[];
|
users: User[];
|
||||||
entities: EntityWithRoles[]
|
entities: EntityWithRoles[];
|
||||||
permissions: PermissionType[];
|
permissions: PermissionType[];
|
||||||
onFinish: () => void;
|
onFinish: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function UserCreator({ user, users, entities = [], permissions, onFinish }: Props) {
|
export default function UserCreator({
|
||||||
|
user,
|
||||||
|
users,
|
||||||
|
entities = [],
|
||||||
|
permissions,
|
||||||
|
onFinish,
|
||||||
|
}: Props) {
|
||||||
const [name, setName] = useState<string>();
|
const [name, setName] = useState<string>();
|
||||||
const [email, setEmail] = useState<string>();
|
const [email, setEmail] = useState<string>();
|
||||||
const [phone, setPhone] = useState<string>();
|
const [phone, setPhone] = useState<string>();
|
||||||
@@ -75,13 +89,15 @@ export default function UserCreator({ user, users, entities = [], permissions, o
|
|||||||
const [password, setPassword] = useState<string>();
|
const [password, setPassword] = useState<string>();
|
||||||
const [confirmPassword, setConfirmPassword] = useState<string>();
|
const [confirmPassword, setConfirmPassword] = useState<string>();
|
||||||
const [expiryDate, setExpiryDate] = useState<Date | null>(
|
const [expiryDate, setExpiryDate] = useState<Date | null>(
|
||||||
user?.subscriptionExpirationDate ? moment(user?.subscriptionExpirationDate).toDate() : null,
|
user?.subscriptionExpirationDate
|
||||||
|
? moment(user?.subscriptionExpirationDate).toDate()
|
||||||
|
: null
|
||||||
);
|
);
|
||||||
const [isExpiryDateEnabled, setIsExpiryDateEnabled] = useState(true);
|
const [isExpiryDateEnabled, setIsExpiryDateEnabled] = useState(true);
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [type, setType] = useState<Type>("student");
|
const [type, setType] = useState<Type>("student");
|
||||||
const [position, setPosition] = useState<string>();
|
const [position, setPosition] = useState<string>();
|
||||||
const [entity, setEntity] = useState((entities || [])[0]?.id || undefined)
|
const [entity, setEntity] = useState((entities || [])[0]?.id || undefined);
|
||||||
|
|
||||||
const { groups } = useEntitiesGroups();
|
const { groups } = useEntitiesGroups();
|
||||||
|
|
||||||
@@ -90,11 +106,16 @@ export default function UserCreator({ user, users, entities = [], permissions, o
|
|||||||
}, [isExpiryDateEnabled]);
|
}, [isExpiryDateEnabled]);
|
||||||
|
|
||||||
const createUser = () => {
|
const createUser = () => {
|
||||||
if (!name || name.trim().length === 0) return toast.error("Please enter a valid name!");
|
if (!name || name.trim().length === 0)
|
||||||
if (!email || email.trim().length === 0) return toast.error("Please enter a valid e-mail address!");
|
return toast.error("Please enter a valid name!");
|
||||||
if (users.map((x) => x.email).includes(email.trim())) return toast.error("That e-mail is already in use!");
|
if (!email || email.trim().length === 0)
|
||||||
if (!password || password.trim().length < 6) return toast.error("Please enter a valid password!");
|
return toast.error("Please enter a valid e-mail address!");
|
||||||
if (password !== confirmPassword) return toast.error("The passwords do not match!");
|
if (users.map((x) => x.email).includes(email.trim()))
|
||||||
|
return toast.error("That e-mail is already in use!");
|
||||||
|
if (!password || password.trim().length < 6)
|
||||||
|
return toast.error("Please enter a valid password!");
|
||||||
|
if (password !== confirmPassword)
|
||||||
|
return toast.error("The passwords do not match!");
|
||||||
|
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
|
|
||||||
@@ -128,8 +149,12 @@ export default function UserCreator({ user, users, entities = [], permissions, o
|
|||||||
setStudentID("");
|
setStudentID("");
|
||||||
setCountry(user?.demographicInformation?.country);
|
setCountry(user?.demographicInformation?.country);
|
||||||
setGroup(null);
|
setGroup(null);
|
||||||
setEntity((entities || [])[0]?.id || undefined)
|
setEntity((entities || [])[0]?.id || undefined);
|
||||||
setExpiryDate(user?.subscriptionExpirationDate ? moment(user?.subscriptionExpirationDate).toDate() : null);
|
setExpiryDate(
|
||||||
|
user?.subscriptionExpirationDate
|
||||||
|
? moment(user?.subscriptionExpirationDate).toDate()
|
||||||
|
: null
|
||||||
|
);
|
||||||
setIsExpiryDateEnabled(true);
|
setIsExpiryDateEnabled(true);
|
||||||
setType("student");
|
setType("student");
|
||||||
setPosition(undefined);
|
setPosition(undefined);
|
||||||
@@ -145,10 +170,34 @@ export default function UserCreator({ user, users, entities = [], permissions, o
|
|||||||
return (
|
return (
|
||||||
<div className="flex flex-col gap-4 border p-4 border-mti-gray-platinum rounded-xl">
|
<div className="flex flex-col gap-4 border p-4 border-mti-gray-platinum rounded-xl">
|
||||||
<div className="grid grid-cols-2 gap-4">
|
<div className="grid grid-cols-2 gap-4">
|
||||||
<Input required label="Name" value={name} onChange={setName} type="text" name="name" placeholder="Name" />
|
<Input
|
||||||
<Input label="E-mail" required value={email} onChange={setEmail} type="email" name="email" placeholder="E-mail" />
|
required
|
||||||
|
label="Name"
|
||||||
|
value={name}
|
||||||
|
onChange={setName}
|
||||||
|
type="text"
|
||||||
|
name="name"
|
||||||
|
placeholder="Name"
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
label="E-mail"
|
||||||
|
required
|
||||||
|
value={email}
|
||||||
|
onChange={setEmail}
|
||||||
|
type="email"
|
||||||
|
name="email"
|
||||||
|
placeholder="E-mail"
|
||||||
|
/>
|
||||||
|
|
||||||
<Input type="password" name="password" label="Password" value={password} onChange={setPassword} placeholder="Password" required />
|
<Input
|
||||||
|
type="password"
|
||||||
|
name="password"
|
||||||
|
label="Password"
|
||||||
|
value={password}
|
||||||
|
onChange={setPassword}
|
||||||
|
placeholder="Password"
|
||||||
|
required
|
||||||
|
/>
|
||||||
<Input
|
<Input
|
||||||
type="password"
|
type="password"
|
||||||
name="confirmPassword"
|
name="confirmPassword"
|
||||||
@@ -160,11 +209,21 @@ export default function UserCreator({ user, users, entities = [], permissions, o
|
|||||||
/>
|
/>
|
||||||
|
|
||||||
<div className="flex flex-col gap-4">
|
<div className="flex flex-col gap-4">
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Country *</label>
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
|
Country *
|
||||||
|
</label>
|
||||||
<CountrySelect value={country} onChange={setCountry} />
|
<CountrySelect value={country} onChange={setCountry} />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Input type="tel" name="phone" label="Phone number" value={phone} onChange={setPhone} placeholder="Phone number" required />
|
<Input
|
||||||
|
type="tel"
|
||||||
|
name="phone"
|
||||||
|
label="Phone number"
|
||||||
|
value={phone}
|
||||||
|
onChange={setPhone}
|
||||||
|
placeholder="Phone number"
|
||||||
|
required
|
||||||
|
/>
|
||||||
|
|
||||||
{type === "student" && (
|
{type === "student" && (
|
||||||
<>
|
<>
|
||||||
@@ -177,14 +236,26 @@ export default function UserCreator({ user, users, entities = [], permissions, o
|
|||||||
placeholder="National ID or Passport number"
|
placeholder="National ID or Passport number"
|
||||||
required
|
required
|
||||||
/>
|
/>
|
||||||
<Input type="text" name="studentID" label="Student ID" onChange={setStudentID} value={studentID} placeholder="Student ID" />
|
<Input
|
||||||
|
type="text"
|
||||||
|
name="studentID"
|
||||||
|
label="Student ID"
|
||||||
|
onChange={setStudentID}
|
||||||
|
value={studentID}
|
||||||
|
placeholder="Student ID"
|
||||||
|
/>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<div className={clsx("flex flex-col gap-4")}>
|
<div className={clsx("flex flex-col gap-4")}>
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Entity</label>
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
|
Entity
|
||||||
|
</label>
|
||||||
<Select
|
<Select
|
||||||
defaultValue={{ value: (entities || [])[0]?.id, label: (entities || [])[0]?.label }}
|
defaultValue={{
|
||||||
|
value: (entities || [])[0]?.id,
|
||||||
|
label: (entities || [])[0]?.label,
|
||||||
|
}}
|
||||||
options={entities.map((e) => ({ value: e.id, label: e.label }))}
|
options={entities.map((e) => ({ value: e.id, label: e.label }))}
|
||||||
onChange={(e) => setEntity(e?.value || undefined)}
|
onChange={(e) => setEntity(e?.value || undefined)}
|
||||||
isClearable={checkAccess(user, ["admin", "developer"])}
|
isClearable={checkAccess(user, ["admin", "developer"])}
|
||||||
@@ -192,11 +263,20 @@ export default function UserCreator({ user, users, entities = [], permissions, o
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
{["corporate", "mastercorporate"].includes(type) && (
|
{["corporate", "mastercorporate"].includes(type) && (
|
||||||
<Input type="text" name="department" label="Department" onChange={setPosition} value={position} placeholder="Department" />
|
<Input
|
||||||
|
type="text"
|
||||||
|
name="department"
|
||||||
|
label="Department"
|
||||||
|
onChange={setPosition}
|
||||||
|
value={position}
|
||||||
|
placeholder="Department"
|
||||||
|
/>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
<div className={clsx("flex flex-col gap-4")}>
|
<div className={clsx("flex flex-col gap-4")}>
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Classroom</label>
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
|
Classroom
|
||||||
|
</label>
|
||||||
<Select
|
<Select
|
||||||
options={groups
|
options={groups
|
||||||
.filter((x) => x.entity?.id === entity)
|
.filter((x) => x.entity?.id === entity)
|
||||||
@@ -209,63 +289,85 @@ export default function UserCreator({ user, users, entities = [], permissions, o
|
|||||||
<div
|
<div
|
||||||
className={clsx(
|
className={clsx(
|
||||||
"flex flex-col gap-4",
|
"flex flex-col gap-4",
|
||||||
!checkAccess(user, ["developer", "admin", "corporate", "mastercorporate"]) && "col-span-2",
|
!checkAccess(user, [
|
||||||
)}>
|
"developer",
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Type</label>
|
"admin",
|
||||||
|
"corporate",
|
||||||
|
"mastercorporate",
|
||||||
|
]) && "col-span-2"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
|
Type
|
||||||
|
</label>
|
||||||
{user && (
|
{user && (
|
||||||
<select
|
<select
|
||||||
defaultValue="student"
|
defaultValue="student"
|
||||||
value={type}
|
value={type}
|
||||||
onChange={(e) => setType(e.target.value as Type)}
|
onChange={(e) => setType(e.target.value as Type)}
|
||||||
className="p-6 w-full min-w-[350px] min-h-[70px] flex justify-center text-sm font-normal rounded-full border focus:outline-none cursor-pointer bg-white">
|
className="p-6 w-full min-w-[350px] min-h-[70px] flex justify-center text-sm font-normal rounded-full border focus:outline-none cursor-pointer bg-white"
|
||||||
{Object.keys(USER_TYPE_LABELS)
|
>
|
||||||
.filter((x) => {
|
{Object.keys(USER_TYPE_LABELS).reduce<string[]>((acc, x) => {
|
||||||
const { list, perm } = USER_TYPE_PERMISSIONS[x as Type];
|
const { list, perm } = USER_TYPE_PERMISSIONS[x as Type];
|
||||||
return checkAccess(user, getTypesOfUser(list), permissions, perm);
|
if (checkAccess(user, getTypesOfUser(list), permissions, perm))
|
||||||
})
|
acc.push(x);
|
||||||
.map((type) => (
|
return acc;
|
||||||
<option key={type} value={type}>
|
}, [])}
|
||||||
{USER_TYPE_LABELS[type as keyof typeof USER_TYPE_LABELS]}
|
|
||||||
</option>
|
|
||||||
))}
|
|
||||||
</select>
|
</select>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex flex-col gap-4">
|
<div className="flex flex-col gap-4">
|
||||||
{user && checkAccess(user, ["developer", "admin", "corporate", "mastercorporate"]) && (
|
{user &&
|
||||||
<>
|
checkAccess(user, [
|
||||||
<div className="-md:flex-row -md:items-center flex justify-between gap-2 md:flex-col 2xl:flex-row 2xl:items-center">
|
"developer",
|
||||||
<label className="text-mti-gray-dim text-base font-normal">Expiry Date</label>
|
"admin",
|
||||||
<Checkbox
|
"corporate",
|
||||||
isChecked={isExpiryDateEnabled}
|
"mastercorporate",
|
||||||
onChange={setIsExpiryDateEnabled}
|
]) && (
|
||||||
disabled={!!user?.subscriptionExpirationDate}>
|
<>
|
||||||
Enabled
|
<div className="-md:flex-row -md:items-center flex justify-between gap-2 md:flex-col 2xl:flex-row 2xl:items-center">
|
||||||
</Checkbox>
|
<label className="text-mti-gray-dim text-base font-normal">
|
||||||
</div>
|
Expiry Date
|
||||||
{isExpiryDateEnabled && (
|
</label>
|
||||||
<ReactDatePicker
|
<Checkbox
|
||||||
className={clsx(
|
isChecked={isExpiryDateEnabled}
|
||||||
"flex min-h-[70px] w-full cursor-pointer justify-center rounded-full border p-6 text-sm font-normal focus:outline-none",
|
onChange={setIsExpiryDateEnabled}
|
||||||
"hover:border-mti-purple tooltip",
|
disabled={!!user?.subscriptionExpirationDate}
|
||||||
"transition duration-300 ease-in-out",
|
>
|
||||||
)}
|
Enabled
|
||||||
filterDate={(date) =>
|
</Checkbox>
|
||||||
moment(date).isAfter(new Date()) &&
|
</div>
|
||||||
(user?.subscriptionExpirationDate ? moment(date).isBefore(user?.subscriptionExpirationDate) : true)
|
{isExpiryDateEnabled && (
|
||||||
}
|
<ReactDatePicker
|
||||||
dateFormat="dd/MM/yyyy"
|
className={clsx(
|
||||||
selected={expiryDate}
|
"flex min-h-[70px] w-full cursor-pointer justify-center rounded-full border p-6 text-sm font-normal focus:outline-none",
|
||||||
onChange={(date) => setExpiryDate(date)}
|
"hover:border-mti-purple tooltip",
|
||||||
/>
|
"transition duration-300 ease-in-out"
|
||||||
)}
|
)}
|
||||||
</>
|
filterDate={(date) =>
|
||||||
)}
|
moment(date).isAfter(new Date()) &&
|
||||||
|
(user?.subscriptionExpirationDate
|
||||||
|
? moment(date).isBefore(
|
||||||
|
user?.subscriptionExpirationDate
|
||||||
|
)
|
||||||
|
: true)
|
||||||
|
}
|
||||||
|
dateFormat="dd/MM/yyyy"
|
||||||
|
selected={expiryDate}
|
||||||
|
onChange={(date) => setExpiryDate(date)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Button onClick={createUser} isLoading={isLoading} disabled={(isExpiryDateEnabled ? !expiryDate : false) || isLoading}>
|
<Button
|
||||||
|
onClick={createUser}
|
||||||
|
isLoading={isLoading}
|
||||||
|
disabled={(isExpiryDateEnabled ? !expiryDate : false) || isLoading}
|
||||||
|
>
|
||||||
Create User
|
Create User
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,9 +1,9 @@
|
|||||||
/* eslint-disable @next/next/no-img-element */
|
/* eslint-disable @next/next/no-img-element */
|
||||||
import { Module } from "@/interfaces";
|
import { Module } from "@/interfaces";
|
||||||
import { useCallback, useEffect, useMemo, useRef, useState } from "react";
|
import React, { useContext, useEffect, useState } from "react";
|
||||||
|
|
||||||
import AbandonPopup from "@/components/AbandonPopup";
|
import AbandonPopup from "@/components/AbandonPopup";
|
||||||
import Layout from "@/components/High/Layout";
|
import { LayoutContext } from "@/components/High/Layout";
|
||||||
import Finish from "@/exams/Finish";
|
import Finish from "@/exams/Finish";
|
||||||
import Level from "@/exams/Level";
|
import Level from "@/exams/Level";
|
||||||
import Listening from "@/exams/Listening";
|
import Listening from "@/exams/Listening";
|
||||||
@@ -11,9 +11,12 @@ import Reading from "@/exams/Reading";
|
|||||||
import Selection from "@/exams/Selection";
|
import Selection from "@/exams/Selection";
|
||||||
import Speaking from "@/exams/Speaking";
|
import Speaking from "@/exams/Speaking";
|
||||||
import Writing from "@/exams/Writing";
|
import Writing from "@/exams/Writing";
|
||||||
import { Exam, LevelExam, UserSolution, Variant, WritingExam } from "@/interfaces/exam";
|
import { Exam, LevelExam, Variant } from "@/interfaces/exam";
|
||||||
import { User } from "@/interfaces/user";
|
import { User } from "@/interfaces/user";
|
||||||
import { evaluateSpeakingAnswer, evaluateWritingAnswer } from "@/utils/evaluation";
|
import {
|
||||||
|
evaluateSpeakingAnswer,
|
||||||
|
evaluateWritingAnswer,
|
||||||
|
} from "@/utils/evaluation";
|
||||||
import { getExam } from "@/utils/exams";
|
import { getExam } from "@/utils/exams";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
@@ -24,325 +27,436 @@ import useExamStore from "@/stores/exam";
|
|||||||
import useEvaluationPolling from "@/hooks/useEvaluationPolling";
|
import useEvaluationPolling from "@/hooks/useEvaluationPolling";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
page: "exams" | "exercises";
|
page: "exams" | "exercises";
|
||||||
user: User;
|
user: User;
|
||||||
destination?: string
|
destination?: string;
|
||||||
hideSidebar?: boolean
|
hideSidebar?: boolean;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function ExamPage({ page, user, destination = "/", hideSidebar = false }: Props) {
|
export default function ExamPage({
|
||||||
const router = useRouter();
|
page,
|
||||||
const [variant, setVariant] = useState<Variant>("full");
|
user,
|
||||||
const [avoidRepeated, setAvoidRepeated] = useState(false);
|
destination = "/",
|
||||||
const [showAbandonPopup, setShowAbandonPopup] = useState(false);
|
hideSidebar = false,
|
||||||
const [moduleLock, setModuleLock] = useState(false);
|
}: Props) {
|
||||||
|
const router = useRouter();
|
||||||
|
const [variant, setVariant] = useState<Variant>("full");
|
||||||
|
const [avoidRepeated, setAvoidRepeated] = useState(false);
|
||||||
|
const [showAbandonPopup, setShowAbandonPopup] = useState(false);
|
||||||
|
const [moduleLock, setModuleLock] = useState(false);
|
||||||
|
|
||||||
const {
|
const {
|
||||||
exam, setExam,
|
exam,
|
||||||
exams,
|
setExam,
|
||||||
sessionId, setSessionId, setPartIndex,
|
exams,
|
||||||
moduleIndex, setModuleIndex,
|
sessionId,
|
||||||
setQuestionIndex, setExerciseIndex,
|
setSessionId,
|
||||||
userSolutions, setUserSolutions,
|
setPartIndex,
|
||||||
showSolutions, setShowSolutions,
|
moduleIndex,
|
||||||
selectedModules, setSelectedModules,
|
setModuleIndex,
|
||||||
setUser,
|
setQuestionIndex,
|
||||||
inactivity,
|
setExerciseIndex,
|
||||||
timeSpent,
|
userSolutions,
|
||||||
assignment,
|
setUserSolutions,
|
||||||
bgColor,
|
showSolutions,
|
||||||
flags,
|
setShowSolutions,
|
||||||
dispatch,
|
selectedModules,
|
||||||
reset: resetStore,
|
setSelectedModules,
|
||||||
saveStats,
|
setUser,
|
||||||
saveSession,
|
inactivity,
|
||||||
setFlags,
|
timeSpent,
|
||||||
setShuffles,
|
assignment,
|
||||||
} = useExamStore();
|
bgColor,
|
||||||
|
flags,
|
||||||
|
dispatch,
|
||||||
|
reset: resetStore,
|
||||||
|
saveStats,
|
||||||
|
saveSession,
|
||||||
|
setFlags,
|
||||||
|
setShuffles,
|
||||||
|
} = useExamStore();
|
||||||
|
|
||||||
const [isFetchingExams, setIsFetchingExams] = useState(false);
|
const [isFetchingExams, setIsFetchingExams] = useState(false);
|
||||||
const [isExamLoaded, setIsExamLoaded] = useState(moduleIndex < selectedModules.length);
|
const [isExamLoaded, setIsExamLoaded] = useState(
|
||||||
|
moduleIndex < selectedModules.length
|
||||||
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setIsExamLoaded(moduleIndex < selectedModules.length);
|
setIsExamLoaded(moduleIndex < selectedModules.length);
|
||||||
}, [showSolutions, moduleIndex, selectedModules]);
|
}, [showSolutions, moduleIndex, selectedModules]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (!showSolutions && sessionId.length === 0 && user?.id) {
|
if (!showSolutions && sessionId.length === 0 && user?.id) {
|
||||||
const shortUID = new ShortUniqueId();
|
const shortUID = new ShortUniqueId();
|
||||||
setUser(user.id);
|
setUser(user.id);
|
||||||
setSessionId(shortUID.randomUUID(8));
|
setSessionId(shortUID.randomUUID(8));
|
||||||
}
|
}
|
||||||
}, [setSessionId, isExamLoaded, sessionId, showSolutions, setUser, user?.id]);
|
}, [setSessionId, isExamLoaded, sessionId, showSolutions, setUser, user?.id]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (user?.type === "developer") console.log(exam);
|
if (user?.type === "developer") console.log(exam);
|
||||||
}, [exam, user]);
|
}, [exam, user]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
(async () => {
|
(async () => {
|
||||||
if (selectedModules.length > 0 && exams.length === 0) {
|
if (selectedModules.length > 0 && exams.length === 0) {
|
||||||
setIsFetchingExams(true);
|
setIsFetchingExams(true);
|
||||||
const examPromises = selectedModules.map((module) =>
|
const examPromises = selectedModules.map((module) =>
|
||||||
getExam(
|
getExam(
|
||||||
module,
|
module,
|
||||||
avoidRepeated,
|
avoidRepeated,
|
||||||
variant,
|
variant,
|
||||||
user?.type === "student" || user?.type === "developer" ? user.preferredGender : undefined,
|
user?.type === "student" || user?.type === "developer"
|
||||||
),
|
? user.preferredGender
|
||||||
);
|
: undefined
|
||||||
Promise.all(examPromises).then((values) => {
|
)
|
||||||
setIsFetchingExams(false);
|
);
|
||||||
if (values.every((x) => !!x)) {
|
Promise.all(examPromises).then((values) => {
|
||||||
dispatch({ type: 'INIT_EXAM', payload: { exams: values.map((x) => x!), modules: selectedModules } })
|
setIsFetchingExams(false);
|
||||||
} else {
|
if (values.every((x) => !!x)) {
|
||||||
toast.error("Something went wrong, please try again");
|
dispatch({
|
||||||
setTimeout(router.reload, 500);
|
type: "INIT_EXAM",
|
||||||
}
|
payload: {
|
||||||
});
|
exams: values.map((x) => x!),
|
||||||
}
|
modules: selectedModules,
|
||||||
})();
|
},
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
});
|
||||||
}, [selectedModules, exams]);
|
} else {
|
||||||
|
toast.error("Something went wrong, please try again");
|
||||||
|
setTimeout(router.reload, 500);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, [selectedModules, exams]);
|
||||||
|
|
||||||
|
const reset = () => {
|
||||||
|
resetStore();
|
||||||
|
setVariant("full");
|
||||||
|
setAvoidRepeated(false);
|
||||||
|
setShowAbandonPopup(false);
|
||||||
|
};
|
||||||
|
|
||||||
const reset = () => {
|
useEvaluationPolling(sessionId ? [sessionId] : [], "exam", user?.id);
|
||||||
resetStore();
|
|
||||||
setVariant("full");
|
|
||||||
setAvoidRepeated(false);
|
|
||||||
setShowAbandonPopup(false);
|
|
||||||
};
|
|
||||||
|
|
||||||
useEvaluationPolling(sessionId ? [sessionId] : [], "exam", user?.id);
|
/* useEffect(() => {
|
||||||
|
setModuleLock(true);
|
||||||
|
}, [flags.finalizeModule]);
|
||||||
|
*/
|
||||||
|
useEffect(() => {
|
||||||
|
if (flags.finalizeModule && !showSolutions) {
|
||||||
|
if (
|
||||||
|
exam &&
|
||||||
|
(exam.module === "writing" || exam.module === "speaking") &&
|
||||||
|
userSolutions.length > 0
|
||||||
|
) {
|
||||||
|
(async () => {
|
||||||
|
try {
|
||||||
|
const results = await Promise.all(
|
||||||
|
exam.exercises.map(async (exercise, index) => {
|
||||||
|
if (exercise.type === "writing") {
|
||||||
|
const sol = await evaluateWritingAnswer(
|
||||||
|
user.id,
|
||||||
|
sessionId,
|
||||||
|
exercise,
|
||||||
|
index + 1,
|
||||||
|
userSolutions.find((x) => x.exercise === exercise.id)!,
|
||||||
|
exercise.attachment?.url
|
||||||
|
);
|
||||||
|
return sol;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
exercise.type === "interactiveSpeaking" ||
|
||||||
|
exercise.type === "speaking"
|
||||||
|
) {
|
||||||
|
const sol = await evaluateSpeakingAnswer(
|
||||||
|
user.id,
|
||||||
|
sessionId,
|
||||||
|
exercise,
|
||||||
|
userSolutions.find((x) => x.exercise === exercise.id)!,
|
||||||
|
index + 1
|
||||||
|
);
|
||||||
|
return sol;
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
})
|
||||||
|
);
|
||||||
|
const updatedSolutions = userSolutions.map((solution) => {
|
||||||
|
const completed = results.find(
|
||||||
|
(c: any) => c && c.exercise === solution.exercise
|
||||||
|
);
|
||||||
|
return completed || solution;
|
||||||
|
});
|
||||||
|
setUserSolutions(updatedSolutions);
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Error during module evaluation:", error);
|
||||||
|
} finally {
|
||||||
|
setModuleLock(false);
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
} else {
|
||||||
|
setModuleLock(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [
|
||||||
|
exam,
|
||||||
|
showSolutions,
|
||||||
|
userSolutions,
|
||||||
|
sessionId,
|
||||||
|
user.id,
|
||||||
|
flags.finalizeModule,
|
||||||
|
setUserSolutions,
|
||||||
|
]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setModuleLock(true);
|
if (flags.finalizeExam && moduleIndex !== -1 && !moduleLock) {
|
||||||
}, [flags.finalizeModule])
|
(async () => {
|
||||||
|
setModuleIndex(-1);
|
||||||
|
await saveStats();
|
||||||
|
await axios.get("/api/stats/update");
|
||||||
|
})();
|
||||||
|
}
|
||||||
|
}, [
|
||||||
|
flags.finalizeExam,
|
||||||
|
moduleIndex,
|
||||||
|
saveStats,
|
||||||
|
setModuleIndex,
|
||||||
|
userSolutions,
|
||||||
|
moduleLock,
|
||||||
|
flags.finalizeModule,
|
||||||
|
]);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (flags.finalizeModule && !showSolutions) {
|
if (
|
||||||
if (exam && (exam.module === "writing" || exam.module === "speaking") && userSolutions.length > 0) {
|
flags.finalizeExam &&
|
||||||
(async () => {
|
!userSolutions.some((s) => s.isDisabled) &&
|
||||||
try {
|
!moduleLock
|
||||||
const results = await Promise.all(
|
) {
|
||||||
exam.exercises.map(async (exercise, index) => {
|
setShowSolutions(true);
|
||||||
if (exercise.type === "writing") {
|
setFlags({ finalizeExam: false });
|
||||||
const sol = await evaluateWritingAnswer(
|
dispatch({ type: "UPDATE_EXAMS" });
|
||||||
user.id, sessionId, exercise, index + 1,
|
}
|
||||||
userSolutions.find((x) => x.exercise === exercise.id)!,
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
exercise.attachment?.url
|
}, [flags.finalizeExam, userSolutions, showSolutions, moduleLock]);
|
||||||
);
|
|
||||||
return sol;
|
|
||||||
}
|
|
||||||
if (exercise.type === "interactiveSpeaking" || exercise.type === "speaking") {
|
|
||||||
const sol = await evaluateSpeakingAnswer(
|
|
||||||
user.id,
|
|
||||||
sessionId,
|
|
||||||
exercise,
|
|
||||||
userSolutions.find((x) => x.exercise === exercise.id)!,
|
|
||||||
index + 1,
|
|
||||||
);
|
|
||||||
return sol;
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
})
|
|
||||||
);
|
|
||||||
const updatedSolutions = userSolutions.map(solution => {
|
|
||||||
const completed = results.filter(r => r !== null).find(
|
|
||||||
(c: any) => c.exercise === solution.exercise
|
|
||||||
);
|
|
||||||
return completed || solution;
|
|
||||||
});
|
|
||||||
setUserSolutions(updatedSolutions);
|
|
||||||
} catch (error) {
|
|
||||||
console.error('Error during module evaluation:', error);
|
|
||||||
} finally {
|
|
||||||
setModuleLock(false);
|
|
||||||
}
|
|
||||||
})();
|
|
||||||
} else {
|
|
||||||
setModuleLock(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, [exam, showSolutions, userSolutions, sessionId, user.id, flags.finalizeModule, setUserSolutions]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
const aggregateScoresByModule = (
|
||||||
if (flags.finalizeExam && moduleIndex !== -1 && !moduleLock) {
|
isPractice?: boolean
|
||||||
(async () => {
|
): {
|
||||||
setModuleIndex(-1);
|
module: Module;
|
||||||
await saveStats();
|
total: number;
|
||||||
await axios.get("/api/stats/update");
|
missing: number;
|
||||||
})()
|
correct: number;
|
||||||
}
|
}[] => {
|
||||||
}, [flags.finalizeExam, moduleIndex, saveStats, setModuleIndex, userSolutions, moduleLock, flags.finalizeModule]);
|
const scores: {
|
||||||
|
[key in Module]: { total: number; missing: number; correct: number };
|
||||||
|
} = {
|
||||||
|
reading: {
|
||||||
|
total: 0,
|
||||||
|
correct: 0,
|
||||||
|
missing: 0,
|
||||||
|
},
|
||||||
|
listening: {
|
||||||
|
total: 0,
|
||||||
|
correct: 0,
|
||||||
|
missing: 0,
|
||||||
|
},
|
||||||
|
writing: {
|
||||||
|
total: 0,
|
||||||
|
correct: 0,
|
||||||
|
missing: 0,
|
||||||
|
},
|
||||||
|
speaking: {
|
||||||
|
total: 0,
|
||||||
|
correct: 0,
|
||||||
|
missing: 0,
|
||||||
|
},
|
||||||
|
level: {
|
||||||
|
total: 0,
|
||||||
|
correct: 0,
|
||||||
|
missing: 0,
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
userSolutions.forEach((x) => {
|
||||||
if (flags.finalizeExam && !userSolutions.some(s => s.isDisabled) && !moduleLock) {
|
if (isPractice ? x.isPractice : !x.isPractice) {
|
||||||
setShowSolutions(true);
|
const examModule =
|
||||||
setFlags({ finalizeExam: false });
|
x.module ||
|
||||||
dispatch({ type: "UPDATE_EXAMS" });
|
(x.type === "writing"
|
||||||
}
|
? "writing"
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
: x.type === "speaking" || x.type === "interactiveSpeaking"
|
||||||
}, [flags.finalizeExam, userSolutions, showSolutions, moduleLock]);
|
? "speaking"
|
||||||
|
: undefined);
|
||||||
|
|
||||||
|
scores[examModule!] = {
|
||||||
|
total: scores[examModule!].total + x.score.total,
|
||||||
|
correct: scores[examModule!].correct + x.score.correct,
|
||||||
|
missing: scores[examModule!].missing + x.score.missing,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
const aggregateScoresByModule = (isPractice?: boolean): {
|
return Object.keys(scores).reduce<
|
||||||
module: Module;
|
{ module: Module; total: number; missing: number; correct: number }[]
|
||||||
total: number;
|
>((accm, x) => {
|
||||||
missing: number;
|
if (scores[x as Module].total > 0)
|
||||||
correct: number;
|
accm.push({ module: x as Module, ...scores[x as Module] });
|
||||||
}[] => {
|
return accm;
|
||||||
const scores: {
|
}, []);
|
||||||
[key in Module]: { total: number; missing: number; correct: number };
|
};
|
||||||
} = {
|
|
||||||
reading: {
|
|
||||||
total: 0,
|
|
||||||
correct: 0,
|
|
||||||
missing: 0,
|
|
||||||
},
|
|
||||||
listening: {
|
|
||||||
total: 0,
|
|
||||||
correct: 0,
|
|
||||||
missing: 0,
|
|
||||||
},
|
|
||||||
writing: {
|
|
||||||
total: 0,
|
|
||||||
correct: 0,
|
|
||||||
missing: 0,
|
|
||||||
},
|
|
||||||
speaking: {
|
|
||||||
total: 0,
|
|
||||||
correct: 0,
|
|
||||||
missing: 0,
|
|
||||||
},
|
|
||||||
level: {
|
|
||||||
total: 0,
|
|
||||||
correct: 0,
|
|
||||||
missing: 0,
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
userSolutions.filter(x => isPractice ? x.isPractice : !x.isPractice).forEach((x) => {
|
const ModuleExamMap: Record<Module, React.ComponentType<ExamProps<Exam>>> = {
|
||||||
const examModule =
|
reading: Reading as React.ComponentType<ExamProps<Exam>>,
|
||||||
x.module || (x.type === "writing" ? "writing" : x.type === "speaking" || x.type === "interactiveSpeaking" ? "speaking" : undefined);
|
listening: Listening as React.ComponentType<ExamProps<Exam>>,
|
||||||
|
writing: Writing as React.ComponentType<ExamProps<Exam>>,
|
||||||
|
speaking: Speaking as React.ComponentType<ExamProps<Exam>>,
|
||||||
|
level: Level as React.ComponentType<ExamProps<Exam>>,
|
||||||
|
};
|
||||||
|
|
||||||
scores[examModule!] = {
|
const CurrentExam = exam?.module ? ModuleExamMap[exam.module] : undefined;
|
||||||
total: scores[examModule!].total + x.score.total,
|
|
||||||
correct: scores[examModule!].correct + x.score.correct,
|
|
||||||
missing: scores[examModule!].missing + x.score.missing,
|
|
||||||
};
|
|
||||||
});
|
|
||||||
|
|
||||||
return Object.keys(scores)
|
const onAbandon = async () => {
|
||||||
.filter((x) => scores[x as Module].total > 0)
|
await saveSession();
|
||||||
.map((x) => ({ module: x as Module, ...scores[x as Module] }));
|
reset();
|
||||||
};
|
};
|
||||||
|
|
||||||
const ModuleExamMap: Record<Module, React.ComponentType<ExamProps<Exam>>> = {
|
const {
|
||||||
"reading": Reading as React.ComponentType<ExamProps<Exam>>,
|
setBgColor,
|
||||||
"listening": Listening as React.ComponentType<ExamProps<Exam>>,
|
setHideSidebar,
|
||||||
"writing": Writing as React.ComponentType<ExamProps<Exam>>,
|
setFocusMode,
|
||||||
"speaking": Speaking as React.ComponentType<ExamProps<Exam>>,
|
setOnFocusLayerMouseEnter,
|
||||||
"level": Level as React.ComponentType<ExamProps<Exam>>,
|
} = React.useContext(LayoutContext);
|
||||||
}
|
|
||||||
|
|
||||||
const CurrentExam = exam?.module ? ModuleExamMap[exam.module] : undefined;
|
useEffect(() => {
|
||||||
|
setOnFocusLayerMouseEnter(() => () => setShowAbandonPopup(true));
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, []);
|
||||||
|
|
||||||
const onAbandon = async () => {
|
useEffect(() => {
|
||||||
await saveSession();
|
setBgColor(bgColor);
|
||||||
reset();
|
setHideSidebar(hideSidebar);
|
||||||
};
|
setFocusMode(
|
||||||
|
selectedModules.length !== 0 &&
|
||||||
|
!showSolutions &&
|
||||||
|
moduleIndex < selectedModules.length
|
||||||
|
);
|
||||||
|
}, [
|
||||||
|
bgColor,
|
||||||
|
hideSidebar,
|
||||||
|
moduleIndex,
|
||||||
|
selectedModules.length,
|
||||||
|
setBgColor,
|
||||||
|
setFocusMode,
|
||||||
|
setHideSidebar,
|
||||||
|
showSolutions,
|
||||||
|
]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<ToastContainer />
|
<ToastContainer />
|
||||||
{user && (
|
{user && (
|
||||||
<Layout
|
<>
|
||||||
user={user}
|
{/* Modules weren't yet set by an INIT_EXAM or INIT_SOLUTIONS dispatch, show Selection component*/}
|
||||||
bgColor={bgColor}
|
{selectedModules.length === 0 && (
|
||||||
hideSidebar={hideSidebar}
|
<Selection
|
||||||
className="justify-between"
|
page={page}
|
||||||
focusMode={selectedModules.length !== 0 && !showSolutions && moduleIndex < selectedModules.length}
|
user={user!}
|
||||||
onFocusLayerMouseEnter={() => setShowAbandonPopup(true)}>
|
onStart={(
|
||||||
<>
|
modules: Module[],
|
||||||
{/* Modules weren't yet set by an INIT_EXAM or INIT_SOLUTIONS dispatch, show Selection component*/}
|
avoid: boolean,
|
||||||
{selectedModules.length === 0 && <Selection
|
variant: Variant
|
||||||
page={page}
|
) => {
|
||||||
user={user!}
|
setModuleIndex(0);
|
||||||
onStart={(modules: Module[], avoid: boolean, variant: Variant) => {
|
setAvoidRepeated(avoid);
|
||||||
setModuleIndex(0);
|
setSelectedModules(modules);
|
||||||
setAvoidRepeated(avoid);
|
setVariant(variant);
|
||||||
setSelectedModules(modules);
|
}}
|
||||||
setVariant(variant);
|
/>
|
||||||
}}
|
)}
|
||||||
/>}
|
{isFetchingExams && (
|
||||||
{isFetchingExams && (
|
<div className="flex flex-grow flex-col items-center justify-center animate-pulse">
|
||||||
<div className="flex flex-grow flex-col items-center justify-center animate-pulse">
|
<span
|
||||||
<span className={`loading loading-infinity w-32 bg-ielts-${selectedModules[0]}`} />
|
className={`loading loading-infinity w-32 bg-ielts-${selectedModules[0]}`}
|
||||||
<span className={`font-bold text-2xl text-ielts-${selectedModules[0]}`}>Loading Exam ...</span>
|
/>
|
||||||
</div>
|
<span
|
||||||
)}
|
className={`font-bold text-2xl text-ielts-${selectedModules[0]}`}
|
||||||
{(moduleIndex === -1 && selectedModules.length !== 0) &&
|
>
|
||||||
<Finish
|
Loading Exam ...
|
||||||
isLoading={userSolutions.some(s => s.isDisabled)}
|
</span>
|
||||||
user={user!}
|
</div>
|
||||||
modules={selectedModules}
|
)}
|
||||||
solutions={userSolutions}
|
{moduleIndex === -1 && selectedModules.length !== 0 && (
|
||||||
assignment={assignment}
|
<Finish
|
||||||
information={{
|
isLoading={userSolutions.some((s) => s.isDisabled)}
|
||||||
timeSpent,
|
user={user!}
|
||||||
inactivity,
|
modules={selectedModules}
|
||||||
}}
|
solutions={userSolutions}
|
||||||
destination={destination}
|
assignment={assignment}
|
||||||
onViewResults={(index?: number) => {
|
information={{
|
||||||
if (exams[0].module === "level") {
|
timeSpent,
|
||||||
const levelExam = exams[0] as LevelExam;
|
inactivity,
|
||||||
const allExercises = levelExam.parts.flatMap((part) => part.exercises);
|
}}
|
||||||
const exerciseOrderMap = new Map(allExercises.map((ex, index) => [ex.id, index]));
|
destination={destination}
|
||||||
const orderedSolutions = userSolutions.slice().sort((a, b) => {
|
onViewResults={(index?: number) => {
|
||||||
const indexA = exerciseOrderMap.get(a.exercise) ?? Infinity;
|
if (exams[0].module === "level") {
|
||||||
const indexB = exerciseOrderMap.get(b.exercise) ?? Infinity;
|
const levelExam = exams[0] as LevelExam;
|
||||||
return indexA - indexB;
|
const allExercises = levelExam.parts.flatMap(
|
||||||
});
|
(part) => part.exercises
|
||||||
setUserSolutions(orderedSolutions);
|
);
|
||||||
} else {
|
const exerciseOrderMap = new Map(
|
||||||
setUserSolutions(userSolutions);
|
allExercises.map((ex, index) => [ex.id, index])
|
||||||
}
|
);
|
||||||
setShuffles([]);
|
const orderedSolutions = userSolutions
|
||||||
if (index === undefined) {
|
.slice()
|
||||||
setFlags({ reviewAll: true });
|
.sort((a, b) => {
|
||||||
setModuleIndex(0);
|
const indexA =
|
||||||
setExam(exams[0]);
|
exerciseOrderMap.get(a.exercise) ?? Infinity;
|
||||||
} else {
|
const indexB =
|
||||||
setModuleIndex(index);
|
exerciseOrderMap.get(b.exercise) ?? Infinity;
|
||||||
setExam(exams[index]);
|
return indexA - indexB;
|
||||||
}
|
});
|
||||||
setShowSolutions(true);
|
setUserSolutions(orderedSolutions);
|
||||||
setQuestionIndex(0);
|
} else {
|
||||||
setExerciseIndex(0);
|
setUserSolutions(userSolutions);
|
||||||
setPartIndex(0);
|
}
|
||||||
}}
|
setShuffles([]);
|
||||||
scores={aggregateScoresByModule()}
|
if (index === undefined) {
|
||||||
practiceScores={aggregateScoresByModule(true)}
|
setFlags({ reviewAll: true });
|
||||||
/>}
|
setModuleIndex(0);
|
||||||
{/* Exam is on going, display it and the abandon modal */}
|
setExam(exams[0]);
|
||||||
{isExamLoaded && moduleIndex !== -1 && (
|
} else {
|
||||||
<>
|
setModuleIndex(index);
|
||||||
{exam && CurrentExam && <CurrentExam exam={exam} showSolutions={showSolutions} />}
|
setExam(exams[index]);
|
||||||
{!showSolutions && <AbandonPopup
|
}
|
||||||
isOpen={showAbandonPopup}
|
setShowSolutions(true);
|
||||||
abandonPopupTitle="Leave Exercise"
|
setQuestionIndex(0);
|
||||||
abandonPopupDescription="Are you sure you want to leave the exercise? Your progress will be saved and this exam can be resumed on the Dashboard."
|
setExerciseIndex(0);
|
||||||
abandonConfirmButtonText="Confirm"
|
setPartIndex(0);
|
||||||
onAbandon={onAbandon}
|
}}
|
||||||
onCancel={() => setShowAbandonPopup(false)}
|
scores={aggregateScoresByModule()}
|
||||||
/>
|
practiceScores={aggregateScoresByModule(true)}
|
||||||
}
|
/>
|
||||||
</>
|
)}
|
||||||
)}
|
{/* Exam is on going, display it and the abandon modal */}
|
||||||
</>
|
{isExamLoaded && moduleIndex !== -1 && (
|
||||||
</Layout>
|
<>
|
||||||
)}
|
{exam && CurrentExam && (
|
||||||
</>
|
<CurrentExam exam={exam} showSolutions={showSolutions} />
|
||||||
);
|
)}
|
||||||
|
{!showSolutions && (
|
||||||
|
<AbandonPopup
|
||||||
|
isOpen={showAbandonPopup}
|
||||||
|
abandonPopupTitle="Leave Exercise"
|
||||||
|
abandonPopupDescription="Are you sure you want to leave the exercise? Your progress will be saved and this exam can be resumed on the Dashboard."
|
||||||
|
abandonConfirmButtonText="Confirm"
|
||||||
|
onAbandon={onAbandon}
|
||||||
|
onCancel={() => setShowAbandonPopup(false)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -13,267 +13,268 @@ import moment from "moment";
|
|||||||
import useAcceptedTerms from "@/hooks/useAcceptedTerms";
|
import useAcceptedTerms from "@/hooks/useAcceptedTerms";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
isLoading: boolean;
|
isLoading: boolean;
|
||||||
setIsLoading: (isLoading: boolean) => void;
|
setIsLoading: (isLoading: boolean) => void;
|
||||||
mutateUser: KeyedMutator<User>;
|
mutateUser: KeyedMutator<User>;
|
||||||
sendEmailVerification: typeof sendEmailVerification;
|
sendEmailVerification: typeof sendEmailVerification;
|
||||||
}
|
}
|
||||||
|
|
||||||
const availableDurations = {
|
const availableDurations = {
|
||||||
"1_month": { label: "1 Month", number: 1 },
|
"1_month": { label: "1 Month", number: 1 },
|
||||||
"3_months": { label: "3 Months", number: 3 },
|
"3_months": { label: "3 Months", number: 3 },
|
||||||
"6_months": { label: "6 Months", number: 6 },
|
"6_months": { label: "6 Months", number: 6 },
|
||||||
"12_months": { label: "12 Months", number: 12 },
|
"12_months": { label: "12 Months", number: 12 },
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function RegisterCorporate({
|
export default function RegisterCorporate({
|
||||||
isLoading,
|
isLoading,
|
||||||
setIsLoading,
|
setIsLoading,
|
||||||
mutateUser,
|
mutateUser,
|
||||||
sendEmailVerification,
|
sendEmailVerification,
|
||||||
}: Props) {
|
}: Props) {
|
||||||
const [name, setName] = useState("");
|
const [name, setName] = useState("");
|
||||||
const [email, setEmail] = useState("");
|
const [email, setEmail] = useState("");
|
||||||
const [password, setPassword] = useState("");
|
const [password, setPassword] = useState("");
|
||||||
const [confirmPassword, setConfirmPassword] = useState("");
|
const [confirmPassword, setConfirmPassword] = useState("");
|
||||||
const [referralAgent, setReferralAgent] = useState<string | undefined>();
|
const [referralAgent, setReferralAgent] = useState<string | undefined>();
|
||||||
|
|
||||||
const [companyName, setCompanyName] = useState("");
|
const [companyName, setCompanyName] = useState("");
|
||||||
const [companyUsers, setCompanyUsers] = useState(0);
|
const [companyUsers, setCompanyUsers] = useState(0);
|
||||||
const [subscriptionDuration, setSubscriptionDuration] = useState(1);
|
const [subscriptionDuration, setSubscriptionDuration] = useState(1);
|
||||||
const { acceptedTerms, renderCheckbox } = useAcceptedTerms();
|
const { acceptedTerms, renderCheckbox } = useAcceptedTerms();
|
||||||
|
|
||||||
const { users } = useUsers();
|
const { users } = useUsers({ type: "agent" });
|
||||||
|
|
||||||
const onSuccess = () =>
|
const onSuccess = () =>
|
||||||
toast.success(
|
toast.success(
|
||||||
"An e-mail has been sent, please make sure to check your spam folder!",
|
"An e-mail has been sent, please make sure to check your spam folder!"
|
||||||
);
|
);
|
||||||
|
|
||||||
const onError = (e: Error) => {
|
const onError = (e: Error) => {
|
||||||
console.error(e);
|
console.error(e);
|
||||||
toast.error("Something went wrong, please logout and re-login.", {
|
toast.error("Something went wrong, please logout and re-login.", {
|
||||||
toastId: "send-verify-error",
|
toastId: "send-verify-error",
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const register = (e: any) => {
|
const register = (e: any) => {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
|
|
||||||
if (confirmPassword !== password) {
|
if (confirmPassword !== password) {
|
||||||
toast.error("Your passwords do not match!", {
|
toast.error("Your passwords do not match!", {
|
||||||
toastId: "password-not-match",
|
toastId: "password-not-match",
|
||||||
});
|
});
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
axios
|
axios
|
||||||
.post("/api/register", {
|
.post("/api/register", {
|
||||||
name,
|
name,
|
||||||
email,
|
email,
|
||||||
password,
|
password,
|
||||||
type: "corporate",
|
type: "corporate",
|
||||||
profilePicture: "/defaultAvatar.png",
|
profilePicture: "/defaultAvatar.png",
|
||||||
subscriptionExpirationDate: moment().subtract(1, "days").toISOString(),
|
subscriptionExpirationDate: moment().subtract(1, "days").toISOString(),
|
||||||
corporateInformation: {
|
corporateInformation: {
|
||||||
monthlyDuration: subscriptionDuration,
|
monthlyDuration: subscriptionDuration,
|
||||||
referralAgent,
|
referralAgent,
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
.then((response) => {
|
.then((response) => {
|
||||||
mutateUser(response.data.user).then(() =>
|
mutateUser(response.data.user).then(() =>
|
||||||
sendEmailVerification(setIsLoading, onSuccess, onError),
|
sendEmailVerification(setIsLoading, onSuccess, onError)
|
||||||
);
|
);
|
||||||
})
|
})
|
||||||
.catch((error) => {
|
.catch((error) => {
|
||||||
console.log(error.response.data);
|
console.log(error.response.data);
|
||||||
|
|
||||||
if (error.response.status === 401) {
|
if (error.response.status === 401) {
|
||||||
toast.error("There is already a user with that e-mail!");
|
toast.error("There is already a user with that e-mail!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (error.response.status === 400) {
|
if (error.response.status === 400) {
|
||||||
toast.error("The provided code is invalid!");
|
toast.error("The provided code is invalid!");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
toast.error("There was something wrong, please try again!");
|
toast.error("There was something wrong, please try again!");
|
||||||
})
|
})
|
||||||
.finally(() => setIsLoading(false));
|
.finally(() => setIsLoading(false));
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<form
|
<form
|
||||||
className="flex w-full flex-col items-center gap-4"
|
className="flex w-full flex-col items-center gap-4"
|
||||||
onSubmit={register}
|
onSubmit={register}
|
||||||
>
|
>
|
||||||
<div className="flex w-full gap-4">
|
<div className="flex w-full gap-4">
|
||||||
<Input
|
<Input
|
||||||
type="text"
|
type="text"
|
||||||
name="name"
|
name="name"
|
||||||
onChange={(e) => setName(e)}
|
onChange={(e) => setName(e)}
|
||||||
placeholder="Enter your name"
|
placeholder="Enter your name"
|
||||||
defaultValue={name}
|
defaultValue={name}
|
||||||
required
|
required
|
||||||
/>
|
/>
|
||||||
<Input
|
<Input
|
||||||
type="email"
|
type="email"
|
||||||
name="email"
|
name="email"
|
||||||
onChange={(e) => setEmail(e.toLowerCase())}
|
onChange={(e) => setEmail(e.toLowerCase())}
|
||||||
placeholder="Enter email address"
|
placeholder="Enter email address"
|
||||||
defaultValue={email}
|
defaultValue={email}
|
||||||
required
|
required
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex w-full gap-4">
|
<div className="flex w-full gap-4">
|
||||||
<Input
|
<Input
|
||||||
type="password"
|
type="password"
|
||||||
name="password"
|
name="password"
|
||||||
onChange={(e) => setPassword(e)}
|
onChange={(e) => setPassword(e)}
|
||||||
placeholder="Enter your password"
|
placeholder="Enter your password"
|
||||||
defaultValue={password}
|
defaultValue={password}
|
||||||
required
|
required
|
||||||
/>
|
/>
|
||||||
<Input
|
<Input
|
||||||
type="password"
|
type="password"
|
||||||
name="confirmPassword"
|
name="confirmPassword"
|
||||||
onChange={(e) => setConfirmPassword(e)}
|
onChange={(e) => setConfirmPassword(e)}
|
||||||
placeholder="Confirm your password"
|
placeholder="Confirm your password"
|
||||||
defaultValue={confirmPassword}
|
defaultValue={confirmPassword}
|
||||||
required
|
required
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<Divider className="!my-2 w-full" />
|
<Divider className="!my-2 w-full" />
|
||||||
|
|
||||||
<div className="flex w-full gap-4">
|
<div className="flex w-full gap-4">
|
||||||
<Input
|
<Input
|
||||||
type="text"
|
type="text"
|
||||||
name="companyName"
|
name="companyName"
|
||||||
onChange={(e) => setCompanyName(e)}
|
onChange={(e) => setCompanyName(e)}
|
||||||
placeholder="Corporate name"
|
placeholder="Corporate name"
|
||||||
label="Corporate name"
|
label="Corporate name"
|
||||||
defaultValue={companyName}
|
defaultValue={companyName}
|
||||||
required
|
required
|
||||||
/>
|
/>
|
||||||
<Input
|
<Input
|
||||||
type="number"
|
type="number"
|
||||||
name="companyUsers"
|
name="companyUsers"
|
||||||
onChange={(e) => setCompanyUsers(parseInt(e))}
|
onChange={(e) => setCompanyUsers(parseInt(e))}
|
||||||
label="Number of users"
|
label="Number of users"
|
||||||
defaultValue={companyUsers}
|
defaultValue={companyUsers}
|
||||||
required
|
required
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div className="flex w-full gap-4">
|
<div className="flex w-full gap-4">
|
||||||
<div className="flex w-full flex-col gap-3">
|
<div className="flex w-full flex-col gap-3">
|
||||||
<label className="text-mti-gray-dim text-base font-normal">
|
<label className="text-mti-gray-dim text-base font-normal">
|
||||||
Referral *
|
Referral *
|
||||||
</label>
|
</label>
|
||||||
<Select
|
<Select
|
||||||
className="placeholder:text-mti-gray-cool disabled:bg-mti-gray-platinum/40 disabled:text-mti-gray-dim border-mti-gray-platinum w-full rounded-full border bg-white px-4 py-4 text-sm font-normal focus:outline-none disabled:cursor-not-allowed"
|
className="placeholder:text-mti-gray-cool disabled:bg-mti-gray-platinum/40 disabled:text-mti-gray-dim border-mti-gray-platinum w-full rounded-full border bg-white px-4 py-4 text-sm font-normal focus:outline-none disabled:cursor-not-allowed"
|
||||||
options={[
|
options={[
|
||||||
{ value: "", label: "No referral" },
|
{ value: "", label: "No referral" },
|
||||||
...users
|
...users.map((x) => ({
|
||||||
.filter((u) => u.type === "agent")
|
value: x.id,
|
||||||
.map((x) => ({ value: x.id, label: `${x.name} - ${x.email}` })),
|
label: `${x.name} - ${x.email}`,
|
||||||
]}
|
})),
|
||||||
defaultValue={{ value: "", label: "No referral" }}
|
]}
|
||||||
onChange={(value) => setReferralAgent(value?.value)}
|
defaultValue={{ value: "", label: "No referral" }}
|
||||||
styles={{
|
onChange={(value) => setReferralAgent(value?.value)}
|
||||||
control: (styles) => ({
|
styles={{
|
||||||
...styles,
|
control: (styles) => ({
|
||||||
paddingLeft: "4px",
|
...styles,
|
||||||
border: "none",
|
paddingLeft: "4px",
|
||||||
outline: "none",
|
border: "none",
|
||||||
":focus": {
|
outline: "none",
|
||||||
outline: "none",
|
":focus": {
|
||||||
},
|
outline: "none",
|
||||||
}),
|
},
|
||||||
option: (styles, state) => ({
|
}),
|
||||||
...styles,
|
option: (styles, state) => ({
|
||||||
backgroundColor: state.isFocused
|
...styles,
|
||||||
? "#D5D9F0"
|
backgroundColor: state.isFocused
|
||||||
: state.isSelected
|
? "#D5D9F0"
|
||||||
? "#7872BF"
|
: state.isSelected
|
||||||
: "white",
|
? "#7872BF"
|
||||||
color: state.isFocused ? "black" : styles.color,
|
: "white",
|
||||||
}),
|
color: state.isFocused ? "black" : styles.color,
|
||||||
}}
|
}),
|
||||||
/>
|
}}
|
||||||
</div>
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div className="flex w-full flex-col gap-3">
|
<div className="flex w-full flex-col gap-3">
|
||||||
<label className="text-mti-gray-dim text-base font-normal">
|
<label className="text-mti-gray-dim text-base font-normal">
|
||||||
Subscription Duration *
|
Subscription Duration *
|
||||||
</label>
|
</label>
|
||||||
<Select
|
<Select
|
||||||
className="placeholder:text-mti-gray-cool disabled:bg-mti-gray-platinum/40 disabled:text-mti-gray-dim border-mti-gray-platinum w-full rounded-full border bg-white px-4 py-4 text-sm font-normal focus:outline-none disabled:cursor-not-allowed"
|
className="placeholder:text-mti-gray-cool disabled:bg-mti-gray-platinum/40 disabled:text-mti-gray-dim border-mti-gray-platinum w-full rounded-full border bg-white px-4 py-4 text-sm font-normal focus:outline-none disabled:cursor-not-allowed"
|
||||||
options={Object.keys(availableDurations).map((value) => ({
|
options={Object.keys(availableDurations).map((value) => ({
|
||||||
value,
|
value,
|
||||||
label:
|
label:
|
||||||
availableDurations[value as keyof typeof availableDurations]
|
availableDurations[value as keyof typeof availableDurations]
|
||||||
.label,
|
.label,
|
||||||
}))}
|
}))}
|
||||||
defaultValue={{
|
defaultValue={{
|
||||||
value: "1_month",
|
value: "1_month",
|
||||||
label: availableDurations["1_month"].label,
|
label: availableDurations["1_month"].label,
|
||||||
}}
|
}}
|
||||||
onChange={(value) =>
|
onChange={(value) =>
|
||||||
setSubscriptionDuration(
|
setSubscriptionDuration(
|
||||||
value
|
value
|
||||||
? availableDurations[
|
? availableDurations[
|
||||||
value.value as keyof typeof availableDurations
|
value.value as keyof typeof availableDurations
|
||||||
].number
|
].number
|
||||||
: 1,
|
: 1
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
styles={{
|
styles={{
|
||||||
control: (styles) => ({
|
control: (styles) => ({
|
||||||
...styles,
|
...styles,
|
||||||
paddingLeft: "4px",
|
paddingLeft: "4px",
|
||||||
border: "none",
|
border: "none",
|
||||||
outline: "none",
|
outline: "none",
|
||||||
":focus": {
|
":focus": {
|
||||||
outline: "none",
|
outline: "none",
|
||||||
},
|
},
|
||||||
}),
|
}),
|
||||||
option: (styles, state) => ({
|
option: (styles, state) => ({
|
||||||
...styles,
|
...styles,
|
||||||
backgroundColor: state.isFocused
|
backgroundColor: state.isFocused
|
||||||
? "#D5D9F0"
|
? "#D5D9F0"
|
||||||
: state.isSelected
|
: state.isSelected
|
||||||
? "#7872BF"
|
? "#7872BF"
|
||||||
: "white",
|
: "white",
|
||||||
color: state.isFocused ? "black" : styles.color,
|
color: state.isFocused ? "black" : styles.color,
|
||||||
}),
|
}),
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex w-full flex-col items-start gap-4">
|
<div className="flex w-full flex-col items-start gap-4">
|
||||||
{renderCheckbox()}
|
{renderCheckbox()}
|
||||||
</div>
|
</div>
|
||||||
<Button
|
<Button
|
||||||
className="w-full lg:mt-8"
|
className="w-full lg:mt-8"
|
||||||
color="purple"
|
color="purple"
|
||||||
disabled={
|
disabled={
|
||||||
isLoading ||
|
isLoading ||
|
||||||
!email ||
|
!email ||
|
||||||
!name ||
|
!name ||
|
||||||
!password ||
|
!password ||
|
||||||
!confirmPassword ||
|
!confirmPassword ||
|
||||||
password !== confirmPassword ||
|
password !== confirmPassword ||
|
||||||
!companyName ||
|
!companyName ||
|
||||||
companyUsers <= 0
|
companyUsers <= 0
|
||||||
}
|
}
|
||||||
>
|
>
|
||||||
Create account
|
Create account
|
||||||
</Button>
|
</Button>
|
||||||
</form>
|
</form>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,18 +1,14 @@
|
|||||||
/* eslint-disable @next/next/no-img-element */
|
/* eslint-disable @next/next/no-img-element */
|
||||||
import Layout from "@/components/High/Layout";
|
|
||||||
import useGroups from "@/hooks/useGroups";
|
|
||||||
import usePackages from "@/hooks/usePackages";
|
|
||||||
import useUsers from "@/hooks/useUsers";
|
import useUsers from "@/hooks/useUsers";
|
||||||
import { User } from "@/interfaces/user";
|
import { User } from "@/interfaces/user";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import { capitalize, sortBy } from "lodash";
|
import { capitalize } from "lodash";
|
||||||
import { useEffect, useMemo, useState } from "react";
|
import { useEffect, useMemo, useState } from "react";
|
||||||
import useInvites from "@/hooks/useInvites";
|
import useInvites from "@/hooks/useInvites";
|
||||||
import { BsArrowRepeat } from "react-icons/bs";
|
import { BsArrowRepeat } from "react-icons/bs";
|
||||||
import InviteCard from "@/components/Medium/InviteCard";
|
import InviteCard from "@/components/Medium/InviteCard";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { ToastContainer } from "react-toastify";
|
import { ToastContainer } from "react-toastify";
|
||||||
import useDiscounts from "@/hooks/useDiscounts";
|
|
||||||
import PaymobPayment from "@/components/PaymobPayment";
|
import PaymobPayment from "@/components/PaymobPayment";
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
@@ -22,241 +18,345 @@ import { useAllowedEntities } from "@/hooks/useEntityPermissions";
|
|||||||
import Select from "@/components/Low/Select";
|
import Select from "@/components/Low/Select";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
user: User
|
user: User;
|
||||||
discounts: Discount[]
|
discounts: Discount[];
|
||||||
packages: Package[]
|
packages: Package[];
|
||||||
entities: EntityWithRoles[]
|
entities: EntityWithRoles[];
|
||||||
hasExpired?: boolean;
|
hasExpired?: boolean;
|
||||||
reload: () => void;
|
reload: () => void;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function PaymentDue({ user, discounts = [], entities = [], packages = [], hasExpired = false, reload }: Props) {
|
export default function PaymentDue({
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
user,
|
||||||
const [entity, setEntity] = useState<EntityWithRoles>()
|
discounts = [],
|
||||||
|
entities = [],
|
||||||
|
packages = [],
|
||||||
|
hasExpired = false,
|
||||||
|
reload,
|
||||||
|
}: Props) {
|
||||||
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
|
const [entity, setEntity] = useState<EntityWithRoles>();
|
||||||
|
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
const { users } = useUsers();
|
const { users } = useUsers();
|
||||||
const { invites, isLoading: isInvitesLoading, reload: reloadInvites } = useInvites({ to: user?.id });
|
const {
|
||||||
|
invites,
|
||||||
|
isLoading: isInvitesLoading,
|
||||||
|
reload: reloadInvites,
|
||||||
|
} = useInvites({ to: user?.id });
|
||||||
|
|
||||||
const isIndividual = useMemo(() => {
|
const isIndividual = useMemo(() => {
|
||||||
if (isAdmin(user)) return false;
|
if (isAdmin(user)) return false;
|
||||||
if (user?.type !== "student") return false;
|
if (user?.type !== "student") return false;
|
||||||
|
|
||||||
return user.entities.length === 0
|
return user.entities.length === 0;
|
||||||
}, [user])
|
}, [user]);
|
||||||
|
|
||||||
const appliedDiscount = useMemo(() => {
|
const appliedDiscount = useMemo(() => {
|
||||||
const biggestDiscount = [...discounts].sort((a, b) => b.percentage - a.percentage).shift();
|
const biggestDiscount = [...discounts]
|
||||||
|
.sort((a, b) => b.percentage - a.percentage)
|
||||||
|
.shift();
|
||||||
|
|
||||||
if (!biggestDiscount || (biggestDiscount.validUntil && moment(biggestDiscount.validUntil).isBefore(moment())))
|
if (
|
||||||
return 0;
|
!biggestDiscount ||
|
||||||
|
(biggestDiscount.validUntil &&
|
||||||
|
moment(biggestDiscount.validUntil).isBefore(moment()))
|
||||||
|
)
|
||||||
|
return 0;
|
||||||
|
|
||||||
return biggestDiscount.percentage
|
return biggestDiscount.percentage;
|
||||||
}, [discounts])
|
}, [discounts]);
|
||||||
|
|
||||||
const entitiesThatCanBePaid = useAllowedEntities(user, entities, 'pay_entity')
|
const entitiesThatCanBePaid = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"pay_entity"
|
||||||
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (entitiesThatCanBePaid.length > 0) setEntity(entitiesThatCanBePaid[0])
|
if (entitiesThatCanBePaid.length > 0) setEntity(entitiesThatCanBePaid[0]);
|
||||||
}, [entitiesThatCanBePaid])
|
}, [entitiesThatCanBePaid]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<ToastContainer />
|
<ToastContainer />
|
||||||
{isLoading && (
|
{isLoading && (
|
||||||
<div className="absolute left-0 top-0 z-[999] h-screen w-screen overflow-hidden bg-black/60">
|
<div className="absolute left-0 top-0 z-[999] h-screen w-screen overflow-hidden bg-black/60">
|
||||||
<div className="absolute left-1/2 top-1/2 flex h-fit w-fit -translate-x-1/2 -translate-y-1/2 flex-col items-center gap-8 text-white">
|
<div className="absolute left-1/2 top-1/2 flex h-fit w-fit -translate-x-1/2 -translate-y-1/2 flex-col items-center gap-8 text-white">
|
||||||
<span className={clsx("loading loading-infinity w-48 animate-pulse")} />
|
<span
|
||||||
<span className={clsx("text-2xl font-bold animate-pulse")}>Completing your payment...</span>
|
className={clsx("loading loading-infinity w-48 animate-pulse")}
|
||||||
<span>If you canceled your payment or it failed, please click the button below to restart</span>
|
/>
|
||||||
<button
|
<span className={clsx("text-2xl font-bold animate-pulse")}>
|
||||||
onClick={() => setIsLoading(false)}
|
Completing your payment...
|
||||||
className="border border-white rounded-full px-4 py-2 hover:bg-white/80 hover:text-black cursor-pointer transition ease-in-out duration-300">
|
</span>
|
||||||
Cancel Payment
|
<span>
|
||||||
</button>
|
If you canceled your payment or it failed, please click the button
|
||||||
</div>
|
below to restart
|
||||||
</div>
|
</span>
|
||||||
)}
|
<button
|
||||||
<Layout user={user} navDisabled={hasExpired}>
|
onClick={() => setIsLoading(false)}
|
||||||
{invites.length > 0 && (
|
className="border border-white rounded-full px-4 py-2 hover:bg-white/80 hover:text-black cursor-pointer transition ease-in-out duration-300"
|
||||||
<section className="flex flex-col gap-1 md:gap-3">
|
>
|
||||||
<div className="flex items-center gap-4">
|
Cancel Payment
|
||||||
<div
|
</button>
|
||||||
onClick={reloadInvites}
|
</div>
|
||||||
className="text-mti-purple-light hover:text-mti-purple-dark flex cursor-pointer items-center gap-2 transition duration-300 ease-in-out">
|
</div>
|
||||||
<span className="text-mti-black text-lg font-bold">Invites</span>
|
)}
|
||||||
<BsArrowRepeat className={clsx("text-xl", isInvitesLoading && "animate-spin")} />
|
<>
|
||||||
</div>
|
{invites.length > 0 && (
|
||||||
</div>
|
<section className="flex flex-col gap-1 md:gap-3">
|
||||||
<span className="text-mti-gray-taupe scrollbar-hide flex gap-8 overflow-x-scroll">
|
<div className="flex items-center gap-4">
|
||||||
{invites.map((invite) => (
|
<div
|
||||||
<InviteCard
|
onClick={reloadInvites}
|
||||||
key={invite.id}
|
className="text-mti-purple-light hover:text-mti-purple-dark flex cursor-pointer items-center gap-2 transition duration-300 ease-in-out"
|
||||||
invite={invite}
|
>
|
||||||
users={users}
|
<span className="text-mti-black text-lg font-bold">
|
||||||
reload={() => {
|
Invites
|
||||||
reloadInvites();
|
</span>
|
||||||
router.reload();
|
<BsArrowRepeat
|
||||||
}}
|
className={clsx(
|
||||||
/>
|
"text-xl",
|
||||||
))}
|
isInvitesLoading && "animate-spin"
|
||||||
</span>
|
)}
|
||||||
</section>
|
/>
|
||||||
)}
|
</div>
|
||||||
|
</div>
|
||||||
|
<span className="text-mti-gray-taupe scrollbar-hide flex gap-8 overflow-x-scroll">
|
||||||
|
{invites.map((invite) => (
|
||||||
|
<InviteCard
|
||||||
|
key={invite.id}
|
||||||
|
invite={invite}
|
||||||
|
users={users}
|
||||||
|
reload={() => {
|
||||||
|
reloadInvites();
|
||||||
|
router.reload();
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</span>
|
||||||
|
</section>
|
||||||
|
)}
|
||||||
|
|
||||||
<div className="flex w-full flex-col items-center justify-center gap-4 text-center">
|
<div className="flex w-full flex-col items-center justify-center gap-4 text-center">
|
||||||
{hasExpired && <span className="text-lg font-bold">You do not have time credits for your account type!</span>}
|
{hasExpired && (
|
||||||
{isIndividual && (
|
<span className="text-lg font-bold">
|
||||||
<div className="scrollbar-hide flex w-full flex-col items-center gap-12 overflow-x-scroll">
|
You do not have time credits for your account type!
|
||||||
<span className="max-w-lg">
|
</span>
|
||||||
To add to your use of EnCoach, please purchase one of the time packages available below:
|
)}
|
||||||
</span>
|
{isIndividual && (
|
||||||
<div className="flex w-full flex-wrap justify-center gap-8">
|
<div className="scrollbar-hide flex w-full flex-col items-center gap-12 overflow-x-scroll">
|
||||||
{packages.map((p) => (
|
<span className="max-w-lg">
|
||||||
<div key={p.id} className={clsx("flex flex-col items-start gap-6 rounded-xl bg-white p-4")}>
|
To add to your use of EnCoach, please purchase one of the time
|
||||||
<div className="mb-2 flex flex-col items-start">
|
packages available below:
|
||||||
<img src="/logo_title.png" alt="EnCoach's Logo" className="w-32" />
|
</span>
|
||||||
<span className="text-xl font-semibold">
|
<div className="flex w-full flex-wrap justify-center gap-8">
|
||||||
EnCoach - {p.duration}{" "}
|
{packages.map((p) => (
|
||||||
{capitalize(
|
<div
|
||||||
p.duration === 1 ? p.duration_unit.slice(0, p.duration_unit.length - 1) : p.duration_unit,
|
key={p.id}
|
||||||
)}
|
className={clsx(
|
||||||
</span>
|
"flex flex-col items-start gap-6 rounded-xl bg-white p-4"
|
||||||
</div>
|
)}
|
||||||
<div className="flex w-full flex-col items-start gap-2">
|
>
|
||||||
{appliedDiscount === 0 && (
|
<div className="mb-2 flex flex-col items-start">
|
||||||
<span className="text-2xl">
|
<img
|
||||||
{p.price} {p.currency}
|
src="/logo_title.png"
|
||||||
</span>
|
alt="EnCoach's Logo"
|
||||||
)}
|
className="w-32"
|
||||||
{appliedDiscount > 0 && (
|
/>
|
||||||
<div className="flex items-center gap-2">
|
<span className="text-xl font-semibold">
|
||||||
<span className="text-2xl line-through">
|
EnCoach - {p.duration}{" "}
|
||||||
{p.price} {p.currency}
|
{capitalize(
|
||||||
</span>
|
p.duration === 1
|
||||||
<span className="text-2xl text-mti-red-light">
|
? p.duration_unit.slice(
|
||||||
{(p.price - p.price * (appliedDiscount / 100)).toFixed(2)} {p.currency}
|
0,
|
||||||
</span>
|
p.duration_unit.length - 1
|
||||||
</div>
|
)
|
||||||
)}
|
: p.duration_unit
|
||||||
<PaymobPayment
|
)}
|
||||||
user={user}
|
</span>
|
||||||
setIsPaymentLoading={setIsLoading}
|
</div>
|
||||||
onSuccess={() => {
|
<div className="flex w-full flex-col items-start gap-2">
|
||||||
setTimeout(reload, 500);
|
{appliedDiscount === 0 && (
|
||||||
}}
|
<span className="text-2xl">
|
||||||
currency={p.currency}
|
{p.price} {p.currency}
|
||||||
duration={p.duration}
|
</span>
|
||||||
duration_unit={p.duration_unit}
|
)}
|
||||||
price={+(p.price - p.price * (appliedDiscount / 100)).toFixed(2)}
|
{appliedDiscount > 0 && (
|
||||||
/>
|
<div className="flex items-center gap-2">
|
||||||
</div>
|
<span className="text-2xl line-through">
|
||||||
<div className="flex flex-col items-start gap-1">
|
{p.price} {p.currency}
|
||||||
<span>This includes:</span>
|
</span>
|
||||||
<ul className="flex flex-col items-start text-sm">
|
<span className="text-2xl text-mti-red-light">
|
||||||
<li>- Train your abilities for the IELTS exam</li>
|
{(
|
||||||
<li>- Gain insights into your weaknesses and strengths</li>
|
p.price -
|
||||||
<li>- Allow yourself to correctly prepare for the exam</li>
|
p.price * (appliedDiscount / 100)
|
||||||
</ul>
|
).toFixed(2)}{" "}
|
||||||
</div>
|
{p.currency}
|
||||||
</div>
|
</span>
|
||||||
))}
|
</div>
|
||||||
</div>
|
)}
|
||||||
</div>
|
<PaymobPayment
|
||||||
)}
|
user={user}
|
||||||
|
setIsPaymentLoading={setIsLoading}
|
||||||
|
onSuccess={() => {
|
||||||
|
setTimeout(reload, 500);
|
||||||
|
}}
|
||||||
|
currency={p.currency}
|
||||||
|
duration={p.duration}
|
||||||
|
duration_unit={p.duration_unit}
|
||||||
|
price={
|
||||||
|
+(
|
||||||
|
p.price -
|
||||||
|
p.price * (appliedDiscount / 100)
|
||||||
|
).toFixed(2)
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col items-start gap-1">
|
||||||
|
<span>This includes:</span>
|
||||||
|
<ul className="flex flex-col items-start text-sm">
|
||||||
|
<li>- Train your abilities for the IELTS exam</li>
|
||||||
|
<li>
|
||||||
|
- Gain insights into your weaknesses and strengths
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
- Allow yourself to correctly prepare for the exam
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
{!isIndividual && entitiesThatCanBePaid.length > 0 &&
|
{!isIndividual &&
|
||||||
entity?.payment && (
|
entitiesThatCanBePaid.length > 0 &&
|
||||||
<div className="flex flex-col items-center gap-8">
|
entity?.payment && (
|
||||||
<div className={clsx("flex flex-col items-center gap-4 w-full")}>
|
<div className="flex flex-col items-center gap-8">
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Entity</label>
|
<div
|
||||||
<Select
|
className={clsx("flex flex-col items-center gap-4 w-full")}
|
||||||
defaultValue={{ value: entity?.id, label: entity?.label }}
|
>
|
||||||
options={entitiesThatCanBePaid.map((e) => ({ value: e.id, label: e.label, entity: e }))}
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
onChange={(e) => e?.value ? setEntity(e?.entity) : null}
|
Entity
|
||||||
className="!w-full max-w-[400px] self-center"
|
</label>
|
||||||
/>
|
<Select
|
||||||
</div>
|
defaultValue={{ value: entity?.id, label: entity?.label }}
|
||||||
|
options={entitiesThatCanBePaid.map((e) => ({
|
||||||
|
value: e.id,
|
||||||
|
label: e.label,
|
||||||
|
entity: e,
|
||||||
|
}))}
|
||||||
|
onChange={(e) => (e?.value ? setEntity(e?.entity) : null)}
|
||||||
|
className="!w-full max-w-[400px] self-center"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
|
||||||
<span className="max-w-lg">
|
<span className="max-w-lg">
|
||||||
To add to your use of EnCoach and that of your students and teachers, please pay your designated package
|
To add to your use of EnCoach and that of your students and
|
||||||
below:
|
teachers, please pay your designated package below:
|
||||||
</span>
|
</span>
|
||||||
<div className={clsx("flex flex-col items-start gap-6 rounded-xl bg-white p-4")}>
|
<div
|
||||||
<div className="mb-2 flex flex-col items-start">
|
className={clsx(
|
||||||
<img src="/logo_title.png" alt="EnCoach's Logo" className="w-32" />
|
"flex flex-col items-start gap-6 rounded-xl bg-white p-4"
|
||||||
<span className="text-xl font-semibold">
|
)}
|
||||||
EnCoach - {12} Months
|
>
|
||||||
</span>
|
<div className="mb-2 flex flex-col items-start">
|
||||||
</div>
|
<img
|
||||||
<div className="flex w-full flex-col items-start gap-2">
|
src="/logo_title.png"
|
||||||
<span className="text-2xl">
|
alt="EnCoach's Logo"
|
||||||
{entity.payment.price} {entity.payment.currency}
|
className="w-32"
|
||||||
</span>
|
/>
|
||||||
<PaymobPayment
|
<span className="text-xl font-semibold">
|
||||||
user={user}
|
EnCoach - {12} Months
|
||||||
setIsPaymentLoading={setIsLoading}
|
</span>
|
||||||
entity={entity}
|
</div>
|
||||||
currency={entity.payment.currency}
|
<div className="flex w-full flex-col items-start gap-2">
|
||||||
price={entity.payment.price}
|
<span className="text-2xl">
|
||||||
duration={12}
|
{entity.payment.price} {entity.payment.currency}
|
||||||
duration_unit="months"
|
</span>
|
||||||
onSuccess={() => {
|
<PaymobPayment
|
||||||
setIsLoading(false);
|
user={user}
|
||||||
setTimeout(reload, 500);
|
setIsPaymentLoading={setIsLoading}
|
||||||
}}
|
entity={entity}
|
||||||
/>
|
currency={entity.payment.currency}
|
||||||
</div>
|
price={entity.payment.price}
|
||||||
<div className="flex flex-col items-start gap-1">
|
duration={12}
|
||||||
<span>This includes:</span>
|
duration_unit="months"
|
||||||
<ul className="flex flex-col items-start text-sm">
|
onSuccess={() => {
|
||||||
<li>
|
setIsLoading(false);
|
||||||
- Allow a total of {entity.licenses} students and teachers to use EnCoach
|
setTimeout(reload, 500);
|
||||||
</li>
|
}}
|
||||||
<li>- Train their abilities for the IELTS exam</li>
|
/>
|
||||||
<li>- Gain insights into your students' weaknesses and strengths</li>
|
</div>
|
||||||
<li>- Allow them to correctly prepare for the exam</li>
|
<div className="flex flex-col items-start gap-1">
|
||||||
</ul>
|
<span>This includes:</span>
|
||||||
</div>
|
<ul className="flex flex-col items-start text-sm">
|
||||||
</div>
|
<li>
|
||||||
</div>
|
- Allow a total of {entity.licenses} students and
|
||||||
)}
|
teachers to use EnCoach
|
||||||
{!isIndividual && entitiesThatCanBePaid.length === 0 && (
|
</li>
|
||||||
<div className="flex flex-col items-center">
|
<li>- Train their abilities for the IELTS exam</li>
|
||||||
<span className="max-w-lg">
|
<li>
|
||||||
You are not the person in charge of your time credits, please contact your administrator about this situation.
|
- Gain insights into your students' weaknesses and
|
||||||
</span>
|
strengths
|
||||||
<span className="max-w-lg">
|
</li>
|
||||||
If you believe this to be a mistake, please contact the platform's administration, thank you for your
|
<li>- Allow them to correctly prepare for the exam</li>
|
||||||
patience.
|
</ul>
|
||||||
</span>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)}
|
</div>
|
||||||
{!isIndividual &&
|
)}
|
||||||
entitiesThatCanBePaid.length > 0 &&
|
{!isIndividual && entitiesThatCanBePaid.length === 0 && (
|
||||||
!entity?.payment && (
|
<div className="flex flex-col items-center">
|
||||||
<div className="flex flex-col items-center gap-8">
|
<span className="max-w-lg">
|
||||||
<div className={clsx("flex flex-col items-center gap-4 w-full")}>
|
You are not the person in charge of your time credits, please
|
||||||
<label className="font-normal text-base text-mti-gray-dim">Entity</label>
|
contact your administrator about this situation.
|
||||||
<Select
|
</span>
|
||||||
defaultValue={{ value: entity?.id || "", label: entity?.label || "" }}
|
<span className="max-w-lg">
|
||||||
options={entitiesThatCanBePaid.map((e) => ({ value: e.id, label: e.label, entity: e }))}
|
If you believe this to be a mistake, please contact the
|
||||||
onChange={(e) => e?.value ? setEntity(e?.entity) : null}
|
platform's administration, thank you for your patience.
|
||||||
className="!w-full max-w-[400px] self-center"
|
</span>
|
||||||
/>
|
</div>
|
||||||
</div>
|
)}
|
||||||
<span className="max-w-lg">
|
{!isIndividual &&
|
||||||
An admin nor your agent have yet set the price intended to your requirements in terms of the amount of users
|
entitiesThatCanBePaid.length > 0 &&
|
||||||
you desire and your expected monthly duration.
|
!entity?.payment && (
|
||||||
</span>
|
<div className="flex flex-col items-center gap-8">
|
||||||
<span className="max-w-lg">
|
<div
|
||||||
Please try again later or contact your agent or an admin, thank you for your patience.
|
className={clsx("flex flex-col items-center gap-4 w-full")}
|
||||||
</span>
|
>
|
||||||
</div>
|
<label className="font-normal text-base text-mti-gray-dim">
|
||||||
)}
|
Entity
|
||||||
</div>
|
</label>
|
||||||
</Layout>
|
<Select
|
||||||
</>
|
defaultValue={{
|
||||||
);
|
value: entity?.id || "",
|
||||||
|
label: entity?.label || "",
|
||||||
|
}}
|
||||||
|
options={entitiesThatCanBePaid.map((e) => ({
|
||||||
|
value: e.id,
|
||||||
|
label: e.label,
|
||||||
|
entity: e,
|
||||||
|
}))}
|
||||||
|
onChange={(e) => (e?.value ? setEntity(e?.entity) : null)}
|
||||||
|
className="!w-full max-w-[400px] self-center"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<span className="max-w-lg">
|
||||||
|
An admin nor your agent have yet set the price intended to
|
||||||
|
your requirements in terms of the amount of users you desire
|
||||||
|
and your expected monthly duration.
|
||||||
|
</span>
|
||||||
|
<span className="max-w-lg">
|
||||||
|
Please try again later or contact your agent or an admin,
|
||||||
|
thank you for your patience.
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
</>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,35 +1,74 @@
|
|||||||
import "@/styles/globals.css";
|
import "@/styles/globals.css";
|
||||||
import "react-toastify/dist/ReactToastify.css";
|
import "react-toastify/dist/ReactToastify.css";
|
||||||
import type {AppProps} from "next/app";
|
import type { AppProps } from "next/app";
|
||||||
|
|
||||||
import "primereact/resources/themes/lara-light-indigo/theme.css";
|
import "primereact/resources/themes/lara-light-indigo/theme.css";
|
||||||
import "primereact/resources/primereact.min.css";
|
import "primereact/resources/primereact.min.css";
|
||||||
import "primeicons/primeicons.css";
|
import "primeicons/primeicons.css";
|
||||||
import "react-datepicker/dist/react-datepicker.css";
|
import "react-datepicker/dist/react-datepicker.css";
|
||||||
import {useRouter} from "next/router";
|
import { Router, useRouter } from "next/router";
|
||||||
import {useEffect} from "react";
|
import { useEffect, useState } from "react";
|
||||||
import useExamStore from "@/stores/exam";
|
import useExamStore from "@/stores/exam";
|
||||||
import usePreferencesStore from "@/stores/preferencesStore";
|
import usePreferencesStore from "@/stores/preferencesStore";
|
||||||
|
import Layout from "../components/High/Layout";
|
||||||
|
import useEntities from "../hooks/useEntities";
|
||||||
|
export default function App({ Component, pageProps }: AppProps) {
|
||||||
|
const [loading, setLoading] = useState(false);
|
||||||
|
|
||||||
export default function App({Component, pageProps}: AppProps) {
|
const { reset } = useExamStore();
|
||||||
const {reset} = useExamStore();
|
|
||||||
const setIsSidebarMinimized = usePreferencesStore((state) => state.setSidebarMinimized);
|
|
||||||
|
|
||||||
const router = useRouter();
|
const setIsSidebarMinimized = usePreferencesStore(
|
||||||
|
(state) => state.setSidebarMinimized
|
||||||
|
);
|
||||||
|
|
||||||
useEffect(() => {
|
const router = useRouter();
|
||||||
if (router.pathname !== "/exam" && router.pathname !== "/exercises") reset();
|
|
||||||
}, [router.pathname, reset]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
const { entities } = useEntities(!pageProps?.user?.id);
|
||||||
if (localStorage.getItem("isSidebarMinimized")) {
|
|
||||||
if (localStorage.getItem("isSidebarMinimized") === "true") {
|
|
||||||
setIsSidebarMinimized(true);
|
|
||||||
} else {
|
|
||||||
setIsSidebarMinimized(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}, [setIsSidebarMinimized]);
|
|
||||||
|
|
||||||
return <Component {...pageProps} />;
|
useEffect(() => {
|
||||||
|
const start = () => {
|
||||||
|
setLoading(true);
|
||||||
|
};
|
||||||
|
const end = () => {
|
||||||
|
setLoading(false);
|
||||||
|
};
|
||||||
|
Router.events.on("routeChangeStart", start);
|
||||||
|
Router.events.on("routeChangeComplete", end);
|
||||||
|
Router.events.on("routeChangeError", end);
|
||||||
|
return () => {
|
||||||
|
Router.events.off("routeChangeStart", start);
|
||||||
|
Router.events.off("routeChangeComplete", end);
|
||||||
|
Router.events.off("routeChangeError", end);
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (router.pathname !== "/exam" && router.pathname !== "/exercises")
|
||||||
|
reset();
|
||||||
|
}, [router.pathname, reset]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (localStorage.getItem("isSidebarMinimized")) {
|
||||||
|
if (localStorage.getItem("isSidebarMinimized") === "true") {
|
||||||
|
setIsSidebarMinimized(true);
|
||||||
|
} else {
|
||||||
|
setIsSidebarMinimized(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [setIsSidebarMinimized]);
|
||||||
|
|
||||||
|
return pageProps?.user ? (
|
||||||
|
<Layout user={pageProps.user} entities={entities} refreshPage={loading}>
|
||||||
|
{loading ? (
|
||||||
|
// TODO: Change this later to a better loading screen (example: skeletons for each page)
|
||||||
|
<div className="min-h-screen flex justify-center items-start">
|
||||||
|
<span className="loading loading-infinity w-32" />
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<Component entities={entities} {...pageProps} />
|
||||||
|
)}
|
||||||
|
</Layout>
|
||||||
|
) : (
|
||||||
|
<Component {...pageProps} />
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
32
src/pages/api/approval-workflows/[id]/edit.ts
Normal file
32
src/pages/api/approval-workflows/[id]/edit.ts
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction
|
||||||
|
import { ApprovalWorkflow } from "@/interfaces/approval.workflow";
|
||||||
|
import { sessionOptions } from "@/lib/session";
|
||||||
|
import { requestUser } from "@/utils/api";
|
||||||
|
import { updateApprovalWorkflow } from "@/utils/approval.workflows.be";
|
||||||
|
import { withIronSessionApiRoute } from "iron-session/next";
|
||||||
|
import { ObjectId } from "mongodb";
|
||||||
|
import type { NextApiRequest, NextApiResponse } from "next";
|
||||||
|
|
||||||
|
export default withIronSessionApiRoute(handler, sessionOptions);
|
||||||
|
|
||||||
|
async function handler(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
if (req.method === "PUT") return await put(req, res);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function put(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
const user = await requestUser(req, res);
|
||||||
|
if (!user) return res.status(401).json({ ok: false });
|
||||||
|
|
||||||
|
if (!["admin", "developer", "teacher", "corporate", "mastercorporate"].includes(user.type)) {
|
||||||
|
return res.status(403).json({ ok: false });
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = req.query as { id?: string };
|
||||||
|
const approvalWorkflow: ApprovalWorkflow = req.body;
|
||||||
|
|
||||||
|
if (id && approvalWorkflow) {
|
||||||
|
approvalWorkflow._id = new ObjectId(id);
|
||||||
|
await updateApprovalWorkflow("active-workflows", approvalWorkflow);
|
||||||
|
return res.status(204).end();
|
||||||
|
}
|
||||||
|
}
|
||||||
74
src/pages/api/approval-workflows/[id]/index.ts
Normal file
74
src/pages/api/approval-workflows/[id]/index.ts
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction
|
||||||
|
import { ApprovalWorkflow } from "@/interfaces/approval.workflow";
|
||||||
|
import { sessionOptions } from "@/lib/session";
|
||||||
|
import { requestUser } from "@/utils/api";
|
||||||
|
import { deleteApprovalWorkflow, getApprovalWorkflow, updateApprovalWorkflow } from "@/utils/approval.workflows.be";
|
||||||
|
import { getEntityWithRoles } from "@/utils/entities.be";
|
||||||
|
import { doesEntityAllow } from "@/utils/permissions";
|
||||||
|
import { withIronSessionApiRoute } from "iron-session/next";
|
||||||
|
import { ObjectId } from "mongodb";
|
||||||
|
import type { NextApiRequest, NextApiResponse } from "next";
|
||||||
|
|
||||||
|
export default withIronSessionApiRoute(handler, sessionOptions);
|
||||||
|
|
||||||
|
async function handler(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
if (req.method === "DELETE") return await del(req, res);
|
||||||
|
if (req.method === "PUT") return await put(req, res);
|
||||||
|
if (req.method === "GET") return await get(req, res);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function del(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
const user = await requestUser(req, res);
|
||||||
|
if (!user) return res.status(401).json({ ok: false });
|
||||||
|
|
||||||
|
if (!["admin", "developer", "teacher", "corporate", "mastercorporate"].includes(user.type)) {
|
||||||
|
return res.status(403).json({ ok: false });
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = req.query as { id: string };
|
||||||
|
const workflow = await getApprovalWorkflow("active-workflows", id);
|
||||||
|
|
||||||
|
if (!workflow) return res.status(404).json({ ok: false });
|
||||||
|
|
||||||
|
const entity = await getEntityWithRoles(workflow.entityId);
|
||||||
|
if (!entity) return res.status(404).json({ ok: false });
|
||||||
|
|
||||||
|
if (!doesEntityAllow(user, entity, "delete_workflow") && !["admin", "developer"].includes(user.type)) {
|
||||||
|
return res.status(403).json({ ok: false });
|
||||||
|
}
|
||||||
|
|
||||||
|
return res.status(200).json(await deleteApprovalWorkflow("active-workflows", id));
|
||||||
|
}
|
||||||
|
|
||||||
|
async function put(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
const user = await requestUser(req, res);
|
||||||
|
if (!user) return res.status(401).json({ ok: false });
|
||||||
|
|
||||||
|
if (!["admin", "developer", "teacher", "corporate", "mastercorporate"].includes(user.type)) {
|
||||||
|
return res.status(403).json({ ok: false });
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = req.query as { id?: string };
|
||||||
|
const workflow: ApprovalWorkflow = req.body;
|
||||||
|
|
||||||
|
if (id && workflow) {
|
||||||
|
workflow._id = new ObjectId(id);
|
||||||
|
await updateApprovalWorkflow("active-workflows", workflow);
|
||||||
|
return res.status(204).end();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function get(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
const user = await requestUser(req, res);
|
||||||
|
if (!user) return res.status(401).json({ ok: false });
|
||||||
|
|
||||||
|
if (!["admin", "developer", "teacher", "corporate", "mastercorporate"].includes(user.type)) {
|
||||||
|
return res.status(403).json({ ok: false });
|
||||||
|
}
|
||||||
|
|
||||||
|
const { id } = req.query as { id?: string };
|
||||||
|
|
||||||
|
if (id) {
|
||||||
|
return res.status(200).json(await getApprovalWorkflow("active-workflows", id));
|
||||||
|
}
|
||||||
|
}
|
||||||
37
src/pages/api/approval-workflows/create.ts
Normal file
37
src/pages/api/approval-workflows/create.ts
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction
|
||||||
|
import { ApprovalWorkflow } from "@/interfaces/approval.workflow";
|
||||||
|
import { Entity } from "@/interfaces/entity";
|
||||||
|
import { sessionOptions } from "@/lib/session";
|
||||||
|
import { requestUser } from "@/utils/api";
|
||||||
|
import { replaceApprovalWorkflowsByEntities } from "@/utils/approval.workflows.be";
|
||||||
|
import { withIronSessionApiRoute } from "iron-session/next";
|
||||||
|
import type { NextApiRequest, NextApiResponse } from "next";
|
||||||
|
|
||||||
|
export default withIronSessionApiRoute(handler, sessionOptions);
|
||||||
|
|
||||||
|
interface ReplaceApprovalWorkflowsRequest {
|
||||||
|
filteredWorkflows: ApprovalWorkflow[];
|
||||||
|
userEntitiesWithLabel: Entity[];
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handler(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
if (req.method === "POST") return await post(req, res);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function post(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
const user = await requestUser(req, res);
|
||||||
|
if (!user) return res.status(401).json({ ok: false });
|
||||||
|
|
||||||
|
if (!["admin", "developer", "teacher", "corporate", "mastercorporate"].includes(user.type)) {
|
||||||
|
return res.status(403).json({ ok: false });
|
||||||
|
}
|
||||||
|
|
||||||
|
const { filteredWorkflows, userEntitiesWithLabel } = req.body as ReplaceApprovalWorkflowsRequest;
|
||||||
|
|
||||||
|
const configuredWorkflows: ApprovalWorkflow[] = filteredWorkflows;
|
||||||
|
const entitiesIds: string[] = userEntitiesWithLabel.map((e) => e.id);
|
||||||
|
|
||||||
|
await replaceApprovalWorkflowsByEntities(configuredWorkflows, entitiesIds);
|
||||||
|
|
||||||
|
return res.status(204).end();
|
||||||
|
}
|
||||||
32
src/pages/api/approval-workflows/index.ts
Normal file
32
src/pages/api/approval-workflows/index.ts
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction
|
||||||
|
import { sessionOptions } from "@/lib/session";
|
||||||
|
import { requestUser } from "@/utils/api";
|
||||||
|
import { getApprovalWorkflows } from "@/utils/approval.workflows.be";
|
||||||
|
import { withIronSessionApiRoute } from "iron-session/next";
|
||||||
|
import type { NextApiRequest, NextApiResponse } from "next";
|
||||||
|
|
||||||
|
export default withIronSessionApiRoute(handler, sessionOptions);
|
||||||
|
|
||||||
|
async function handler(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
if (req.method === "GET") return await get(req, res);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function get(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
const user = await requestUser(req, res);
|
||||||
|
if (!user) return res.status(401).json({ ok: false });
|
||||||
|
|
||||||
|
if (!["admin", "developer", "teacher", "corporate", "mastercorporate"].includes(user.type)) {
|
||||||
|
return res.status(403).json({ ok: false });
|
||||||
|
}
|
||||||
|
|
||||||
|
const entityIdsString = req.query.entityIds as string;
|
||||||
|
|
||||||
|
const entityIdsArray = entityIdsString.split(",");
|
||||||
|
|
||||||
|
if (!["admin", "developer"].includes(user.type)) {
|
||||||
|
// filtering workflows that have user as assignee in at least one of the steps
|
||||||
|
return res.status(200).json(await getApprovalWorkflows("active-workflows", entityIdsArray, undefined, user.id));
|
||||||
|
} else {
|
||||||
|
return res.status(200).json(await getApprovalWorkflows("active-workflows", entityIdsArray));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -3,16 +3,8 @@ import type { NextApiRequest, NextApiResponse } from "next";
|
|||||||
import client from "@/lib/mongodb";
|
import client from "@/lib/mongodb";
|
||||||
import { withIronSessionApiRoute } from "iron-session/next";
|
import { withIronSessionApiRoute } from "iron-session/next";
|
||||||
import { sessionOptions } from "@/lib/session";
|
import { sessionOptions } from "@/lib/session";
|
||||||
import { Code, Group, Type } from "@/interfaces/user";
|
import { Code, } from "@/interfaces/user";
|
||||||
import { PERMISSIONS } from "@/constants/userPermissions";
|
|
||||||
import { prepareMailer, prepareMailOptions } from "@/email";
|
|
||||||
import { isAdmin } from "@/utils/users";
|
|
||||||
import { requestUser } from "@/utils/api";
|
import { requestUser } from "@/utils/api";
|
||||||
import { doesEntityAllow } from "@/utils/permissions";
|
|
||||||
import { getEntity, getEntityWithRoles } from "@/utils/entities.be";
|
|
||||||
import { findBy } from "@/utils";
|
|
||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
|
||||||
|
|
||||||
const db = client.db(process.env.MONGODB_DB);
|
const db = client.db(process.env.MONGODB_DB);
|
||||||
|
|
||||||
export default withIronSessionApiRoute(handler, sessionOptions);
|
export default withIronSessionApiRoute(handler, sessionOptions);
|
||||||
@@ -30,7 +22,7 @@ async function get(req: NextApiRequest, res: NextApiResponse) {
|
|||||||
|
|
||||||
const { entities } = req.query as { entities?: string[] };
|
const { entities } = req.query as { entities?: string[] };
|
||||||
if (entities)
|
if (entities)
|
||||||
return res.status(200).json(await db.collection("codes").find<Code>({ entity: { $in: entities } }).toArray());
|
return res.status(200).json(await db.collection("codes").find<Code>({ entity: { $in: Array.isArray(entities) ? entities : [entities] } }).toArray());
|
||||||
|
|
||||||
return res.status(200).json(await db.collection("codes").find<Code>({}).toArray());
|
return res.status(200).json(await db.collection("codes").find<Code>({}).toArray());
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ async function get(req: NextApiRequest, res: NextApiResponse) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const { entity } = req.query as { entity?: string };
|
const { entity } = req.query as { entity?: string };
|
||||||
|
|
||||||
const snapshot = await db.collection("codes").find(entity ? { entity } : {}).toArray();
|
const snapshot = await db.collection("codes").find(entity ? { entity } : {}).toArray();
|
||||||
|
|
||||||
res.status(200).json(snapshot);
|
res.status(200).json(snapshot);
|
||||||
|
|||||||
@@ -1,89 +1,179 @@
|
|||||||
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction
|
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction
|
||||||
import type { NextApiRequest, NextApiResponse } from "next";
|
|
||||||
import client from "@/lib/mongodb";
|
|
||||||
import { withIronSessionApiRoute } from "iron-session/next";
|
|
||||||
import { sessionOptions } from "@/lib/session";
|
|
||||||
import { Exam, ExamBase, InstructorGender, Variant } from "@/interfaces/exam";
|
|
||||||
import { getExams } from "@/utils/exams.be";
|
|
||||||
import { Module } from "@/interfaces";
|
import { Module } from "@/interfaces";
|
||||||
import { getUserCorporate } from "@/utils/groups.be";
|
import { Exam, ExamBase, InstructorGender, LevelExam, ListeningExam, ReadingExam, SpeakingExam, Variant } from "@/interfaces/exam";
|
||||||
import { requestUser } from "@/utils/api";
|
import { createApprovalWorkflowOnExamCreation } from "@/lib/createWorkflowsOnExamCreation";
|
||||||
import { isAdmin } from "@/utils/users";
|
import client from "@/lib/mongodb";
|
||||||
|
import { sessionOptions } from "@/lib/session";
|
||||||
import { mapBy } from "@/utils";
|
import { mapBy } from "@/utils";
|
||||||
|
import { requestUser } from "@/utils/api";
|
||||||
|
import { getApprovalWorkflowsByExamId, updateApprovalWorkflows } from "@/utils/approval.workflows.be";
|
||||||
|
import { generateExamDifferences } from "@/utils/exam.differences";
|
||||||
|
import { getExams } from "@/utils/exams.be";
|
||||||
|
import { isAdmin } from "@/utils/users";
|
||||||
|
import { uuidv4 } from "@firebase/util";
|
||||||
|
import { access } from "fs";
|
||||||
|
import { withIronSessionApiRoute } from "iron-session/next";
|
||||||
|
import type { NextApiRequest, NextApiResponse } from "next";
|
||||||
|
|
||||||
const db = client.db(process.env.MONGODB_DB);
|
const db = client.db(process.env.MONGODB_DB);
|
||||||
|
|
||||||
export default withIronSessionApiRoute(handler, sessionOptions);
|
export default withIronSessionApiRoute(handler, sessionOptions);
|
||||||
|
|
||||||
async function handler(req: NextApiRequest, res: NextApiResponse) {
|
// Temporary: Adding UUID here but later move to backend.
|
||||||
if (req.method === "GET") return await GET(req, res);
|
function addUUIDs(exam: ReadingExam | ListeningExam | LevelExam): ExamBase {
|
||||||
if (req.method === "POST") return await POST(req, res);
|
const arraysToUpdate = ["solutions", "words", "questions", "sentences", "options"];
|
||||||
|
|
||||||
res.status(404).json({ ok: false });
|
exam.parts = exam.parts.map((part) => {
|
||||||
|
const updatedExercises = part.exercises.map((exercise: any) => {
|
||||||
|
arraysToUpdate.forEach((arrayName) => {
|
||||||
|
if (exercise[arrayName] && Array.isArray(exercise[arrayName])) {
|
||||||
|
exercise[arrayName] = exercise[arrayName].map((item: any) => (item.uuid ? item : { ...item, uuid: uuidv4() }));
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return exercise;
|
||||||
|
});
|
||||||
|
return { ...part, exercises: updatedExercises };
|
||||||
|
});
|
||||||
|
return exam;
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handler(req: NextApiRequest, res: NextApiResponse) {
|
||||||
|
if (req.method === "GET") return await GET(req, res);
|
||||||
|
if (req.method === "POST") return await POST(req, res);
|
||||||
|
|
||||||
|
res.status(404).json({ ok: false });
|
||||||
}
|
}
|
||||||
|
|
||||||
async function GET(req: NextApiRequest, res: NextApiResponse) {
|
async function GET(req: NextApiRequest, res: NextApiResponse) {
|
||||||
if (!req.session.user) {
|
if (!req.session.user) {
|
||||||
res.status(401).json({ ok: false });
|
res.status(401).json({ ok: false });
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const { module, avoidRepeated, variant, instructorGender } = req.query as {
|
const { module, avoidRepeated, variant, instructorGender } = req.query as {
|
||||||
module: Module;
|
module: Module;
|
||||||
avoidRepeated: string;
|
avoidRepeated: string;
|
||||||
variant?: Variant;
|
variant?: Variant;
|
||||||
instructorGender?: InstructorGender;
|
instructorGender?: InstructorGender;
|
||||||
};
|
};
|
||||||
|
|
||||||
const exams: Exam[] = await getExams(db, module, avoidRepeated, req.session.user.id, variant, instructorGender);
|
const exams: Exam[] = await getExams(db, module, avoidRepeated, req.session.user.id, variant, instructorGender);
|
||||||
res.status(200).json(exams);
|
res.status(200).json(exams);
|
||||||
}
|
}
|
||||||
|
|
||||||
async function POST(req: NextApiRequest, res: NextApiResponse) {
|
async function POST(req: NextApiRequest, res: NextApiResponse) {
|
||||||
const user = await requestUser(req, res)
|
const user = await requestUser(req, res);
|
||||||
if (!user) return res.status(401).json({ ok: false });
|
if (!user) return res.status(401).json({ ok: false });
|
||||||
|
|
||||||
const { module } = req.query as { module: string };
|
const { module } = req.query as { module: string };
|
||||||
|
|
||||||
const session = client.startSession();
|
const session = client.startSession();
|
||||||
const entities = isAdmin(user) ? [] : mapBy(user.entities, 'id')
|
const entities = isAdmin(user) ? [] : mapBy(user.entities, "id");
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const exam = {
|
let exam = {
|
||||||
...req.body,
|
access: "public", // default access is public
|
||||||
module: module,
|
...req.body,
|
||||||
entities,
|
module: module,
|
||||||
createdBy: user.id,
|
entities,
|
||||||
createdAt: new Date().toISOString(),
|
createdBy: user.id,
|
||||||
};
|
createdAt: new Date().toISOString(),
|
||||||
|
};
|
||||||
|
|
||||||
await session.withTransaction(async () => {
|
// Temporary: Adding UUID here but later move to backend.
|
||||||
const docSnap = await db.collection(module).findOne<ExamBase>({ id: req.body.id }, { session });
|
exam = addUUIDs(exam);
|
||||||
|
|
||||||
// Check whether the id of the exam matches another exam with different
|
let responseStatus: number;
|
||||||
// owners, throw exception if there is, else allow editing
|
let responseMessage: string;
|
||||||
const ownersSet = new Set(docSnap?.owners || []);
|
|
||||||
|
|
||||||
if (docSnap !== null && docSnap?.owners?.length === exam.owners.lenght && exam.owners.every((e: string) => ownersSet.has(e))) {
|
await session.withTransaction(async () => {
|
||||||
throw new Error("Name already exists");
|
const docSnap = await db.collection(module).findOne<ExamBase>({ id: req.body.id }, { session });
|
||||||
}
|
|
||||||
|
|
||||||
await db.collection(module).updateOne(
|
// Check whether the id of the exam matches another exam with different
|
||||||
{ id: req.body.id },
|
// owners, throw exception if there is, else allow editing
|
||||||
{ $set: { id: req.body.id, ...exam } },
|
const existingExamOwners = docSnap?.owners ?? [];
|
||||||
{
|
const newExamOwners = exam.owners ?? [];
|
||||||
upsert: true,
|
|
||||||
session
|
|
||||||
}
|
|
||||||
);
|
|
||||||
});
|
|
||||||
|
|
||||||
res.status(200).json(exam);
|
const ownersSet = new Set(existingExamOwners);
|
||||||
|
|
||||||
} catch (error) {
|
if (docSnap !== null && (existingExamOwners.length !== newExamOwners.length || !newExamOwners.every((e: string) => ownersSet.has(e)))) {
|
||||||
console.error("Transaction failed: ", error);
|
throw new Error("Name already exists");
|
||||||
res.status(500).json({ ok: false, error: (error as any).message });
|
}
|
||||||
} finally {
|
|
||||||
session.endSession();
|
if (exam.requiresApproval === true) {
|
||||||
}
|
exam.access = "confidential";
|
||||||
|
}
|
||||||
|
|
||||||
|
await db.collection(module).updateOne(
|
||||||
|
{ id: req.body.id },
|
||||||
|
{ $set: { id: req.body.id, ...exam } },
|
||||||
|
{
|
||||||
|
upsert: true,
|
||||||
|
session,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
// if it doesn't enter the next if condition it means the exam was updated and not created, so we can send this response.
|
||||||
|
responseStatus = 200;
|
||||||
|
responseMessage = `Successfully updated exam with ID: "${exam.id}"`;
|
||||||
|
|
||||||
|
// create workflow only if exam is being created for the first time
|
||||||
|
if (docSnap === null) {
|
||||||
|
try {
|
||||||
|
if (exam.requiresApproval === false) {
|
||||||
|
responseStatus = 200;
|
||||||
|
responseMessage = `Successfully created exam "${exam.id}" and skipped Approval Workflow due to user request.`;
|
||||||
|
} else if (isAdmin(user)) {
|
||||||
|
responseStatus = 200;
|
||||||
|
responseMessage = `Successfully created exam "${exam.id}" and skipped Approval Workflow due to admin rights.`;
|
||||||
|
} else {
|
||||||
|
const { successCount, totalCount } = await createApprovalWorkflowOnExamCreation(exam.createdBy, exam.entities, exam.id, module);
|
||||||
|
|
||||||
|
if (successCount === totalCount) {
|
||||||
|
responseStatus = 200;
|
||||||
|
responseMessage = `Successfully created exam "${exam.id}" and started its Approval Workflow.`;
|
||||||
|
} else if (successCount > 0) {
|
||||||
|
responseStatus = 207;
|
||||||
|
responseMessage = `Successfully created exam with ID: "${exam.id}" but was not able to start/find an Approval Workflow for all the author's entities.`;
|
||||||
|
} else {
|
||||||
|
responseStatus = 207;
|
||||||
|
responseMessage = `Successfully created exam with ID: "${exam.id}" but skipping approval process because no approval workflow was found configured for the exam author.`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Workflow creation error:", error);
|
||||||
|
responseStatus = 207;
|
||||||
|
responseMessage = `Successfully created exam with ID: "${exam.id}" but something went wrong while creating the Approval Workflow(s).`;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// if exam was updated, log the updates
|
||||||
|
const approvalWorkflows = await getApprovalWorkflowsByExamId(exam.id);
|
||||||
|
|
||||||
|
if (approvalWorkflows) {
|
||||||
|
const differences = generateExamDifferences(docSnap as Exam, exam as Exam);
|
||||||
|
if (differences) {
|
||||||
|
approvalWorkflows.forEach((workflow) => {
|
||||||
|
const currentStepIndex = workflow.steps.findIndex((step) => !step.completed || step.rejected);
|
||||||
|
|
||||||
|
if (workflow.steps[currentStepIndex].examChanges === undefined) {
|
||||||
|
workflow.steps[currentStepIndex].examChanges = [...differences];
|
||||||
|
} else {
|
||||||
|
workflow.steps[currentStepIndex].examChanges!.push(...differences);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
await updateApprovalWorkflows("active-workflows", approvalWorkflows);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
res.status(responseStatus).json({
|
||||||
|
message: responseMessage,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Transaction failed: ", error);
|
||||||
|
res.status(500).json({ ok: false, error: (error as any).message });
|
||||||
|
} finally {
|
||||||
|
session.endSession();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,11 +1,13 @@
|
|||||||
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction
|
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction
|
||||||
import type {NextApiRequest, NextApiResponse} from "next";
|
import type { NextApiRequest, NextApiResponse } from "next";
|
||||||
import client from "@/lib/mongodb";
|
import client from "@/lib/mongodb";
|
||||||
import {withIronSessionApiRoute} from "iron-session/next";
|
import { withIronSessionApiRoute } from "iron-session/next";
|
||||||
import {sessionOptions} from "@/lib/session";
|
import { sessionOptions } from "@/lib/session";
|
||||||
import {flatten} from "lodash";
|
import { flatten } from "lodash";
|
||||||
import {Exam} from "@/interfaces/exam";
|
import { AccessType, Exam } from "@/interfaces/exam";
|
||||||
import {MODULE_ARRAY} from "@/utils/moduleUtils";
|
import { MODULE_ARRAY } from "@/utils/moduleUtils";
|
||||||
|
import { requestUser } from "../../../utils/api";
|
||||||
|
import { mapBy } from "../../../utils";
|
||||||
|
|
||||||
const db = client.db(process.env.MONGODB_DB);
|
const db = client.db(process.env.MONGODB_DB);
|
||||||
|
|
||||||
@@ -14,17 +16,37 @@ export default withIronSessionApiRoute(handler, sessionOptions);
|
|||||||
async function handler(req: NextApiRequest, res: NextApiResponse) {
|
async function handler(req: NextApiRequest, res: NextApiResponse) {
|
||||||
if (req.method === "GET") return await GET(req, res);
|
if (req.method === "GET") return await GET(req, res);
|
||||||
|
|
||||||
res.status(404).json({ok: false});
|
res.status(404).json({ ok: false });
|
||||||
}
|
}
|
||||||
|
|
||||||
async function GET(req: NextApiRequest, res: NextApiResponse) {
|
async function GET(req: NextApiRequest, res: NextApiResponse) {
|
||||||
if (!req.session.user) {
|
if (!req.session.user) {
|
||||||
res.status(401).json({ok: false});
|
res.status(401).json({ ok: false });
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
const user = await requestUser(req, res)
|
||||||
|
if (!user)
|
||||||
|
return res.status(401).json({ ok: false, reason: "You must be logged in!" })
|
||||||
|
const isAdmin = ["admin", "developer"].includes(user.type)
|
||||||
|
const { entities = [] } = req.query as { access?: AccessType, entities?: string[] | string };
|
||||||
|
let entitiesToFetch = Array.isArray(entities) ? entities : entities ? [entities] : []
|
||||||
|
|
||||||
|
if (!isAdmin) {
|
||||||
|
const userEntitiesIDs = mapBy(user.entities || [], 'id')
|
||||||
|
entitiesToFetch = entities ? entitiesToFetch.filter((entity): entity is string => entity ? userEntitiesIDs.includes(entity) : false) : userEntitiesIDs
|
||||||
|
if ((entitiesToFetch.length ?? 0) === 0) {
|
||||||
|
res.status(200).json([])
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const moduleExamsPromises = MODULE_ARRAY.map(async (module) => {
|
const moduleExamsPromises = MODULE_ARRAY.map(async (module) => {
|
||||||
const snapshot = await db.collection(module).find<Exam>({ isDiagnostic: false }).toArray();
|
const snapshot = await db.collection(module).find<Exam>({
|
||||||
|
isDiagnostic: false, ...(isAdmin && (entitiesToFetch.length ?? 0) === 0 ? {
|
||||||
|
} : {
|
||||||
|
entity: { $in: entitiesToFetch }
|
||||||
|
})
|
||||||
|
}).toArray();
|
||||||
|
|
||||||
return snapshot.map((doc) => ({
|
return snapshot.map((doc) => ({
|
||||||
...doc,
|
...doc,
|
||||||
|
|||||||
@@ -1,24 +1,20 @@
|
|||||||
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction
|
// Next.js API route support: https://nextjs.org/docs/api-routes/introduction
|
||||||
import type {NextApiRequest, NextApiResponse} from "next";
|
import type { NextApiRequest, NextApiResponse } from "next";
|
||||||
import client from "@/lib/mongodb";
|
import { withIronSessionApiRoute } from "iron-session/next";
|
||||||
import {withIronSessionApiRoute} from "iron-session/next";
|
import { sessionOptions } from "@/lib/session";
|
||||||
import {sessionOptions} from "@/lib/session";
|
import { getDetailedStatsByUser } from "../../../../utils/stats.be";
|
||||||
|
|
||||||
const db = client.db(process.env.MONGODB_DB);
|
|
||||||
|
|
||||||
export default withIronSessionApiRoute(handler, sessionOptions);
|
export default withIronSessionApiRoute(handler, sessionOptions);
|
||||||
|
|
||||||
async function handler(req: NextApiRequest, res: NextApiResponse) {
|
async function handler(req: NextApiRequest, res: NextApiResponse) {
|
||||||
if (!req.session.user) {
|
if (!req.session.user) {
|
||||||
res.status(401).json({ok: false});
|
res.status(401).json({ ok: false });
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
const {user} = req.query;
|
const { user, query } = req.query as { user: string, query?: string };
|
||||||
const snapshot = await db.collection("stats").aggregate([
|
|
||||||
{ $match: { user: user } },
|
|
||||||
{ $sort: { "date": 1 } }
|
|
||||||
]).toArray();
|
|
||||||
|
|
||||||
|
const snapshot = await getDetailedStatsByUser(user, query);
|
||||||
res.status(200).json(snapshot);
|
res.status(200).json(snapshot);
|
||||||
}
|
}
|
||||||
@@ -33,7 +33,7 @@ async function get(req: NextApiRequest, res: NextApiResponse) {
|
|||||||
res.status(401).json({ ok: false });
|
res.status(401).json({ ok: false });
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
const docs = await db.collection("tickets").find<Ticket>({ assignedTo: req.session.user.id }).toArray();
|
const docs = await db.collection("tickets").find<Ticket>({ assignedTo: req.session.user.id, status: { $ne: "completed" } }).toArray();
|
||||||
|
|
||||||
res.status(200).json(docs);
|
res.status(200).json(docs);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -25,8 +25,10 @@ async function post(req: NextApiRequest, res: NextApiResponse) {
|
|||||||
Authorization: `Bearer ${process.env.BACKEND_JWT}`,
|
Authorization: `Bearer ${process.env.BACKEND_JWT}`,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
console.log('response', response.data);
|
||||||
res.status(response.status).json(response.data);
|
res.status(response.status).json(response.data);
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
console.error('Error fetching data:', error);
|
||||||
res.status(500).json({ message: 'An unexpected error occurred' });
|
res.status(500).json({ message: 'An unexpected error occurred' });
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
193
src/pages/approval-workflows/[id]/edit.tsx
Normal file
193
src/pages/approval-workflows/[id]/edit.tsx
Normal file
@@ -0,0 +1,193 @@
|
|||||||
|
import RequestedBy from "@/components/ApprovalWorkflows/RequestedBy";
|
||||||
|
import StartedOn from "@/components/ApprovalWorkflows/StartedOn";
|
||||||
|
import Status from "@/components/ApprovalWorkflows/Status";
|
||||||
|
import WorkflowForm from "@/components/ApprovalWorkflows/WorkflowForm";
|
||||||
|
import Layout from "@/components/High/Layout";
|
||||||
|
import { ApprovalWorkflow, EditableApprovalWorkflow, EditableWorkflowStep, getUserTypeLabelShort } from "@/interfaces/approval.workflow";
|
||||||
|
import { CorporateUser, DeveloperUser, MasterCorporateUser, TeacherUser, User } from "@/interfaces/user";
|
||||||
|
import { sessionOptions } from "@/lib/session";
|
||||||
|
import { findBy, redirect, serialize } from "@/utils";
|
||||||
|
import { requestUser } from "@/utils/api";
|
||||||
|
import { getApprovalWorkflow } from "@/utils/approval.workflows.be";
|
||||||
|
import { getEntityWithRoles } from "@/utils/entities.be";
|
||||||
|
import { shouldRedirectHome } from "@/utils/navigation.disabled";
|
||||||
|
import { doesEntityAllow } from "@/utils/permissions";
|
||||||
|
import { getEntityUsers } from "@/utils/users.be";
|
||||||
|
import axios from "axios";
|
||||||
|
import { LayoutGroup, motion } from "framer-motion";
|
||||||
|
import { withIronSessionSsr } from "iron-session/next";
|
||||||
|
import Head from "next/head";
|
||||||
|
import Link from "next/link";
|
||||||
|
import { useEffect, useState } from "react";
|
||||||
|
import { BsChevronLeft } from "react-icons/bs";
|
||||||
|
import { toast, ToastContainer } from "react-toastify";
|
||||||
|
|
||||||
|
export const getServerSideProps = withIronSessionSsr(async ({ req, res, params }) => {
|
||||||
|
const user = await requestUser(req, res);
|
||||||
|
if (!user) return redirect("/login")
|
||||||
|
|
||||||
|
if (shouldRedirectHome(user) || !["admin", "developer", "teacher", "corporate", "mastercorporate"].includes(user.type))
|
||||||
|
return redirect("/")
|
||||||
|
|
||||||
|
const { id } = params as { id: string };
|
||||||
|
|
||||||
|
const workflow: ApprovalWorkflow | null = await getApprovalWorkflow("active-workflows", id);
|
||||||
|
if (!workflow) return redirect("/approval-workflows");
|
||||||
|
|
||||||
|
const entityWithRole = await getEntityWithRoles(workflow.entityId);
|
||||||
|
if (!entityWithRole) return redirect("/approval-workflows");
|
||||||
|
|
||||||
|
if (!doesEntityAllow(user, entityWithRole, "edit_workflow")) return redirect("/approval-workflows");
|
||||||
|
|
||||||
|
return {
|
||||||
|
props: serialize({
|
||||||
|
user,
|
||||||
|
workflow,
|
||||||
|
workflowEntityApprovers: await getEntityUsers(workflow.entityId, undefined, { type: { $in: ["teacher", "corporate", "mastercorporate", "developer"] } }) as (TeacherUser | CorporateUser | MasterCorporateUser | DeveloperUser)[],
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
}, sessionOptions);
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
user: User,
|
||||||
|
workflow: ApprovalWorkflow,
|
||||||
|
workflowEntityApprovers: (TeacherUser | CorporateUser | MasterCorporateUser | DeveloperUser)[],
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function Home({ user, workflow, workflowEntityApprovers }: Props) {
|
||||||
|
const [updatedWorkflow, setUpdatedWorkflow] = useState<EditableApprovalWorkflow | null>(null);
|
||||||
|
const [isLoading, setIsLoading] = useState<boolean>(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const editableSteps: EditableWorkflowStep[] = workflow.steps.map(step => ({
|
||||||
|
key: step.stepNumber + 999, // just making sure they are unique because new steps that users add will have key=3 key=4 etc
|
||||||
|
stepType: step.stepType,
|
||||||
|
stepNumber: step.stepNumber,
|
||||||
|
completed: step.completed,
|
||||||
|
completedBy: step.completedBy || undefined,
|
||||||
|
completedDate: step.completedDate || undefined,
|
||||||
|
assignees: step.assignees,
|
||||||
|
firstStep: step.firstStep || false,
|
||||||
|
finalStep: step.finalStep || false,
|
||||||
|
onDelete: undefined,
|
||||||
|
}));
|
||||||
|
|
||||||
|
const editableWorkflow: EditableApprovalWorkflow = {
|
||||||
|
...workflow,
|
||||||
|
id: workflow._id?.toString() ?? "",
|
||||||
|
requester: user.id, // should it change to the editor?
|
||||||
|
steps: editableSteps,
|
||||||
|
};
|
||||||
|
|
||||||
|
setUpdatedWorkflow(editableWorkflow);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
const handleSubmit = (e: React.FormEvent<HTMLFormElement>) => {
|
||||||
|
e.preventDefault();
|
||||||
|
setIsLoading(true);
|
||||||
|
|
||||||
|
if (!updatedWorkflow) {
|
||||||
|
setIsLoading(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const step of updatedWorkflow.steps) {
|
||||||
|
if (step.assignees.every(x => !x)) {
|
||||||
|
toast.warning("There is at least one empty step in the workflow.");
|
||||||
|
setIsLoading(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const filteredWorkflow: ApprovalWorkflow = {
|
||||||
|
...updatedWorkflow,
|
||||||
|
steps: updatedWorkflow.steps.map(step => ({
|
||||||
|
...step,
|
||||||
|
assignees: step.assignees.filter((assignee): assignee is string => assignee !== null && assignee !== undefined)
|
||||||
|
}))
|
||||||
|
};
|
||||||
|
|
||||||
|
axios
|
||||||
|
.put(`/api/approval-workflows/${updatedWorkflow.id}/edit`, filteredWorkflow)
|
||||||
|
.then(() => {
|
||||||
|
toast.success("Approval Workflow edited successfully.");
|
||||||
|
setIsLoading(false);
|
||||||
|
})
|
||||||
|
.catch((reason) => {
|
||||||
|
if (reason.response.status === 401) {
|
||||||
|
toast.error("Not logged in!");
|
||||||
|
} else if (reason.response.status === 403) {
|
||||||
|
toast.error("You do not have permission to edit Approval Workflows!");
|
||||||
|
} else {
|
||||||
|
toast.error("Something went wrong, please try again later.");
|
||||||
|
}
|
||||||
|
setIsLoading(false);
|
||||||
|
console.log("Submitted Values:", filteredWorkflow);
|
||||||
|
return;
|
||||||
|
})
|
||||||
|
};
|
||||||
|
|
||||||
|
const onWorkflowChange = (updatedWorkflow: EditableApprovalWorkflow) => {
|
||||||
|
setUpdatedWorkflow(updatedWorkflow);
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Head>
|
||||||
|
<title> Edit Workflow | EnCoach</title>
|
||||||
|
<meta
|
||||||
|
name="description"
|
||||||
|
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
||||||
|
/>
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
|
<link rel="icon" href="/favicon.ico" />
|
||||||
|
</Head>
|
||||||
|
<ToastContainer />
|
||||||
|
<section className="flex items-center gap-2">
|
||||||
|
<Link
|
||||||
|
href="/approval-workflows"
|
||||||
|
className="text-mti-purple hover:text-mti-purple-dark transition ease-in-out duration-300 text-xl">
|
||||||
|
<BsChevronLeft />
|
||||||
|
</Link>
|
||||||
|
<h1 className="text-2xl font-semibold">{workflow.name}</h1>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section className="flex flex-col gap-6">
|
||||||
|
<div className="flex flex-row gap-6">
|
||||||
|
<RequestedBy
|
||||||
|
prefix={getUserTypeLabelShort(user.type)}
|
||||||
|
name={user.name}
|
||||||
|
profileImage={user.profilePicture}
|
||||||
|
/>
|
||||||
|
<StartedOn
|
||||||
|
date={workflow.startDate}
|
||||||
|
/>
|
||||||
|
<Status
|
||||||
|
status={workflow.status}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<form onSubmit={handleSubmit}>
|
||||||
|
<LayoutGroup key={workflow.name}>
|
||||||
|
<motion.div
|
||||||
|
key="form"
|
||||||
|
initial={{ opacity: 0, y: -30 }}
|
||||||
|
animate={{ opacity: 1, y: 0 }}
|
||||||
|
exit={{ opacity: 0, x: 60 }}
|
||||||
|
transition={{ duration: 0.20 }}
|
||||||
|
>
|
||||||
|
{updatedWorkflow &&
|
||||||
|
<WorkflowForm
|
||||||
|
workflow={updatedWorkflow}
|
||||||
|
onWorkflowChange={onWorkflowChange}
|
||||||
|
entityApprovers={workflowEntityApprovers}
|
||||||
|
isLoading={isLoading}
|
||||||
|
/>
|
||||||
|
}
|
||||||
|
</motion.div>
|
||||||
|
</LayoutGroup>
|
||||||
|
</form>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
615
src/pages/approval-workflows/[id]/index.tsx
Normal file
615
src/pages/approval-workflows/[id]/index.tsx
Normal file
@@ -0,0 +1,615 @@
|
|||||||
|
import RequestedBy from "@/components/ApprovalWorkflows/RequestedBy";
|
||||||
|
import StartedOn from "@/components/ApprovalWorkflows/StartedOn";
|
||||||
|
import Status from "@/components/ApprovalWorkflows/Status";
|
||||||
|
import Tip from "@/components/ApprovalWorkflows/Tip";
|
||||||
|
import UserWithProfilePic from "@/components/ApprovalWorkflows/UserWithProfilePic";
|
||||||
|
import WorkflowStepComponent from "@/components/ApprovalWorkflows/WorkflowStepComponent";
|
||||||
|
import Layout from "@/components/High/Layout";
|
||||||
|
import Button from "@/components/Low/Button";
|
||||||
|
import useApprovalWorkflow from "@/hooks/useApprovalWorkflow";
|
||||||
|
import { ApprovalWorkflow, getUserTypeLabelShort, WorkflowStep } from "@/interfaces/approval.workflow";
|
||||||
|
import { User } from "@/interfaces/user";
|
||||||
|
import { sessionOptions } from "@/lib/session";
|
||||||
|
import useExamStore from "@/stores/exam";
|
||||||
|
import { redirect, serialize } from "@/utils";
|
||||||
|
import { requestUser } from "@/utils/api";
|
||||||
|
import { getApprovalWorkflow } from "@/utils/approval.workflows.be";
|
||||||
|
import { getEntityWithRoles } from "@/utils/entities.be";
|
||||||
|
import { getExamById } from "@/utils/exams";
|
||||||
|
import { shouldRedirectHome } from "@/utils/navigation.disabled";
|
||||||
|
import { doesEntityAllow } from "@/utils/permissions";
|
||||||
|
import { getSpecificUsers, getUser } from "@/utils/users.be";
|
||||||
|
import axios from "axios";
|
||||||
|
import { AnimatePresence, LayoutGroup, motion } from "framer-motion";
|
||||||
|
import { withIronSessionSsr } from "iron-session/next";
|
||||||
|
import Head from "next/head";
|
||||||
|
import Link from "next/link";
|
||||||
|
import { useRouter } from "next/router";
|
||||||
|
import { useState } from "react";
|
||||||
|
import { BsChevronLeft } from "react-icons/bs";
|
||||||
|
import { FaSpinner, FaWpforms } from "react-icons/fa6";
|
||||||
|
import { FiSave } from "react-icons/fi";
|
||||||
|
import { IoMdCheckmarkCircleOutline } from "react-icons/io";
|
||||||
|
import { IoDocumentTextOutline } from "react-icons/io5";
|
||||||
|
import { MdKeyboardArrowDown, MdKeyboardArrowUp, MdOutlineDoubleArrow } from "react-icons/md";
|
||||||
|
import { RiThumbUpLine } from "react-icons/ri";
|
||||||
|
import { RxCrossCircled } from "react-icons/rx";
|
||||||
|
import { TiEdit } from "react-icons/ti";
|
||||||
|
import { toast, ToastContainer } from "react-toastify";
|
||||||
|
|
||||||
|
export const getServerSideProps = withIronSessionSsr(async ({ req, res, params }) => {
|
||||||
|
const user = await requestUser(req, res);
|
||||||
|
if (!user) return redirect("/login")
|
||||||
|
|
||||||
|
if (shouldRedirectHome(user) || !["admin", "developer", "teacher", "corporate", "mastercorporate"].includes(user.type))
|
||||||
|
return redirect("/")
|
||||||
|
|
||||||
|
const { id } = params as { id: string };
|
||||||
|
|
||||||
|
const workflow: ApprovalWorkflow | null = await getApprovalWorkflow("active-workflows", id);
|
||||||
|
|
||||||
|
if (!workflow) return redirect("/approval-workflows")
|
||||||
|
|
||||||
|
const entityWithRole = await getEntityWithRoles(workflow.entityId);
|
||||||
|
if (!entityWithRole) return redirect("/approval-workflows");
|
||||||
|
|
||||||
|
if (!doesEntityAllow(user, entityWithRole, "view_workflows")) return redirect("/approval-workflows");
|
||||||
|
|
||||||
|
const allAssigneeIds: string[] = [
|
||||||
|
...new Set(
|
||||||
|
workflow.steps
|
||||||
|
.map((step) => {
|
||||||
|
const assignees = step.assignees;
|
||||||
|
if (step.completedBy) {
|
||||||
|
assignees.push(step.completedBy);
|
||||||
|
}
|
||||||
|
return assignees;
|
||||||
|
})
|
||||||
|
.flat()
|
||||||
|
)
|
||||||
|
];
|
||||||
|
|
||||||
|
return {
|
||||||
|
props: serialize({
|
||||||
|
user,
|
||||||
|
initialWorkflow: workflow,
|
||||||
|
id,
|
||||||
|
workflowAssignees: await getSpecificUsers(allAssigneeIds),
|
||||||
|
workflowRequester: await getUser(workflow.requester),
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
}, sessionOptions);
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
user: User,
|
||||||
|
initialWorkflow: ApprovalWorkflow,
|
||||||
|
id: string,
|
||||||
|
workflowAssignees: User[],
|
||||||
|
workflowRequester: User,
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function Home({ user, initialWorkflow, id, workflowAssignees, workflowRequester }: Props) {
|
||||||
|
|
||||||
|
const { workflow, reload, isLoading } = useApprovalWorkflow(id);
|
||||||
|
|
||||||
|
const currentWorkflow = workflow || initialWorkflow;
|
||||||
|
|
||||||
|
let currentStepIndex = currentWorkflow.steps.findIndex(step => !step.completed || step.rejected);
|
||||||
|
if (currentStepIndex === -1)
|
||||||
|
currentStepIndex = currentWorkflow.steps.length - 1;
|
||||||
|
|
||||||
|
const [selectedStepIndex, setSelectedStepIndex] = useState<number>(currentStepIndex);
|
||||||
|
const [selectedStep, setSelectedStep] = useState<WorkflowStep>(currentWorkflow.steps[selectedStepIndex]);
|
||||||
|
const [isPanelOpen, setIsPanelOpen] = useState(true);
|
||||||
|
const [isAccordionOpen, setIsAccordionOpen] = useState(false);
|
||||||
|
const [comments, setComments] = useState<string>(selectedStep.comments || "");
|
||||||
|
const [viewExamIsLoading, setViewExamIsLoading] = useState<boolean>(false);
|
||||||
|
const [editExamIsLoading, setEditExamIsLoading] = useState<boolean>(false);
|
||||||
|
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
const handleStepClick = (index: number, stepInfo: WorkflowStep) => {
|
||||||
|
setSelectedStep(stepInfo);
|
||||||
|
setSelectedStepIndex(index);
|
||||||
|
setComments(stepInfo.comments || "");
|
||||||
|
setIsPanelOpen(true);
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleSaveComments = () => {
|
||||||
|
const updatedWorkflow: ApprovalWorkflow = {
|
||||||
|
...currentWorkflow,
|
||||||
|
steps: currentWorkflow.steps.map((step, index) =>
|
||||||
|
index === selectedStepIndex ?
|
||||||
|
{
|
||||||
|
...step,
|
||||||
|
comments: comments,
|
||||||
|
}
|
||||||
|
: step
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
axios
|
||||||
|
.put(`/api/approval-workflows/${id}`, updatedWorkflow)
|
||||||
|
.then(() => {
|
||||||
|
toast.success("Comments saved successfully.");
|
||||||
|
reload();
|
||||||
|
})
|
||||||
|
.catch((reason) => {
|
||||||
|
if (reason.response.status === 401) {
|
||||||
|
toast.error("Not logged in!");
|
||||||
|
} else if (reason.response.status === 403) {
|
||||||
|
toast.error("You do not have permission to approve this step!");
|
||||||
|
} else {
|
||||||
|
toast.error("Something went wrong, please try again later.");
|
||||||
|
}
|
||||||
|
console.log("Submitted Values:", updatedWorkflow);
|
||||||
|
return;
|
||||||
|
})
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleApproveStep = () => {
|
||||||
|
const isLastStep = (selectedStepIndex + 1 === currentWorkflow.steps.length);
|
||||||
|
if (isLastStep) {
|
||||||
|
if (!confirm(`Are you sure you want to approve the last step and complete the approval process?`)) return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const updatedWorkflow: ApprovalWorkflow = {
|
||||||
|
...currentWorkflow,
|
||||||
|
status: selectedStepIndex === currentWorkflow.steps.length - 1 ? "approved" : "pending",
|
||||||
|
steps: currentWorkflow.steps.map((step, index) =>
|
||||||
|
index === selectedStepIndex ?
|
||||||
|
{
|
||||||
|
...step,
|
||||||
|
completed: true,
|
||||||
|
completedBy: user.id,
|
||||||
|
completedDate: Date.now(),
|
||||||
|
}
|
||||||
|
: step
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
axios
|
||||||
|
.put(`/api/approval-workflows/${id}`, updatedWorkflow)
|
||||||
|
.then(() => {
|
||||||
|
toast.success("Step approved successfully.");
|
||||||
|
reload();
|
||||||
|
})
|
||||||
|
.catch((reason) => {
|
||||||
|
if (reason.response.status === 401) {
|
||||||
|
toast.error("Not logged in!");
|
||||||
|
} else if (reason.response.status === 403) {
|
||||||
|
toast.error("You do not have permission to approve this step!");
|
||||||
|
} else {
|
||||||
|
toast.error("Something went wrong, please try again later.");
|
||||||
|
}
|
||||||
|
console.log("Submitted Values:", updatedWorkflow);
|
||||||
|
return;
|
||||||
|
})
|
||||||
|
|
||||||
|
if (isLastStep) {
|
||||||
|
setIsPanelOpen(false);
|
||||||
|
const examModule = currentWorkflow.modules[0];
|
||||||
|
const examId = currentWorkflow.examId;
|
||||||
|
|
||||||
|
axios
|
||||||
|
.patch(`/api/exam/${examModule}/${examId}`, { approved: true })
|
||||||
|
.then(() => toast.success(`The exam was successfuly approved and this workflow has been completed.`))
|
||||||
|
.catch((reason) => {
|
||||||
|
if (reason.response.status === 404) {
|
||||||
|
toast.error("Exam not found!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (reason.response.status === 403) {
|
||||||
|
toast.error("You do not have permission to update this exam!");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
toast.error("Something went wrong, please try again later.");
|
||||||
|
})
|
||||||
|
.finally(reload);
|
||||||
|
} else {
|
||||||
|
handleStepClick(selectedStepIndex + 1, currentWorkflow.steps[selectedStepIndex + 1]);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleRejectStep = () => {
|
||||||
|
if (!confirm(`Are you sure you want to reject this step? Doing so will terminate this approval workflow.`)) return;
|
||||||
|
|
||||||
|
const updatedWorkflow: ApprovalWorkflow = {
|
||||||
|
...currentWorkflow,
|
||||||
|
status: "rejected",
|
||||||
|
steps: currentWorkflow.steps.map((step, index) =>
|
||||||
|
index === selectedStepIndex ?
|
||||||
|
{
|
||||||
|
...step,
|
||||||
|
completed: true,
|
||||||
|
completedBy: user.id,
|
||||||
|
completedDate: Date.now(),
|
||||||
|
rejected: true,
|
||||||
|
}
|
||||||
|
: step
|
||||||
|
)
|
||||||
|
};
|
||||||
|
|
||||||
|
axios
|
||||||
|
.put(`/api/approval-workflows/${id}`, updatedWorkflow)
|
||||||
|
.then(() => {
|
||||||
|
toast.success("Step rejected successfully.");
|
||||||
|
reload();
|
||||||
|
})
|
||||||
|
.catch((reason) => {
|
||||||
|
if (reason.response.status === 401) {
|
||||||
|
toast.error("Not logged in!");
|
||||||
|
} else if (reason.response.status === 403) {
|
||||||
|
toast.error("You do not have permission to approve this step!");
|
||||||
|
} else {
|
||||||
|
toast.error("Something went wrong, please try again later.");
|
||||||
|
}
|
||||||
|
console.log("Submitted Values:", updatedWorkflow);
|
||||||
|
return;
|
||||||
|
})
|
||||||
|
};
|
||||||
|
|
||||||
|
const dispatch = useExamStore((store) => store.dispatch);
|
||||||
|
const handleViewExam = async () => {
|
||||||
|
setViewExamIsLoading(true);
|
||||||
|
const examModule = currentWorkflow.modules[0];
|
||||||
|
const examId = currentWorkflow.examId;
|
||||||
|
|
||||||
|
if (examModule && examId) {
|
||||||
|
const exam = await getExamById(examModule, examId.trim());
|
||||||
|
if (!exam) {
|
||||||
|
toast.error("Something went wrong while fetching exam!");
|
||||||
|
setViewExamIsLoading(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
dispatch({
|
||||||
|
type: "INIT_EXAM",
|
||||||
|
payload: { exams: [exam], modules: [examModule] },
|
||||||
|
});
|
||||||
|
router.push("/exam");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleEditExam = () => {
|
||||||
|
setEditExamIsLoading(true);
|
||||||
|
const examModule = currentWorkflow.modules[0];
|
||||||
|
const examId = currentWorkflow.examId;
|
||||||
|
|
||||||
|
router.push(`/generation?id=${examId}&module=${examModule}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Head>
|
||||||
|
<title> Workflow | EnCoach</title>
|
||||||
|
<meta
|
||||||
|
name="description"
|
||||||
|
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
||||||
|
/>
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
|
<link rel="icon" href="/favicon.ico" />
|
||||||
|
</Head>
|
||||||
|
<ToastContainer />
|
||||||
|
|
||||||
|
<section className="flex items-center gap-2">
|
||||||
|
<Link
|
||||||
|
href="/approval-workflows"
|
||||||
|
className="text-mti-purple hover:text-mti-purple-dark transition ease-in-out duration-300 text-xl">
|
||||||
|
<BsChevronLeft />
|
||||||
|
</Link>
|
||||||
|
<h1 className="text-2xl font-semibold">{currentWorkflow.name}</h1>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section className="flex flex-col gap-6">
|
||||||
|
<div className="flex flex-row gap-6">
|
||||||
|
<RequestedBy
|
||||||
|
prefix={getUserTypeLabelShort(workflowRequester.type)}
|
||||||
|
name={workflowRequester.name}
|
||||||
|
profileImage={workflowRequester.profilePicture}
|
||||||
|
/>
|
||||||
|
<StartedOn
|
||||||
|
date={currentWorkflow.startDate}
|
||||||
|
/>
|
||||||
|
<Status
|
||||||
|
status={currentWorkflow.status}
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-row gap-3">
|
||||||
|
<Button
|
||||||
|
color="purple"
|
||||||
|
variant="solid"
|
||||||
|
onClick={handleViewExam}
|
||||||
|
disabled={viewExamIsLoading}
|
||||||
|
padding="px-6 py-2"
|
||||||
|
className="w-[240px] text-lg flex items-center justify-center gap-2 text-left"
|
||||||
|
>
|
||||||
|
{viewExamIsLoading ? (
|
||||||
|
<>
|
||||||
|
<FaSpinner className="animate-spin size-5" />
|
||||||
|
Loading...
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<IoDocumentTextOutline />
|
||||||
|
Load Exam
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
color="purple"
|
||||||
|
variant="solid"
|
||||||
|
onClick={handleEditExam}
|
||||||
|
padding="px-6 py-2"
|
||||||
|
disabled={(!currentWorkflow.steps[currentStepIndex].assignees.includes(user.id) && user.type !== "admin" && user.type !== "developer") || editExamIsLoading}
|
||||||
|
className="w-[240px] text-lg flex items-center justify-center gap-2 text-left"
|
||||||
|
>
|
||||||
|
{editExamIsLoading ? (
|
||||||
|
<>
|
||||||
|
<FaSpinner className="animate-spin size-5" />
|
||||||
|
Loading...
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<TiEdit size={20} />
|
||||||
|
Edit Exam
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
{currentWorkflow.steps.find((step) => !step.completed) === undefined &&
|
||||||
|
<Tip text="All steps in this instance have been completed." />
|
||||||
|
}
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section className="flex flex-col gap-0">
|
||||||
|
{currentWorkflow.steps.map((step, index) => (
|
||||||
|
<WorkflowStepComponent
|
||||||
|
workflowAssignees={workflowAssignees}
|
||||||
|
key={index}
|
||||||
|
completed={step.completed}
|
||||||
|
completedBy={step.completedBy}
|
||||||
|
rejected={step.rejected}
|
||||||
|
stepNumber={step.stepNumber}
|
||||||
|
stepType={step.stepType}
|
||||||
|
assignees={step.assignees}
|
||||||
|
finalStep={index === currentWorkflow.steps.length - 1}
|
||||||
|
currentStep={index === currentStepIndex}
|
||||||
|
selected={index === selectedStepIndex}
|
||||||
|
onClick={() => handleStepClick(index, step)}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</section>
|
||||||
|
|
||||||
|
{/* Side panel */}
|
||||||
|
<AnimatePresence mode="wait">
|
||||||
|
<LayoutGroup key="sidePanel">
|
||||||
|
<section className={`absolute inset-y-0 right-0 h-full overflow-y-auto bg-mti-purple-ultralight bg-opacity-50 shadow-xl shadow-mti-purple transition-all duration-300 overflow-hidden ${isPanelOpen ? 'w-[500px]' : 'w-0'}`}>
|
||||||
|
{isPanelOpen && selectedStep && (
|
||||||
|
<motion.div
|
||||||
|
className="p-6"
|
||||||
|
key={selectedStep.stepNumber}
|
||||||
|
initial={{ opacity: 0, x: 30 }}
|
||||||
|
animate={{ opacity: 1, x: 0 }}
|
||||||
|
exit={{ opacity: 0, x: 30 }}
|
||||||
|
transition={{ duration: 0.2 }}
|
||||||
|
>
|
||||||
|
<hr className="my-4 h-[4px] bg-mti-purple-ultralight rounded-full w-full" />
|
||||||
|
<div className="flex flex-row gap-2">
|
||||||
|
<p className="text-2xl font-medium text-left align-middle">Step {selectedStepIndex + 1}</p>
|
||||||
|
<div className="ml-auto flex flex-row">
|
||||||
|
<button
|
||||||
|
className="min-w-fit max-h-fit text-lg font-medium flex items-center gap-2 text-left"
|
||||||
|
onClick={() => setIsPanelOpen(false)}
|
||||||
|
>
|
||||||
|
Collapse
|
||||||
|
<MdOutlineDoubleArrow size={20} />
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr className="my-4 h-[4px] bg-mti-purple-ultralight rounded-full w-full" />
|
||||||
|
|
||||||
|
<div>
|
||||||
|
<div className="my-8 flex flex-row gap-4 items-center text-lg font-medium">
|
||||||
|
{selectedStep.stepType === "approval-by" ? (
|
||||||
|
<>
|
||||||
|
<RiThumbUpLine size={30} />
|
||||||
|
Approval Step
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<FaWpforms size={30} />
|
||||||
|
Form Intake Step
|
||||||
|
</>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{selectedStep.completed ? (
|
||||||
|
<div className={"text-base font-medium text-gray-500 flex flex-col gap-6"}>
|
||||||
|
{selectedStep.rejected ? "Rejected" : "Approved"} on {new Date(selectedStep.completedDate!).toLocaleString("en-CA", {
|
||||||
|
year: "numeric",
|
||||||
|
month: "2-digit",
|
||||||
|
day: "2-digit",
|
||||||
|
hour: "2-digit",
|
||||||
|
minute: "2-digit",
|
||||||
|
second: "2-digit",
|
||||||
|
hour12: false,
|
||||||
|
}).replace(", ", " at ")}
|
||||||
|
<div className="flex flex-row gap-1 text-sm">
|
||||||
|
<p className="text-base">{selectedStep.rejected ? "Rejected" : "Approved"} by:</p>
|
||||||
|
{(() => {
|
||||||
|
const assignee = workflowAssignees.find(
|
||||||
|
(assignee) => assignee.id === selectedStep.completedBy
|
||||||
|
);
|
||||||
|
return assignee ? (
|
||||||
|
<UserWithProfilePic
|
||||||
|
textSize="text-base"
|
||||||
|
prefix={getUserTypeLabelShort(assignee.type)}
|
||||||
|
name={assignee.name}
|
||||||
|
profileImage={assignee.profilePicture}
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
"Unknown"
|
||||||
|
);
|
||||||
|
})()}
|
||||||
|
</div>
|
||||||
|
<p className="text-sm">No additional actions are required.</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
) : (
|
||||||
|
<div className={"text-base font-medium text-gray-500 mb-6"}>
|
||||||
|
One assignee is required to sign off to complete this step:
|
||||||
|
<div className="flex flex-col gap-2 mt-3">
|
||||||
|
{workflowAssignees.filter(user => selectedStep.assignees.includes(user.id)).map(user => (
|
||||||
|
<span key={user.id}>
|
||||||
|
<UserWithProfilePic
|
||||||
|
textSize="text-sm"
|
||||||
|
prefix={`- ${getUserTypeLabelShort(user.type)}`}
|
||||||
|
name={user.name}
|
||||||
|
profileImage={user.profilePicture}
|
||||||
|
/>
|
||||||
|
</span>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{selectedStepIndex === currentStepIndex && !selectedStep.completed && !selectedStep.rejected &&
|
||||||
|
<div className="flex flex-row gap-2 ">
|
||||||
|
<Button
|
||||||
|
type="submit"
|
||||||
|
color="purple"
|
||||||
|
variant="solid"
|
||||||
|
disabled={(!selectedStep.assignees.includes(user.id) && user.type !== "admin" && user.type !== "developer") || isLoading}
|
||||||
|
onClick={handleApproveStep}
|
||||||
|
padding="px-6 py-2"
|
||||||
|
className="mb-3 w-full text-lg flex items-center justify-center gap-2 text-left"
|
||||||
|
>
|
||||||
|
{isLoading ? (
|
||||||
|
<>
|
||||||
|
<FaSpinner className="animate-spin size-5" />
|
||||||
|
Loading...
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<IoMdCheckmarkCircleOutline size={20} />
|
||||||
|
Approve Step
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
type="submit"
|
||||||
|
color="red"
|
||||||
|
variant="solid"
|
||||||
|
disabled={(!selectedStep.assignees.includes(user.id) && user.type !== "admin" && user.type !== "developer") || isLoading}
|
||||||
|
onClick={handleRejectStep}
|
||||||
|
padding="px-6 py-2"
|
||||||
|
className="mb-3 w-1/2 text-lg flex items-center justify-center gap-2 text-left"
|
||||||
|
>
|
||||||
|
{isLoading ? (
|
||||||
|
<>
|
||||||
|
<FaSpinner className="animate-spin size-5" />
|
||||||
|
Loading...
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<RxCrossCircled size={20} />
|
||||||
|
Reject
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
|
||||||
|
<hr className="my-4 h-[4px] bg-mti-purple-ultralight rounded-full w-full" />
|
||||||
|
|
||||||
|
{/* Accordion for Exam Changes */}
|
||||||
|
<div className="mb-4">
|
||||||
|
<div
|
||||||
|
className="flex items-center justify-between cursor-pointer p-2 rounded-lg"
|
||||||
|
onClick={() => setIsAccordionOpen((prev) => !prev)}
|
||||||
|
>
|
||||||
|
<h2 className="font-medium text-gray-500">
|
||||||
|
Changes ({currentWorkflow.steps[selectedStepIndex].examChanges?.length || "0"})
|
||||||
|
</h2>
|
||||||
|
{isAccordionOpen ? (
|
||||||
|
<MdKeyboardArrowUp size={24} />
|
||||||
|
) : (
|
||||||
|
<MdKeyboardArrowDown size={24} />
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<AnimatePresence>
|
||||||
|
{isAccordionOpen && (
|
||||||
|
<motion.div
|
||||||
|
initial={{ height: 0, opacity: 0 }}
|
||||||
|
animate={{ height: "auto", opacity: 1 }}
|
||||||
|
exit={{ height: 0, opacity: 0 }}
|
||||||
|
transition={{ duration: 0.3 }}
|
||||||
|
className="overflow-hidden mt-2"
|
||||||
|
>
|
||||||
|
<div className="p-3 border border-gray-300 rounded-xl bg-white bg-opacity-80 overflow-y-auto max-h-[300px]">
|
||||||
|
{currentWorkflow.steps[selectedStepIndex].examChanges?.length ? (
|
||||||
|
currentWorkflow.steps[selectedStepIndex].examChanges!.map((change, index) => (
|
||||||
|
<>
|
||||||
|
<p key={index} className="whitespace-pre-wrap text-sm text-gray-500 mb-2">
|
||||||
|
<span className="text-mti-purple-light text-lg">{change.charAt(0)}</span>
|
||||||
|
{change.slice(1)}
|
||||||
|
</p>
|
||||||
|
<hr className="my-3 h-[3px] bg-mti-purple-light rounded-full w-full" />
|
||||||
|
</>
|
||||||
|
))
|
||||||
|
) : (
|
||||||
|
<p className="text-normal text-opacity-70 text-gray-500">No changes made so far.</p>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</motion.div>
|
||||||
|
)}
|
||||||
|
</AnimatePresence>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<hr className="my-4 h-[4px] bg-mti-purple-ultralight rounded-full w-full" />
|
||||||
|
|
||||||
|
<textarea
|
||||||
|
value={comments}
|
||||||
|
onChange={(e) => setComments(e.target.value)}
|
||||||
|
placeholder="Input comments here"
|
||||||
|
className="w-full h-[200px] p-2 border-2 rounded-xl shadow-lg focus:border-mti-purple focus:outline-none mt-3 resize-none"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Button
|
||||||
|
type="submit"
|
||||||
|
color="purple"
|
||||||
|
variant="solid"
|
||||||
|
onClick={handleSaveComments}
|
||||||
|
disabled={isLoading}
|
||||||
|
padding="px-6 py-2"
|
||||||
|
className="mt-6 mb-3 w-full text-lg flex items-center justify-center gap-2 text-left"
|
||||||
|
>
|
||||||
|
{isLoading ? (
|
||||||
|
<>
|
||||||
|
<FaSpinner className="animate-spin size-5" />
|
||||||
|
Loading...
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<>
|
||||||
|
<FiSave size={20} />
|
||||||
|
Save Comments
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
<hr className="my-4 h-[4px] bg-mti-purple-ultralight rounded-full w-full" />
|
||||||
|
|
||||||
|
</div>
|
||||||
|
</motion.div>
|
||||||
|
)}
|
||||||
|
</section>
|
||||||
|
</LayoutGroup>
|
||||||
|
</AnimatePresence>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
415
src/pages/approval-workflows/create.tsx
Normal file
415
src/pages/approval-workflows/create.tsx
Normal file
@@ -0,0 +1,415 @@
|
|||||||
|
import Tip from "@/components/ApprovalWorkflows/Tip";
|
||||||
|
import WorkflowForm from "@/components/ApprovalWorkflows/WorkflowForm";
|
||||||
|
import Button from "@/components/Low/Button";
|
||||||
|
import Input from "@/components/Low/Input";
|
||||||
|
import Select from "@/components/Low/Select";
|
||||||
|
import { ApprovalWorkflow, EditableApprovalWorkflow } from "@/interfaces/approval.workflow";
|
||||||
|
import { Entity } from "@/interfaces/entity";
|
||||||
|
import { CorporateUser, DeveloperUser, MasterCorporateUser, TeacherUser, User } from "@/interfaces/user";
|
||||||
|
import { sessionOptions } from "@/lib/session";
|
||||||
|
import { mapBy, redirect, serialize } from "@/utils";
|
||||||
|
import { requestUser } from "@/utils/api";
|
||||||
|
import { getApprovalWorkflowsByEntities } from "@/utils/approval.workflows.be";
|
||||||
|
import { getEntitiesWithRoles } from "@/utils/entities.be";
|
||||||
|
import { shouldRedirectHome } from "@/utils/navigation.disabled";
|
||||||
|
import { findAllowedEntities } from "@/utils/permissions";
|
||||||
|
import { isAdmin } from "@/utils/users";
|
||||||
|
import { getEntitiesUsers } from "@/utils/users.be";
|
||||||
|
import axios from "axios";
|
||||||
|
import { AnimatePresence, LayoutGroup, motion } from "framer-motion";
|
||||||
|
import { withIronSessionSsr } from "iron-session/next";
|
||||||
|
import Head from "next/head";
|
||||||
|
import Link from "next/link";
|
||||||
|
import { useRouter } from "next/router";
|
||||||
|
import { useEffect, useState } from "react";
|
||||||
|
import { BsChevronLeft, BsTrash } from "react-icons/bs";
|
||||||
|
import { FaRegClone } from "react-icons/fa6";
|
||||||
|
import { MdFormatListBulletedAdd } from "react-icons/md";
|
||||||
|
import { toast, ToastContainer } from "react-toastify";
|
||||||
|
import { v4 as uuidv4 } from 'uuid';
|
||||||
|
|
||||||
|
export const getServerSideProps = withIronSessionSsr(async ({ req, res }) => {
|
||||||
|
const user = await requestUser(req, res)
|
||||||
|
if (!user) return redirect("/login")
|
||||||
|
|
||||||
|
if (shouldRedirectHome(user) || !["admin", "developer", "teacher", "corporate", "mastercorporate"].includes(user.type))
|
||||||
|
return redirect("/")
|
||||||
|
|
||||||
|
const entityIDS = mapBy(user.entities, "id");
|
||||||
|
const entities = await getEntitiesWithRoles(isAdmin(user) ? undefined : entityIDS);
|
||||||
|
const userEntitiesWithLabel = findAllowedEntities(user, entities, "configure_workflows");
|
||||||
|
|
||||||
|
const allConfiguredWorkflows = await getApprovalWorkflowsByEntities("configured-workflows", userEntitiesWithLabel.map(entity => entity.id));
|
||||||
|
|
||||||
|
const approverTypes = ["teacher", "corporate", "mastercorporate"];
|
||||||
|
|
||||||
|
if (user.type === "developer") {
|
||||||
|
approverTypes.push("developer");
|
||||||
|
}
|
||||||
|
|
||||||
|
const userEntitiesApprovers = await getEntitiesUsers(userEntitiesWithLabel.map(entity => entity.id), { type: { $in: approverTypes } }) as (TeacherUser | CorporateUser | MasterCorporateUser | DeveloperUser)[];
|
||||||
|
|
||||||
|
return {
|
||||||
|
props: serialize({
|
||||||
|
user,
|
||||||
|
allConfiguredWorkflows,
|
||||||
|
userEntitiesWithLabel,
|
||||||
|
userEntitiesApprovers,
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
}, sessionOptions);
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
user: User,
|
||||||
|
allConfiguredWorkflows: EditableApprovalWorkflow[],
|
||||||
|
userEntitiesWithLabel: Entity[],
|
||||||
|
userEntitiesApprovers: (TeacherUser | CorporateUser | MasterCorporateUser | DeveloperUser)[],
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function Home({ user, allConfiguredWorkflows, userEntitiesWithLabel, userEntitiesApprovers }: Props) {
|
||||||
|
const [workflows, setWorkflows] = useState<EditableApprovalWorkflow[]>(allConfiguredWorkflows);
|
||||||
|
const [selectedWorkflowId, setSelectedWorkflowId] = useState<string | undefined>(undefined);
|
||||||
|
const [entityId, setEntityId] = useState<string | null | undefined>(null);
|
||||||
|
const [entityApprovers, setEntityApprovers] = useState<(TeacherUser | CorporateUser | MasterCorporateUser | DeveloperUser)[]>([]);
|
||||||
|
const [entityAvailableFormIntakers, setEntityAvailableFormIntakers] = useState<(TeacherUser | CorporateUser | MasterCorporateUser | DeveloperUser)[]>([]);
|
||||||
|
const [isAdding, setIsAdding] = useState<boolean>(false); // used to temporary timeout new workflow button. With animations, clicking too fast might cause state inconsistencies between renders.
|
||||||
|
const [isLoading, setIsLoading] = useState<boolean>(false);
|
||||||
|
const [isRedirecting, setIsRedirecting] = useState<boolean>(false);
|
||||||
|
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (entityId) {
|
||||||
|
setEntityApprovers(
|
||||||
|
userEntitiesApprovers.filter(approver =>
|
||||||
|
approver.entities.some(entity => entity.id === entityId)
|
||||||
|
)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}, [entityId, userEntitiesApprovers]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (entityId) {
|
||||||
|
// Get all workflows for the selected entity
|
||||||
|
const workflowsForEntity = workflows.filter(wf => wf.entityId === entityId);
|
||||||
|
|
||||||
|
// For all workflows except the current one, collect the first step assignees
|
||||||
|
const assignedFormIntakers = workflowsForEntity.reduce<string[]>((acc, wf) => {
|
||||||
|
if (wf.id === selectedWorkflowId) return acc; // skip current workflow so its selection isn’t removed
|
||||||
|
|
||||||
|
const formIntakeStep = wf.steps.find(step => step.stepType === "form-intake");
|
||||||
|
if (formIntakeStep) {
|
||||||
|
// Only consider non-null assignees
|
||||||
|
const validAssignees = formIntakeStep.assignees.filter(
|
||||||
|
(assignee): assignee is string => !!assignee
|
||||||
|
);
|
||||||
|
return acc.concat(validAssignees);
|
||||||
|
}
|
||||||
|
return acc;
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
// Now filter out any user from entityApprovers whose id is in the assignedFormIntakers list.
|
||||||
|
// (The selected one in the current workflow is allowed even if it is in the list.)
|
||||||
|
const availableFormIntakers = entityApprovers.filter(assignee =>
|
||||||
|
!assignedFormIntakers.includes(assignee.id)
|
||||||
|
);
|
||||||
|
|
||||||
|
setEntityAvailableFormIntakers(availableFormIntakers);
|
||||||
|
}
|
||||||
|
}, [entityId, entityApprovers, workflows, selectedWorkflowId]);
|
||||||
|
|
||||||
|
|
||||||
|
const currentWorkflow = workflows.find(wf => wf.id === selectedWorkflowId);
|
||||||
|
|
||||||
|
const ENTITY_OPTIONS = userEntitiesWithLabel.map(entity => ({
|
||||||
|
label: entity.label,
|
||||||
|
value: entity.id,
|
||||||
|
filter: (x: EditableApprovalWorkflow) => x.entityId === entity.id,
|
||||||
|
}));
|
||||||
|
|
||||||
|
const handleSubmit = (e: React.FormEvent<HTMLFormElement>) => {
|
||||||
|
e.preventDefault();
|
||||||
|
setIsLoading(true);
|
||||||
|
|
||||||
|
if (workflows.length === 0) {
|
||||||
|
setIsLoading(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const workflow of workflows) {
|
||||||
|
for (const step of workflow.steps) {
|
||||||
|
if (step.assignees.every(x => !x)) {
|
||||||
|
toast.warning("There are empty steps in at least one of the configured workflows.");
|
||||||
|
setIsLoading(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const filteredWorkflows: ApprovalWorkflow[] = workflows.map(workflow => ({
|
||||||
|
...workflow,
|
||||||
|
steps: workflow.steps.map(step => ({
|
||||||
|
...step,
|
||||||
|
assignees: step.assignees.filter((assignee): assignee is string => assignee !== null && assignee !== undefined)
|
||||||
|
}))
|
||||||
|
}));
|
||||||
|
|
||||||
|
const requestData = { filteredWorkflows, userEntitiesWithLabel };
|
||||||
|
|
||||||
|
axios
|
||||||
|
.post(`/api/approval-workflows/create`, requestData)
|
||||||
|
.then(() => {
|
||||||
|
toast.success("Approval Workflows created successfully.");
|
||||||
|
setIsRedirecting(true);
|
||||||
|
router.push("/approval-workflows");
|
||||||
|
})
|
||||||
|
.catch((reason) => {
|
||||||
|
if (reason.response.status === 401) {
|
||||||
|
toast.error("Not logged in!");
|
||||||
|
}
|
||||||
|
else if (reason.response.status === 403) {
|
||||||
|
toast.error("You do not have permission to create Approval Workflows!");
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
toast.error("Something went wrong, please try again later.");
|
||||||
|
}
|
||||||
|
setIsLoading(false);
|
||||||
|
console.log("Submitted Values:", filteredWorkflows);
|
||||||
|
return;
|
||||||
|
})
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleAddNewWorkflow = () => {
|
||||||
|
if (isAdding) return;
|
||||||
|
setIsAdding(true);
|
||||||
|
|
||||||
|
const newId = uuidv4(); // this id is only used in UI. it is ommited on submission to DB and lets mongo handle unique id.
|
||||||
|
const newWorkflow: EditableApprovalWorkflow = {
|
||||||
|
id: newId,
|
||||||
|
name: "",
|
||||||
|
entityId: "",
|
||||||
|
modules: [],
|
||||||
|
requester: user.id,
|
||||||
|
startDate: Date.now(),
|
||||||
|
status: "pending",
|
||||||
|
steps: [
|
||||||
|
{ key: 9998, stepType: "form-intake", stepNumber: 1, completed: false, firstStep: true, finalStep: false, assignees: [null] },
|
||||||
|
{ key: 9999, stepType: "approval-by", stepNumber: 2, completed: false, firstStep: false, finalStep: true, assignees: [null] },
|
||||||
|
],
|
||||||
|
};
|
||||||
|
setWorkflows((prev) => [...prev, newWorkflow]);
|
||||||
|
handleSelectWorkflow(newId);
|
||||||
|
|
||||||
|
setTimeout(() => setIsAdding(false), 300);
|
||||||
|
};
|
||||||
|
|
||||||
|
const onWorkflowChange = (updatedWorkflow: EditableApprovalWorkflow) => {
|
||||||
|
setWorkflows(prev =>
|
||||||
|
prev.map(wf => (wf.id === updatedWorkflow.id ? updatedWorkflow : wf))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleSelectWorkflow = (id: string | undefined) => {
|
||||||
|
setSelectedWorkflowId(id);
|
||||||
|
const selectedWorkflow = workflows.find(wf => wf.id === id);
|
||||||
|
if (selectedWorkflow) {
|
||||||
|
setEntityId(selectedWorkflow.entityId || null);
|
||||||
|
} else {
|
||||||
|
setEntityId(null);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleCloneWorkflow = (id: string) => {
|
||||||
|
const workflowToClone = workflows.find(wf => wf.id === id);
|
||||||
|
if (!workflowToClone) return;
|
||||||
|
|
||||||
|
const newId = uuidv4();
|
||||||
|
|
||||||
|
const clonedWorkflow: EditableApprovalWorkflow = {
|
||||||
|
...workflowToClone,
|
||||||
|
id: newId,
|
||||||
|
steps: workflowToClone.steps.map(step => ({
|
||||||
|
...step,
|
||||||
|
assignees: step.firstStep ? [null] : [...step.assignees], // we can't have more than one form intaker per teacher per entity
|
||||||
|
})),
|
||||||
|
};
|
||||||
|
|
||||||
|
setWorkflows(prev => {
|
||||||
|
const updatedWorkflows = [...prev, clonedWorkflow];
|
||||||
|
setSelectedWorkflowId(newId);
|
||||||
|
setEntityId(clonedWorkflow.entityId || null);
|
||||||
|
return updatedWorkflows;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleDeleteWorkflow = (id: string) => {
|
||||||
|
if (!confirm(`Are you sure you want to delete this Approval Workflow?`)) return;
|
||||||
|
|
||||||
|
const updatedWorkflows = workflows.filter(wf => wf.id !== id);
|
||||||
|
|
||||||
|
setWorkflows(updatedWorkflows);
|
||||||
|
|
||||||
|
if (selectedWorkflowId === id) {
|
||||||
|
handleSelectWorkflow(updatedWorkflows.find(wf => wf.id)?.id);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const handleEntityChange = (wf: EditableApprovalWorkflow, entityId: string) => {
|
||||||
|
const updatedWorkflow = {
|
||||||
|
...wf,
|
||||||
|
entityId: entityId,
|
||||||
|
steps: wf.steps.map(step => ({
|
||||||
|
...step,
|
||||||
|
assignees: step.assignees.map(() => null)
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
onWorkflowChange(updatedWorkflow);
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Head>
|
||||||
|
<title> Configure Workflows | EnCoach</title>
|
||||||
|
<meta
|
||||||
|
name="description"
|
||||||
|
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
||||||
|
/>
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
|
<link rel="icon" href="/favicon.ico" />
|
||||||
|
</Head>
|
||||||
|
<ToastContainer />
|
||||||
|
<section className="flex flex-col">
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<Link
|
||||||
|
href="/approval-workflows"
|
||||||
|
className="text-mti-purple hover:text-mti-purple-dark transition ease-in-out duration-300 text-xl">
|
||||||
|
<BsChevronLeft />
|
||||||
|
</Link>
|
||||||
|
<h1 className="text-2xl font-semibold">{"Configure Approval Workflows"}</h1>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<Tip text="Setting a teacher as a Form Intaker means the configured workflow will be instantiated when said teacher publishes an exam. Only one Form Intake per teacher per entity is allowed." />
|
||||||
|
|
||||||
|
<section className="flex flex-row gap-6">
|
||||||
|
<Button
|
||||||
|
color="purple"
|
||||||
|
variant="solid"
|
||||||
|
onClick={handleAddNewWorkflow}
|
||||||
|
className="min-w-fit max-h-fit text-lg font-medium flex items-center gap-2 text-left"
|
||||||
|
>
|
||||||
|
<MdFormatListBulletedAdd className="size-6" />
|
||||||
|
Add New Workflow
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
{workflows.length !== 0 && <div className="bg-gray-300 w-[1px]"></div>}
|
||||||
|
|
||||||
|
<div className="flex flex-wrap gap-2">
|
||||||
|
{workflows.map((workflow) => (
|
||||||
|
<Button
|
||||||
|
key={workflow.id}
|
||||||
|
color="purple"
|
||||||
|
variant={
|
||||||
|
selectedWorkflowId === workflow.id
|
||||||
|
? "solid"
|
||||||
|
: "outline"
|
||||||
|
}
|
||||||
|
onClick={() => handleSelectWorkflow(workflow.id)}
|
||||||
|
className="min-w-fit text-lg font-medium flex items-center gap-2 text-left"
|
||||||
|
>
|
||||||
|
{workflow.name.trim() === "" ? "Workflow" : workflow.name}
|
||||||
|
</Button>
|
||||||
|
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<form onSubmit={handleSubmit}>
|
||||||
|
{currentWorkflow && (
|
||||||
|
<>
|
||||||
|
<div className="mb-8 flex flex-row gap-6 items-end">
|
||||||
|
<Input
|
||||||
|
label="Name:"
|
||||||
|
type="text"
|
||||||
|
name={currentWorkflow.name}
|
||||||
|
placeholder="Enter workflow name"
|
||||||
|
value={currentWorkflow.name}
|
||||||
|
onChange={(updatedName) => {
|
||||||
|
const updatedWorkflow = {
|
||||||
|
...currentWorkflow,
|
||||||
|
name: updatedName,
|
||||||
|
};
|
||||||
|
onWorkflowChange(updatedWorkflow);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
<Select
|
||||||
|
label="Entity:"
|
||||||
|
options={ENTITY_OPTIONS}
|
||||||
|
value={
|
||||||
|
currentWorkflow.entityId === ""
|
||||||
|
? null
|
||||||
|
: ENTITY_OPTIONS.find(option => option.value === currentWorkflow.entityId)
|
||||||
|
}
|
||||||
|
onChange={(selectedEntity) => {
|
||||||
|
if (currentWorkflow.entityId) {
|
||||||
|
if (!confirm("Clearing or changing entity will clear all the assignees for all steps in this workflow. Are you sure you want to proceed?")) return;
|
||||||
|
}
|
||||||
|
if (selectedEntity?.value) {
|
||||||
|
setEntityId(selectedEntity.value);
|
||||||
|
handleEntityChange(currentWorkflow, selectedEntity.value);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
isClearable
|
||||||
|
placeholder="Enter workflow entity"
|
||||||
|
/>
|
||||||
|
<Button
|
||||||
|
color="gray"
|
||||||
|
variant="solid"
|
||||||
|
onClick={() => handleCloneWorkflow(currentWorkflow.id)}
|
||||||
|
type="button"
|
||||||
|
className="min-w-fit h-[72px] text-lg font-medium flex items-center gap-2 text-left"
|
||||||
|
>
|
||||||
|
Clone Workflow
|
||||||
|
<FaRegClone className="size-6" />
|
||||||
|
</Button>
|
||||||
|
<Button
|
||||||
|
color="red"
|
||||||
|
variant="solid"
|
||||||
|
onClick={() => handleDeleteWorkflow(currentWorkflow.id)}
|
||||||
|
type="button"
|
||||||
|
className="min-w-fit h-[72px] text-lg font-medium flex items-center gap-2 text-left"
|
||||||
|
>
|
||||||
|
Delete Workflow
|
||||||
|
<BsTrash className="size-6" />
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<AnimatePresence mode="wait">
|
||||||
|
<LayoutGroup key={currentWorkflow.id}>
|
||||||
|
<motion.div
|
||||||
|
key="form"
|
||||||
|
initial={{ opacity: 0, y: -30 }}
|
||||||
|
animate={{ opacity: 1, y: 0 }}
|
||||||
|
exit={{ opacity: 0, x: 60 }}
|
||||||
|
transition={{ duration: 0.20 }}
|
||||||
|
>
|
||||||
|
{(!currentWorkflow.name || !currentWorkflow.entityId) && (
|
||||||
|
<Tip text="Please fill in workflow name and associated entity to start configuring workflow." />
|
||||||
|
)}
|
||||||
|
<WorkflowForm
|
||||||
|
workflow={currentWorkflow}
|
||||||
|
onWorkflowChange={onWorkflowChange}
|
||||||
|
entityApprovers={entityApprovers}
|
||||||
|
entityAvailableFormIntakers={entityAvailableFormIntakers}
|
||||||
|
isLoading={isLoading}
|
||||||
|
isRedirecting={isRedirecting}
|
||||||
|
/>
|
||||||
|
</motion.div>
|
||||||
|
</LayoutGroup>
|
||||||
|
</AnimatePresence>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</form>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
455
src/pages/approval-workflows/index.tsx
Normal file
455
src/pages/approval-workflows/index.tsx
Normal file
@@ -0,0 +1,455 @@
|
|||||||
|
import Tip from "@/components/ApprovalWorkflows/Tip";
|
||||||
|
import Button from "@/components/Low/Button";
|
||||||
|
import Input from "@/components/Low/Input";
|
||||||
|
import Select from "@/components/Low/Select";
|
||||||
|
import useApprovalWorkflows from "@/hooks/useApprovalWorkflows";
|
||||||
|
import { Module, ModuleTypeLabels } from "@/interfaces";
|
||||||
|
import { ApprovalWorkflow, ApprovalWorkflowStatus, ApprovalWorkflowStatusLabel, StepTypeLabel } from "@/interfaces/approval.workflow";
|
||||||
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
|
import { User } from "@/interfaces/user";
|
||||||
|
import { sessionOptions } from "@/lib/session";
|
||||||
|
import { mapBy, redirect, serialize } from "@/utils";
|
||||||
|
import { requestUser } from "@/utils/api";
|
||||||
|
import { getApprovalWorkflows } from "@/utils/approval.workflows.be";
|
||||||
|
import { getEntitiesWithRoles } from "@/utils/entities.be";
|
||||||
|
import { shouldRedirectHome } from "@/utils/navigation.disabled";
|
||||||
|
import { doesEntityAllow, findAllowedEntities } from "@/utils/permissions";
|
||||||
|
import { isAdmin } from "@/utils/users";
|
||||||
|
import { getSpecificUsers } from "@/utils/users.be";
|
||||||
|
import { createColumnHelper, flexRender, getCoreRowModel, useReactTable, getPaginationRowModel } from "@tanstack/react-table";
|
||||||
|
import axios from "axios";
|
||||||
|
import clsx from "clsx";
|
||||||
|
import { withIronSessionSsr } from "iron-session/next";
|
||||||
|
import Head from "next/head";
|
||||||
|
import Link from "next/link";
|
||||||
|
import { useRouter } from "next/router";
|
||||||
|
import { useEffect, useState } from "react";
|
||||||
|
import { BsTrash } from "react-icons/bs";
|
||||||
|
import { FaRegEdit } from "react-icons/fa";
|
||||||
|
import { IoIosAddCircleOutline } from "react-icons/io";
|
||||||
|
import { toast, ToastContainer } from "react-toastify";
|
||||||
|
|
||||||
|
const StatusClassNames: { [key in ApprovalWorkflowStatus]: string } = {
|
||||||
|
approved:
|
||||||
|
"bg-green-100 text-green-800 border border-green-300 before:content-[''] before:w-2 before:h-2 before:bg-green-500 before:rounded-full before:inline-block before:mr-2",
|
||||||
|
pending:
|
||||||
|
"bg-orange-100 text-orange-800 border border-orange-300 before:content-[''] before:w-2 before:h-2 before:bg-orange-500 before:rounded-full before:inline-block before:mr-2",
|
||||||
|
rejected:
|
||||||
|
"bg-red-100 text-red-800 border border-red-300 before:content-[''] before:w-2 before:h-2 before:bg-red-500 before:rounded-full before:inline-block before:mr-2",
|
||||||
|
};
|
||||||
|
|
||||||
|
type CustomStatus = ApprovalWorkflowStatus | undefined;
|
||||||
|
type CustomEntity = EntityWithRoles["id"] | undefined;
|
||||||
|
|
||||||
|
const STATUS_OPTIONS = [
|
||||||
|
{
|
||||||
|
label: "Approved",
|
||||||
|
value: "approved",
|
||||||
|
filter: (x: ApprovalWorkflow) => x.status === "approved",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Pending",
|
||||||
|
value: "pending",
|
||||||
|
filter: (x: ApprovalWorkflow) => x.status === "pending",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: "Rejected",
|
||||||
|
value: "rejected",
|
||||||
|
filter: (x: ApprovalWorkflow) => x.status === "rejected",
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
const columnHelper = createColumnHelper<ApprovalWorkflow>();
|
||||||
|
|
||||||
|
export const getServerSideProps = withIronSessionSsr(async ({ req, res }) => {
|
||||||
|
const user = await requestUser(req, res);
|
||||||
|
if (!user) return redirect("/login");
|
||||||
|
|
||||||
|
if (shouldRedirectHome(user) || !["admin", "developer", "teacher", "corporate", "mastercorporate"].includes(user.type)) return redirect("/");
|
||||||
|
|
||||||
|
const entityIDS = mapBy(user.entities, "id");
|
||||||
|
const entities = await getEntitiesWithRoles(isAdmin(user) ? undefined : entityIDS);
|
||||||
|
const allowedEntities = findAllowedEntities(user, entities, "view_workflows");
|
||||||
|
|
||||||
|
const workflows = await getApprovalWorkflows("active-workflows", allowedEntities.map(entity => entity.id));
|
||||||
|
|
||||||
|
const allAssigneeIds: string[] = [
|
||||||
|
...new Set(
|
||||||
|
workflows
|
||||||
|
.map(workflow => workflow.steps
|
||||||
|
.map(step => step.assignees)
|
||||||
|
.flat()
|
||||||
|
).flat()
|
||||||
|
)
|
||||||
|
];
|
||||||
|
|
||||||
|
return {
|
||||||
|
props: serialize({
|
||||||
|
user,
|
||||||
|
initialWorkflows: workflows,
|
||||||
|
workflowsAssignees: await getSpecificUsers(allAssigneeIds),
|
||||||
|
userEntitiesWithLabel: allowedEntities,
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
}, sessionOptions);
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
user: User;
|
||||||
|
initialWorkflows: ApprovalWorkflow[];
|
||||||
|
workflowsAssignees: User[];
|
||||||
|
userEntitiesWithLabel: EntityWithRoles[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export default function ApprovalWorkflows({ user, initialWorkflows, workflowsAssignees, userEntitiesWithLabel }: Props) {
|
||||||
|
const entitiesString = userEntitiesWithLabel.map(entity => entity.id).join(",");
|
||||||
|
const { workflows, reload } = useApprovalWorkflows(entitiesString);
|
||||||
|
const currentWorkflows = workflows || initialWorkflows;
|
||||||
|
|
||||||
|
const [filteredWorkflows, setFilteredWorkflows] = useState<ApprovalWorkflow[]>([]);
|
||||||
|
|
||||||
|
const [statusFilter, setStatusFilter] = useState<CustomStatus>(undefined);
|
||||||
|
const [entityFilter, setEntityFilter] = useState<CustomEntity>(undefined);
|
||||||
|
const [nameFilter, setNameFilter] = useState<string>("");
|
||||||
|
|
||||||
|
const router = useRouter();
|
||||||
|
|
||||||
|
/* const allowedEntities = useAllowedEntities(user, userEntitiesWithLabel, "view_workflows");
|
||||||
|
const allowedSomeEntities = useAllowedEntitiesSomePermissions(user, userEntitiesWithLabel, ["view_workflows", "create_workflow"]); */
|
||||||
|
|
||||||
|
const ENTITY_OPTIONS = [
|
||||||
|
...userEntitiesWithLabel
|
||||||
|
.map((entity) => ({
|
||||||
|
label: entity.label,
|
||||||
|
value: entity.id,
|
||||||
|
filter: (x: ApprovalWorkflow) => x.entityId === entity.id,
|
||||||
|
}))
|
||||||
|
.sort((a, b) => a.label.localeCompare(b.label)),
|
||||||
|
];
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const filters: Array<(workflow: ApprovalWorkflow) => boolean> = [];
|
||||||
|
|
||||||
|
if (statusFilter && statusFilter !== undefined) {
|
||||||
|
const statusOption = STATUS_OPTIONS.find((x) => x.value === statusFilter);
|
||||||
|
if (statusOption && statusOption.filter) {
|
||||||
|
filters.push(statusOption.filter);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (entityFilter && entityFilter !== undefined) {
|
||||||
|
const entityOption = ENTITY_OPTIONS.find((x) => x.value === entityFilter);
|
||||||
|
if (entityOption && entityOption.filter) {
|
||||||
|
filters.push(entityOption.filter);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (nameFilter.trim() !== "") {
|
||||||
|
const nameFilterFunction = (workflow: ApprovalWorkflow) => workflow.name.toLowerCase().includes(nameFilter.toLowerCase());
|
||||||
|
filters.push(nameFilterFunction);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Apply all filters
|
||||||
|
const filtered = currentWorkflows.filter((workflow) => filters.every((filterFn) => filterFn(workflow)));
|
||||||
|
setFilteredWorkflows(filtered);
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, [currentWorkflows, statusFilter, entityFilter, nameFilter]);
|
||||||
|
|
||||||
|
const handleNameFilterChange = (name: ApprovalWorkflow["name"]) => {
|
||||||
|
setNameFilter(name);
|
||||||
|
};
|
||||||
|
|
||||||
|
const deleteApprovalWorkflow = (id: string | undefined, name: string) => {
|
||||||
|
if (id === undefined) return;
|
||||||
|
if (!confirm(`Are you sure you want to delete this Approval Workflow?`)) return;
|
||||||
|
|
||||||
|
axios
|
||||||
|
.delete(`/api/approval-workflows/${id}`)
|
||||||
|
.then(() => {
|
||||||
|
toast.success(`Successfully deleted ${name} Approval Workflow.`);
|
||||||
|
reload();
|
||||||
|
})
|
||||||
|
.catch((reason) => {
|
||||||
|
if (reason.response.status === 403) {
|
||||||
|
toast.error("You do not have permission to delete this Approval Workflow!");
|
||||||
|
} else {
|
||||||
|
toast.error("Something went wrong, please try again later.");
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
const columns = [
|
||||||
|
columnHelper.accessor("name", {
|
||||||
|
header: "EXAM NAME",
|
||||||
|
cell: (info) => <span className="font-medium">{info.getValue()}</span>,
|
||||||
|
}),
|
||||||
|
columnHelper.accessor("modules", {
|
||||||
|
header: "MODULES",
|
||||||
|
cell: (info) => (
|
||||||
|
<div className="flex flex-wrap gap-2">
|
||||||
|
{info.getValue().map((module: Module, index: number) => (
|
||||||
|
<span
|
||||||
|
key={index}
|
||||||
|
/* className="inline-block rounded-full px-3 py-1 text-sm font-medium bg-indigo-100 border border-indigo-300 text-indigo-900"> */
|
||||||
|
className={clsx("inline-block rounded-full px-3 py-1 text-sm font-medium text-white",
|
||||||
|
module === "speaking" ? "bg-ielts-speaking" :
|
||||||
|
module === "reading" ? "bg-ielts-reading" :
|
||||||
|
module === "writing" ? "bg-ielts-writing" :
|
||||||
|
module === "listening" ? "bg-ielts-listening" :
|
||||||
|
module === "level" ? "bg-ielts-level" :
|
||||||
|
"bg-slate-700"
|
||||||
|
)}>
|
||||||
|
{ModuleTypeLabels[module]}
|
||||||
|
</span>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
),
|
||||||
|
}),
|
||||||
|
columnHelper.accessor("status", {
|
||||||
|
header: "STATUS",
|
||||||
|
cell: (info) => (
|
||||||
|
<span
|
||||||
|
className={clsx(
|
||||||
|
"inline-block rounded-full px-3 py-1 text-sm font-medium text-left w-[110px]",
|
||||||
|
StatusClassNames[info.getValue()],
|
||||||
|
)}>
|
||||||
|
{ApprovalWorkflowStatusLabel[info.getValue()]}
|
||||||
|
</span>
|
||||||
|
),
|
||||||
|
}),
|
||||||
|
columnHelper.accessor("entityId", {
|
||||||
|
header: "ENTITY",
|
||||||
|
cell: (info) => <span className="font-medium">{userEntitiesWithLabel.find((entity) => entity.id === info.getValue())?.label}</span>,
|
||||||
|
}),
|
||||||
|
columnHelper.accessor("steps", {
|
||||||
|
id: "currentAssignees",
|
||||||
|
header: "CURRENT ASSIGNEES",
|
||||||
|
cell: (info) => {
|
||||||
|
const steps = info.row.original.steps;
|
||||||
|
const currentStep = steps.find((step) => !step.completed);
|
||||||
|
const rejected = steps.find((step) => step.rejected);
|
||||||
|
|
||||||
|
if (rejected) return "";
|
||||||
|
|
||||||
|
const assignees = currentStep?.assignees.map((assigneeId) => {
|
||||||
|
const assignee = workflowsAssignees.find((user) => user.id === assigneeId);
|
||||||
|
return assignee?.name || "Unknown Assignee";
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex flex-wrap gap-2">
|
||||||
|
{assignees?.map((assigneeName: string, index: number) => (
|
||||||
|
<span
|
||||||
|
key={index}
|
||||||
|
className="inline-block rounded-full px-3 py-1 text-sm font-medium bg-gray-100 border border-gray-300 text-gray-900">
|
||||||
|
{assigneeName}
|
||||||
|
</span>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
columnHelper.accessor("steps", {
|
||||||
|
id: "currentStep",
|
||||||
|
header: "CURRENT STEP",
|
||||||
|
cell: (info) => {
|
||||||
|
const steps = info.row.original.steps;
|
||||||
|
const currentStep = steps.find((step) => !step.completed);
|
||||||
|
const rejected = steps.find((step) => step.rejected);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<span className="font-medium">
|
||||||
|
{currentStep && !rejected ? `Step ${currentStep.stepNumber}: ${StepTypeLabel[currentStep.stepType]}` : "Completed"}
|
||||||
|
</span>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
columnHelper.accessor("steps", {
|
||||||
|
header: "ACTIONS",
|
||||||
|
id: "actions",
|
||||||
|
cell: ({ row }) => {
|
||||||
|
const steps = row.original.steps;
|
||||||
|
const currentStep = steps.find((step) => !step.completed);
|
||||||
|
const rejected = steps.find((step) => step.rejected);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="flex gap-4">
|
||||||
|
<button
|
||||||
|
data-tip="Delete"
|
||||||
|
className="cursor-pointer tooltip"
|
||||||
|
disabled={!doesEntityAllow(user, userEntitiesWithLabel.find(entity => entity.id === row.original.entityId)!, "delete_workflow")}
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
deleteApprovalWorkflow(row.original._id?.toString(), row.original.name);
|
||||||
|
}}>
|
||||||
|
<BsTrash className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
|
</button>
|
||||||
|
|
||||||
|
{currentStep && !rejected && (
|
||||||
|
<button
|
||||||
|
data-tip="Edit"
|
||||||
|
className="cursor-pointer tooltip"
|
||||||
|
disabled={!doesEntityAllow(user, userEntitiesWithLabel.find(entity => entity.id === row.original.entityId)!, "edit_workflow")}
|
||||||
|
onClick={(e) => {
|
||||||
|
e.stopPropagation();
|
||||||
|
router.push(`/approval-workflows/${row.original._id?.toString()}/edit`);
|
||||||
|
}}>
|
||||||
|
<FaRegEdit className="hover:text-mti-purple-light transition ease-in-out duration-300" />
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
];
|
||||||
|
|
||||||
|
const [pagination, setPagination] = useState({
|
||||||
|
pageIndex: 0,
|
||||||
|
pageSize: 10,
|
||||||
|
});
|
||||||
|
|
||||||
|
const table = useReactTable({
|
||||||
|
data: filteredWorkflows,
|
||||||
|
columns: columns,
|
||||||
|
state: {
|
||||||
|
pagination,
|
||||||
|
},
|
||||||
|
onPaginationChange: setPagination,
|
||||||
|
getCoreRowModel: getCoreRowModel(),
|
||||||
|
getPaginationRowModel: getPaginationRowModel(),
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<Head>
|
||||||
|
<title>Approval Workflows Panel | EnCoach</title>
|
||||||
|
<meta
|
||||||
|
name="description"
|
||||||
|
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
||||||
|
/>
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
|
<link rel="icon" href="/favicon.ico" />
|
||||||
|
</Head>
|
||||||
|
<ToastContainer />
|
||||||
|
<h1 className="text-2xl font-semibold">Approval Workflows</h1>
|
||||||
|
|
||||||
|
<div className="flex flex-row">
|
||||||
|
<Link href={"/approval-workflows/create"}>
|
||||||
|
<Button color="purple" variant="solid" className="min-w-fit text-lg font-medium flex items-center gap-2 text-left">
|
||||||
|
<IoIosAddCircleOutline className="size-6" />
|
||||||
|
Configure Workflows
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div className="flex w-full items-center gap-4">
|
||||||
|
<div className="flex w-full flex-col gap-3">
|
||||||
|
<label className="text-mti-gray-dim text-base font-normal">Name</label>
|
||||||
|
<Input name="nameFilter" type="text" value={nameFilter} onChange={handleNameFilterChange} placeholder="Filter by name..." />
|
||||||
|
</div>
|
||||||
|
<div className="flex w-full flex-col gap-3">
|
||||||
|
<label className="text-mti-gray-dim text-base font-normal">Status</label>
|
||||||
|
<Select
|
||||||
|
options={STATUS_OPTIONS}
|
||||||
|
value={STATUS_OPTIONS.find((x) => x.value === statusFilter)}
|
||||||
|
onChange={(value) => setStatusFilter((value?.value as ApprovalWorkflowStatus) ?? undefined)}
|
||||||
|
isClearable
|
||||||
|
placeholder="Filter by status..."
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div className="flex w-full flex-col gap-3">
|
||||||
|
<label className="text-mti-gray-dim text-base font-normal">Entity</label>
|
||||||
|
<Select
|
||||||
|
options={ENTITY_OPTIONS}
|
||||||
|
value={ENTITY_OPTIONS.find((x) => x.value === entityFilter)}
|
||||||
|
onChange={(value) => setEntityFilter((value?.value as CustomEntity) ?? undefined)}
|
||||||
|
isClearable
|
||||||
|
placeholder="Filter by entity..."
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<Tip text="An exam submission will instantiate the approval workflow configured for the exam author. The exam will be valid only when all the steps of the workflow have been approved."></Tip>
|
||||||
|
|
||||||
|
<div className="px-6 pb-4 bg-mti-purple-ultralight rounded-2xl border-2 border-mti-purple-light border-opacity-40">
|
||||||
|
<table className="w-full table-auto border-separate border-spacing-y-2" style={{ tableLayout: "auto" }}>
|
||||||
|
<thead>
|
||||||
|
{table.getHeaderGroups().map((headerGroup) => (
|
||||||
|
<tr key={headerGroup.id}>
|
||||||
|
{headerGroup.headers.map((header) => (
|
||||||
|
<th key={header.id} className="px-3 py-2 text-left text-mti-purple-ultradark">
|
||||||
|
{header.isPlaceholder ? null : flexRender(header.column.columnDef.header, header.getContext())}
|
||||||
|
</th>
|
||||||
|
))}
|
||||||
|
</tr>
|
||||||
|
))}
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{table.getRowModel().rows.map((row) => (
|
||||||
|
<tr
|
||||||
|
key={row.id}
|
||||||
|
onClick={() => (window.location.href = `/approval-workflows/${row.original._id?.toString()}`)}
|
||||||
|
style={{ cursor: "pointer" }}
|
||||||
|
className="bg-purple-50">
|
||||||
|
{row.getVisibleCells().map((cell, cellIndex) => {
|
||||||
|
const lastCellIndex = row.getVisibleCells().length - 1;
|
||||||
|
|
||||||
|
let cellClasses = "pl-3 pr-4 py-2 border-y-2 border-mti-purple-light border-opacity-60";
|
||||||
|
if (cellIndex === 0) {
|
||||||
|
cellClasses += " border-l-2 rounded-l-2xl";
|
||||||
|
}
|
||||||
|
if (cellIndex === lastCellIndex) {
|
||||||
|
cellClasses += " border-r-2 rounded-r-2xl";
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
<td key={cellIndex} className={cellClasses}>
|
||||||
|
{flexRender(cell.column.columnDef.cell, cell.getContext())}
|
||||||
|
</td>
|
||||||
|
);
|
||||||
|
})}
|
||||||
|
</tr>
|
||||||
|
))}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div className="mt-2 flex flex-row gap-2 w-full justify-end items-center">
|
||||||
|
<button
|
||||||
|
onClick={() => table.setPageIndex(0)}
|
||||||
|
disabled={!table.getCanPreviousPage()}
|
||||||
|
className="px-3 py-2 rounded-md text-sm font-semibold text-mti-purple-ultradark border border-mti-purple-light
|
||||||
|
bg-white hover:bg-mti-purple-light hover:text-white transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
{"<<"}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={() => table.previousPage()}
|
||||||
|
disabled={!table.getCanPreviousPage()}
|
||||||
|
className="px-3 py-2 rounded-md text-sm font-semibold text-mti-purple-ultradark border border-mti-purple-light
|
||||||
|
bg-white hover:bg-mti-purple-light hover:text-white transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
{"<"}
|
||||||
|
</button>
|
||||||
|
<span className="px-4 text-sm font-medium">
|
||||||
|
Page {table.getState().pagination.pageIndex + 1} of {table.getPageCount()}
|
||||||
|
</span>
|
||||||
|
<button
|
||||||
|
onClick={() => table.nextPage()}
|
||||||
|
disabled={!table.getCanNextPage()}
|
||||||
|
className="px-3 py-2 rounded-md text-sm font-semibold text-mti-purple-ultradark border border-mti-purple-light
|
||||||
|
bg-white hover:bg-mti-purple-light hover:text-white transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
{">"}
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={() => table.setPageIndex(table.getPageCount() - 1)}
|
||||||
|
disabled={!table.getCanNextPage()}
|
||||||
|
className="px-3 py-2 rounded-md text-sm font-semibold text-mti-purple-ultradark border border-mti-purple-light
|
||||||
|
bg-white hover:bg-mti-purple-light hover:text-white transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
|
||||||
|
>
|
||||||
|
{">>"}
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
File diff suppressed because it is too large
Load Diff
@@ -1,229 +1,336 @@
|
|||||||
import Layout from "@/components/High/Layout";
|
|
||||||
import Separator from "@/components/Low/Separator";
|
import Separator from "@/components/Low/Separator";
|
||||||
import AssignmentCard from "@/components/AssignmentCard";
|
import AssignmentCard from "@/components/AssignmentCard";
|
||||||
import AssignmentView from "@/components/AssignmentView";
|
|
||||||
import { useAllowedEntities } from "@/hooks/useEntityPermissions";
|
import { useAllowedEntities } from "@/hooks/useEntityPermissions";
|
||||||
import { useListSearch } from "@/hooks/useListSearch";
|
import { useListSearch } from "@/hooks/useListSearch";
|
||||||
import usePagination from "@/hooks/usePagination";
|
import usePagination from "@/hooks/usePagination";
|
||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
import { Assignment } from "@/interfaces/results";
|
import { Assignment } from "@/interfaces/results";
|
||||||
import { CorporateUser, Group, User } from "@/interfaces/user";
|
import { User } from "@/interfaces/user";
|
||||||
import { sessionOptions } from "@/lib/session";
|
import { sessionOptions } from "@/lib/session";
|
||||||
import { getUserCompanyName } from "@/resources/user";
|
|
||||||
import { findBy, mapBy, redirect, serialize } from "@/utils";
|
import { findBy, mapBy, redirect, serialize } from "@/utils";
|
||||||
import { requestUser } from "@/utils/api";
|
import { requestUser } from "@/utils/api";
|
||||||
import {
|
import {
|
||||||
activeAssignmentFilter,
|
activeAssignmentFilter,
|
||||||
archivedAssignmentFilter,
|
archivedAssignmentFilter,
|
||||||
futureAssignmentFilter,
|
futureAssignmentFilter,
|
||||||
pastAssignmentFilter,
|
pastAssignmentFilter,
|
||||||
startHasExpiredAssignmentFilter,
|
startHasExpiredAssignmentFilter,
|
||||||
} from "@/utils/assignments";
|
} from "@/utils/assignments";
|
||||||
import { getAssignments, getEntitiesAssignments } from "@/utils/assignments.be";
|
import { getAssignments, getEntitiesAssignments } from "@/utils/assignments.be";
|
||||||
import { getEntitiesWithRoles } from "@/utils/entities.be";
|
import { getEntitiesWithRoles } from "@/utils/entities.be";
|
||||||
import { getGroups, getGroupsByEntities } from "@/utils/groups.be";
|
|
||||||
import { checkAccess, findAllowedEntities } from "@/utils/permissions";
|
import { checkAccess, findAllowedEntities } from "@/utils/permissions";
|
||||||
import { getEntitiesUsers, getUsers } from "@/utils/users.be";
|
import { getEntitiesUsers, getUsers } from "@/utils/users.be";
|
||||||
import { withIronSessionSsr } from "iron-session/next";
|
import { withIronSessionSsr } from "iron-session/next";
|
||||||
import { groupBy } from "lodash";
|
|
||||||
import Head from "next/head";
|
import Head from "next/head";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { useMemo, useState } from "react";
|
import { useMemo } from "react";
|
||||||
import { BsChevronLeft, BsPlus } from "react-icons/bs";
|
import { BsChevronLeft, BsPlus } from "react-icons/bs";
|
||||||
|
|
||||||
export const getServerSideProps = withIronSessionSsr(async ({ req, res }) => {
|
export const getServerSideProps = withIronSessionSsr(async ({ req, res }) => {
|
||||||
const user = await requestUser(req, res)
|
const user = await requestUser(req, res);
|
||||||
if (!user) return redirect("/login")
|
if (!user) return redirect("/login");
|
||||||
|
|
||||||
if (!checkAccess(user, ["admin", "developer", "corporate", "teacher", "mastercorporate"]))
|
if (
|
||||||
return redirect("/")
|
!checkAccess(user, [
|
||||||
|
"admin",
|
||||||
|
"developer",
|
||||||
|
"corporate",
|
||||||
|
"teacher",
|
||||||
|
"mastercorporate",
|
||||||
|
])
|
||||||
|
)
|
||||||
|
return redirect("/");
|
||||||
|
const isAdmin = checkAccess(user, ["developer", "admin"]);
|
||||||
|
const entityIDS = mapBy(user.entities, "id") || [];
|
||||||
|
const entities = await (isAdmin
|
||||||
|
? getEntitiesWithRoles()
|
||||||
|
: getEntitiesWithRoles(entityIDS));
|
||||||
|
|
||||||
const entityIDS = mapBy(user.entities, "id") || [];
|
const allowedEntities = findAllowedEntities(
|
||||||
const entities = await (checkAccess(user, ["developer", "admin"]) ? getEntitiesWithRoles() : getEntitiesWithRoles(entityIDS));
|
user,
|
||||||
|
entities,
|
||||||
|
"view_assignments"
|
||||||
|
);
|
||||||
|
const [users, assignments] = await Promise.all([
|
||||||
|
await (isAdmin
|
||||||
|
? getUsers({}, 0, {}, { _id: 0, id: 1, name: 1 })
|
||||||
|
: getEntitiesUsers(mapBy(allowedEntities, "id"), {}, 0, {
|
||||||
|
_id: 0,
|
||||||
|
id: 1,
|
||||||
|
name: 1,
|
||||||
|
})),
|
||||||
|
await (isAdmin
|
||||||
|
? getAssignments()
|
||||||
|
: getEntitiesAssignments(mapBy(allowedEntities, "id"))),
|
||||||
|
]);
|
||||||
|
|
||||||
const allowedEntities = findAllowedEntities(user, entities, "view_assignments")
|
return {
|
||||||
|
props: serialize({ user, users, entities: allowedEntities, assignments }),
|
||||||
const users =
|
};
|
||||||
await (checkAccess(user, ["developer", "admin"]) ? getUsers() : getEntitiesUsers(mapBy(allowedEntities, 'id')));
|
|
||||||
|
|
||||||
const assignments =
|
|
||||||
await (checkAccess(user, ["developer", "admin"]) ? getAssignments() : getEntitiesAssignments(mapBy(allowedEntities, 'id')));
|
|
||||||
|
|
||||||
return { props: serialize({ user, users, entities: allowedEntities, assignments }) };
|
|
||||||
}, sessionOptions);
|
}, sessionOptions);
|
||||||
|
|
||||||
const SEARCH_FIELDS = [["name"]];
|
const SEARCH_FIELDS = [["name"]];
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
assignments: Assignment[];
|
assignments: Assignment[];
|
||||||
entities: EntityWithRoles[]
|
entities: EntityWithRoles[];
|
||||||
user: User;
|
user: User;
|
||||||
users: User[];
|
users: User[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function AssignmentsPage({ assignments, entities, user, users }: Props) {
|
export default function AssignmentsPage({
|
||||||
const entitiesAllowCreate = useAllowedEntities(user, entities, 'create_assignment')
|
assignments,
|
||||||
const entitiesAllowEdit = useAllowedEntities(user, entities, 'edit_assignment')
|
entities,
|
||||||
const entitiesAllowArchive = useAllowedEntities(user, entities, 'archive_assignment')
|
user,
|
||||||
|
users,
|
||||||
|
}: Props) {
|
||||||
|
const entitiesAllowCreate = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"create_assignment"
|
||||||
|
);
|
||||||
|
const entitiesAllowEdit = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"edit_assignment"
|
||||||
|
);
|
||||||
|
const entitiesAllowArchive = useAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"archive_assignment"
|
||||||
|
);
|
||||||
|
|
||||||
const activeAssignments = useMemo(() => assignments.filter(activeAssignmentFilter), [assignments]);
|
const activeAssignments = useMemo(
|
||||||
const plannedAssignments = useMemo(() => assignments.filter(futureAssignmentFilter), [assignments]);
|
() => assignments.filter(activeAssignmentFilter),
|
||||||
const pastAssignments = useMemo(() => assignments.filter(pastAssignmentFilter), [assignments]);
|
[assignments]
|
||||||
const startExpiredAssignments = useMemo(() => assignments.filter(startHasExpiredAssignmentFilter), [assignments]);
|
);
|
||||||
const archivedAssignments = useMemo(() => assignments.filter(archivedAssignmentFilter), [assignments]);
|
const plannedAssignments = useMemo(
|
||||||
|
() => assignments.filter(futureAssignmentFilter),
|
||||||
|
[assignments]
|
||||||
|
);
|
||||||
|
const pastAssignments = useMemo(
|
||||||
|
() => assignments.filter(pastAssignmentFilter),
|
||||||
|
[assignments]
|
||||||
|
);
|
||||||
|
const startExpiredAssignments = useMemo(
|
||||||
|
() => assignments.filter(startHasExpiredAssignmentFilter),
|
||||||
|
[assignments]
|
||||||
|
);
|
||||||
|
const archivedAssignments = useMemo(
|
||||||
|
() => assignments.filter(archivedAssignmentFilter),
|
||||||
|
[assignments]
|
||||||
|
);
|
||||||
|
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
const { rows: activeRows, renderSearch: renderActive } = useListSearch(SEARCH_FIELDS, activeAssignments);
|
const { rows: activeRows, renderSearch: renderActive } = useListSearch(
|
||||||
const { rows: plannedRows, renderSearch: renderPlanned } = useListSearch(SEARCH_FIELDS, plannedAssignments);
|
SEARCH_FIELDS,
|
||||||
const { rows: pastRows, renderSearch: renderPast } = useListSearch(SEARCH_FIELDS, pastAssignments);
|
activeAssignments
|
||||||
const { rows: expiredRows, renderSearch: renderExpired } = useListSearch(SEARCH_FIELDS, startExpiredAssignments);
|
);
|
||||||
const { rows: archivedRows, renderSearch: renderArchived } = useListSearch(SEARCH_FIELDS, archivedAssignments);
|
const { rows: plannedRows, renderSearch: renderPlanned } = useListSearch(
|
||||||
|
SEARCH_FIELDS,
|
||||||
|
plannedAssignments
|
||||||
|
);
|
||||||
|
const { rows: pastRows, renderSearch: renderPast } = useListSearch(
|
||||||
|
SEARCH_FIELDS,
|
||||||
|
pastAssignments
|
||||||
|
);
|
||||||
|
const { rows: expiredRows, renderSearch: renderExpired } = useListSearch(
|
||||||
|
SEARCH_FIELDS,
|
||||||
|
startExpiredAssignments
|
||||||
|
);
|
||||||
|
const { rows: archivedRows, renderSearch: renderArchived } = useListSearch(
|
||||||
|
SEARCH_FIELDS,
|
||||||
|
archivedAssignments
|
||||||
|
);
|
||||||
|
|
||||||
const { items: activeItems, renderMinimal: paginationActive } = usePagination(activeRows, 16);
|
const { items: activeItems, renderMinimal: paginationActive } = usePagination(
|
||||||
const { items: plannedItems, renderMinimal: paginationPlanned } = usePagination(plannedRows, 16);
|
activeRows,
|
||||||
const { items: pastItems, renderMinimal: paginationPast } = usePagination(pastRows, 16);
|
16
|
||||||
const { items: expiredItems, renderMinimal: paginationExpired } = usePagination(expiredRows, 16);
|
);
|
||||||
const { items: archivedItems, renderMinimal: paginationArchived } = usePagination(archivedRows, 16);
|
const { items: plannedItems, renderMinimal: paginationPlanned } =
|
||||||
|
usePagination(plannedRows, 16);
|
||||||
|
const { items: pastItems, renderMinimal: paginationPast } = usePagination(
|
||||||
|
pastRows,
|
||||||
|
16
|
||||||
|
);
|
||||||
|
const { items: expiredItems, renderMinimal: paginationExpired } =
|
||||||
|
usePagination(expiredRows, 16);
|
||||||
|
const { items: archivedItems, renderMinimal: paginationArchived } =
|
||||||
|
usePagination(archivedRows, 16);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Head>
|
<Head>
|
||||||
<title>Assignments | EnCoach</title>
|
<title>Assignments | EnCoach</title>
|
||||||
<meta
|
<meta
|
||||||
name="description"
|
name="description"
|
||||||
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
||||||
/>
|
/>
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
<link rel="icon" href="/favicon.ico" />
|
<link rel="icon" href="/favicon.ico" />
|
||||||
</Head>
|
</Head>
|
||||||
<Layout user={user}>
|
<>
|
||||||
<div className="flex flex-col gap-4">
|
<div className="flex flex-col gap-4">
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-2">
|
||||||
<Link href="/dashboard" className="text-mti-purple hover:text-mti-purple-dark transition ease-in-out duration-300 text-xl">
|
<Link
|
||||||
<BsChevronLeft />
|
href="/dashboard"
|
||||||
</Link>
|
className="text-mti-purple hover:text-mti-purple-dark transition ease-in-out duration-300 text-xl"
|
||||||
<h2 className="font-bold text-2xl">Assignments</h2>
|
>
|
||||||
</div>
|
<BsChevronLeft />
|
||||||
<Separator />
|
</Link>
|
||||||
</div>
|
<h2 className="font-bold text-2xl">Assignments</h2>
|
||||||
<div className="flex flex-col gap-2">
|
</div>
|
||||||
<span className="text-lg font-bold">Active Assignments Status</span>
|
<Separator />
|
||||||
<div className="flex items-center gap-4">
|
</div>
|
||||||
<span>
|
<div className="flex flex-col gap-2">
|
||||||
<b>Total:</b> {activeAssignments.reduce((acc, curr) => acc + curr.results.length, 0)}/
|
<span className="text-lg font-bold">Active Assignments Status</span>
|
||||||
{activeAssignments.reduce((acc, curr) => curr.exams.length + acc, 0)}
|
<div className="flex items-center gap-4">
|
||||||
</span>
|
<span>
|
||||||
</div>
|
<b>Total:</b>{" "}
|
||||||
</div>
|
{activeAssignments.reduce(
|
||||||
|
(acc, curr) => acc + curr.results.length,
|
||||||
|
0
|
||||||
|
)}
|
||||||
|
/
|
||||||
|
{activeAssignments.reduce(
|
||||||
|
(acc, curr) => curr.exams.length + acc,
|
||||||
|
0
|
||||||
|
)}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<section className="flex flex-col gap-4">
|
<section className="flex flex-col gap-4">
|
||||||
<h2 className="text-2xl font-semibold">Active Assignments ({activeAssignments.length})</h2>
|
<h2 className="text-2xl font-semibold">
|
||||||
<div className="w-full flex items-center gap-4">
|
Active Assignments ({activeAssignments.length})
|
||||||
{renderActive()}
|
</h2>
|
||||||
{paginationActive()}
|
<div className="w-full flex items-center gap-4">
|
||||||
</div>
|
{renderActive()}
|
||||||
<div className="flex flex-wrap gap-2">
|
{paginationActive()}
|
||||||
{activeItems.map((a) => (
|
</div>
|
||||||
<AssignmentCard {...a} entityObj={findBy(entities, 'id', a.entity)} users={users} onClick={() => router.push(`/assignments/${a.id}`)} key={a.id} />
|
<div className="flex flex-wrap gap-2">
|
||||||
))}
|
{activeItems.map((a) => (
|
||||||
</div>
|
<AssignmentCard
|
||||||
</section>
|
{...a}
|
||||||
|
entityObj={findBy(entities, "id", a.entity)}
|
||||||
|
users={users}
|
||||||
|
onClick={() => router.push(`/assignments/${a.id}`)}
|
||||||
|
key={a.id}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
<section className="flex flex-col gap-4">
|
<section className="flex flex-col gap-4">
|
||||||
<h2 className="text-2xl font-semibold">Planned Assignments ({plannedAssignments.length})</h2>
|
<h2 className="text-2xl font-semibold">
|
||||||
<div className="w-full flex items-center gap-4">
|
Planned Assignments ({plannedAssignments.length})
|
||||||
{renderPlanned()}
|
</h2>
|
||||||
{paginationPlanned()}
|
<div className="w-full flex items-center gap-4">
|
||||||
</div>
|
{renderPlanned()}
|
||||||
<div className="flex flex-wrap gap-2">
|
{paginationPlanned()}
|
||||||
<Link
|
</div>
|
||||||
href={entitiesAllowCreate.length > 0 ? "/assignments/creator" : ""}
|
<div className="flex flex-wrap gap-2">
|
||||||
className="w-[250px] h-[200px] flex flex-col gap-2 items-center justify-center bg-white hover:bg-mti-purple-ultralight text-mti-purple-light hover:text-mti-purple-dark border border-mti-gray-platinum hover:drop-shadow p-4 cursor-pointer rounded-xl transition ease-in-out duration-300">
|
<Link
|
||||||
<BsPlus className="text-6xl" />
|
href={
|
||||||
<span className="text-lg">New Assignment</span>
|
entitiesAllowCreate.length > 0 ? "/assignments/creator" : ""
|
||||||
</Link>
|
}
|
||||||
{plannedItems.map((a) => (
|
className="w-[250px] h-[200px] flex flex-col gap-2 items-center justify-center bg-white hover:bg-mti-purple-ultralight text-mti-purple-light hover:text-mti-purple-dark border border-mti-gray-platinum hover:drop-shadow p-4 cursor-pointer rounded-xl transition ease-in-out duration-300"
|
||||||
<AssignmentCard
|
>
|
||||||
{...a}
|
<BsPlus className="text-6xl" />
|
||||||
users={users}
|
<span className="text-lg">New Assignment</span>
|
||||||
entityObj={findBy(entities, 'id', a.entity)}
|
</Link>
|
||||||
onClick={
|
{plannedItems.map((a) => (
|
||||||
mapBy(entitiesAllowEdit, 'id').includes(a.entity || "")
|
<AssignmentCard
|
||||||
? () => router.push(`/assignments/creator/${a.id}`)
|
{...a}
|
||||||
: () => router.push(`/assignments/${a.id}`)
|
users={users}
|
||||||
}
|
entityObj={findBy(entities, "id", a.entity)}
|
||||||
key={a.id}
|
onClick={
|
||||||
/>
|
mapBy(entitiesAllowEdit, "id").includes(a.entity || "")
|
||||||
))}
|
? () => router.push(`/assignments/creator/${a.id}`)
|
||||||
</div>
|
: () => router.push(`/assignments/${a.id}`)
|
||||||
</section>
|
}
|
||||||
|
key={a.id}
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
<section className="flex flex-col gap-4">
|
<section className="flex flex-col gap-4">
|
||||||
<h2 className="text-2xl font-semibold">Past Assignments ({pastAssignments.length})</h2>
|
<h2 className="text-2xl font-semibold">
|
||||||
<div className="w-full flex items-center gap-4">
|
Past Assignments ({pastAssignments.length})
|
||||||
{renderPast()}
|
</h2>
|
||||||
{paginationPast()}
|
<div className="w-full flex items-center gap-4">
|
||||||
</div>
|
{renderPast()}
|
||||||
<div className="flex flex-wrap gap-2">
|
{paginationPast()}
|
||||||
{pastItems.map((a) => (
|
</div>
|
||||||
<AssignmentCard
|
<div className="flex flex-wrap gap-2">
|
||||||
{...a}
|
{pastItems.map((a) => (
|
||||||
users={users}
|
<AssignmentCard
|
||||||
entityObj={findBy(entities, 'id', a.entity)}
|
{...a}
|
||||||
onClick={() => router.push(`/assignments/${a.id}`)}
|
users={users}
|
||||||
key={a.id}
|
entityObj={findBy(entities, "id", a.entity)}
|
||||||
allowDownload
|
onClick={() => router.push(`/assignments/${a.id}`)}
|
||||||
allowArchive={mapBy(entitiesAllowArchive, 'id').includes(a.entity || "")}
|
key={a.id}
|
||||||
allowExcelDownload
|
allowDownload
|
||||||
/>
|
allowArchive={mapBy(entitiesAllowArchive, "id").includes(
|
||||||
))}
|
a.entity || ""
|
||||||
</div>
|
)}
|
||||||
</section>
|
allowExcelDownload
|
||||||
<section className="flex flex-col gap-4">
|
/>
|
||||||
<h2 className="text-2xl font-semibold">Assignments start expired ({startExpiredAssignments.length})</h2>
|
))}
|
||||||
<div className="w-full flex items-center gap-4">
|
</div>
|
||||||
{renderExpired()}
|
</section>
|
||||||
{paginationExpired()}
|
<section className="flex flex-col gap-4">
|
||||||
</div>
|
<h2 className="text-2xl font-semibold">
|
||||||
<div className="flex flex-wrap gap-2">
|
Assignments start expired ({startExpiredAssignments.length})
|
||||||
{expiredItems.map((a) => (
|
</h2>
|
||||||
<AssignmentCard
|
<div className="w-full flex items-center gap-4">
|
||||||
{...a}
|
{renderExpired()}
|
||||||
users={users}
|
{paginationExpired()}
|
||||||
entityObj={findBy(entities, 'id', a.entity)}
|
</div>
|
||||||
onClick={() => router.push(`/assignments/${a.id}`)}
|
<div className="flex flex-wrap gap-2">
|
||||||
key={a.id}
|
{expiredItems.map((a) => (
|
||||||
allowDownload
|
<AssignmentCard
|
||||||
allowArchive={mapBy(entitiesAllowArchive, 'id').includes(a.entity || "")}
|
{...a}
|
||||||
allowExcelDownload
|
users={users}
|
||||||
/>
|
entityObj={findBy(entities, "id", a.entity)}
|
||||||
))}
|
onClick={() => router.push(`/assignments/${a.id}`)}
|
||||||
</div>
|
key={a.id}
|
||||||
</section>
|
allowDownload
|
||||||
<section className="flex flex-col gap-4">
|
allowArchive={mapBy(entitiesAllowArchive, "id").includes(
|
||||||
<h2 className="text-2xl font-semibold">Archived Assignments ({archivedAssignments.length})</h2>
|
a.entity || ""
|
||||||
<div className="w-full flex items-center gap-4">
|
)}
|
||||||
{renderArchived()}
|
allowExcelDownload
|
||||||
{paginationArchived()}
|
/>
|
||||||
</div>
|
))}
|
||||||
<div className="flex flex-wrap gap-2">
|
</div>
|
||||||
{archivedItems.map((a) => (
|
</section>
|
||||||
<AssignmentCard
|
<section className="flex flex-col gap-4">
|
||||||
{...a}
|
<h2 className="text-2xl font-semibold">
|
||||||
users={users}
|
Archived Assignments ({archivedAssignments.length})
|
||||||
onClick={() => router.push(`/assignments/${a.id}`)}
|
</h2>
|
||||||
key={a.id}
|
<div className="w-full flex items-center gap-4">
|
||||||
entityObj={findBy(entities, 'id', a.entity)}
|
{renderArchived()}
|
||||||
allowDownload
|
{paginationArchived()}
|
||||||
allowUnarchive
|
</div>
|
||||||
allowExcelDownload
|
<div className="flex flex-wrap gap-2">
|
||||||
/>
|
{archivedItems.map((a) => (
|
||||||
))}
|
<AssignmentCard
|
||||||
</div>
|
{...a}
|
||||||
</section>
|
users={users}
|
||||||
</Layout>
|
onClick={() => router.push(`/assignments/${a.id}`)}
|
||||||
</>
|
key={a.id}
|
||||||
);
|
entityObj={findBy(entities, "id", a.entity)}
|
||||||
|
allowDownload
|
||||||
|
allowUnarchive
|
||||||
|
allowExcelDownload
|
||||||
|
/>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</>
|
||||||
|
</>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
/* eslint-disable @next/next/no-img-element */
|
/* eslint-disable @next/next/no-img-element */
|
||||||
import Layout from "@/components/High/Layout";
|
|
||||||
import Tooltip from "@/components/Low/Tooltip";
|
import Tooltip from "@/components/Low/Tooltip";
|
||||||
import { useEntityPermission } from "@/hooks/useEntityPermissions";
|
import { useEntityPermission } from "@/hooks/useEntityPermissions";
|
||||||
import { useListSearch } from "@/hooks/useListSearch";
|
import { useListSearch } from "@/hooks/useListSearch";
|
||||||
@@ -19,328 +18,510 @@ import { getEntityUsers, getSpecificUsers } from "@/utils/users.be";
|
|||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import { withIronSessionSsr } from "iron-session/next";
|
import { withIronSessionSsr } from "iron-session/next";
|
||||||
import { capitalize } from "lodash";
|
import { capitalize, last } from "lodash";
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
import Head from "next/head";
|
import Head from "next/head";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { Divider } from "primereact/divider";
|
import { Divider } from "primereact/divider";
|
||||||
import { useEffect, useMemo, useState } from "react";
|
import { useEffect, useMemo, useState } from "react";
|
||||||
import { BsBuilding, BsChevronLeft, BsClockFill, BsEnvelopeFill, BsFillPersonVcardFill, BsPlus, BsStopwatchFill, BsTag, BsTrash, BsX } from "react-icons/bs";
|
import {
|
||||||
|
BsBuilding,
|
||||||
|
BsChevronLeft,
|
||||||
|
BsClockFill,
|
||||||
|
BsEnvelopeFill,
|
||||||
|
BsFillPersonVcardFill,
|
||||||
|
BsPlus,
|
||||||
|
BsStopwatchFill,
|
||||||
|
BsTag,
|
||||||
|
BsTrash,
|
||||||
|
BsX,
|
||||||
|
} from "react-icons/bs";
|
||||||
import { toast, ToastContainer } from "react-toastify";
|
import { toast, ToastContainer } from "react-toastify";
|
||||||
|
|
||||||
export const getServerSideProps = withIronSessionSsr(async ({ req, res, params }) => {
|
export const getServerSideProps = withIronSessionSsr(
|
||||||
const user = await requestUser(req, res)
|
async ({ req, res, params }) => {
|
||||||
if (!user) return redirect("/login")
|
const user = await requestUser(req, res);
|
||||||
|
if (!user) return redirect("/login");
|
||||||
|
|
||||||
if (shouldRedirectHome(user)) return redirect("/")
|
if (shouldRedirectHome(user)) return redirect("/");
|
||||||
|
|
||||||
const { id } = params as { id: string };
|
const { id } = params as { id: string };
|
||||||
|
|
||||||
const group = await getGroup(id);
|
const group = await getGroup(id);
|
||||||
if (!group || !group.entity) return redirect("/classrooms")
|
if (!group || !group.entity) return redirect("/classrooms");
|
||||||
|
|
||||||
const entity = await getEntityWithRoles(group.entity)
|
const entity = await getEntityWithRoles(group.entity);
|
||||||
if (!entity) return redirect("/classrooms")
|
if (!entity) return redirect("/classrooms");
|
||||||
|
|
||||||
const canView = doesEntityAllow(user, entity, "view_classrooms")
|
const canView = doesEntityAllow(user, entity, "view_classrooms");
|
||||||
if (!canView) return redirect("/")
|
if (!canView) return redirect("/");
|
||||||
|
const [linkedUsers, users] = await Promise.all([
|
||||||
|
getEntityUsers(
|
||||||
|
entity.id,
|
||||||
|
0,
|
||||||
|
{},
|
||||||
|
{
|
||||||
|
_id: 0,
|
||||||
|
id: 1,
|
||||||
|
name: 1,
|
||||||
|
email: 1,
|
||||||
|
corporateInformation: 1,
|
||||||
|
type: 1,
|
||||||
|
profilePicture: 1,
|
||||||
|
subscriptionExpirationDate: 1,
|
||||||
|
lastLogin: 1,
|
||||||
|
}
|
||||||
|
),
|
||||||
|
getSpecificUsers([...group.participants, group.admin], {
|
||||||
|
_id: 0,
|
||||||
|
id: 1,
|
||||||
|
name: 1,
|
||||||
|
email: 1,
|
||||||
|
corporateInformation: 1,
|
||||||
|
type: 1,
|
||||||
|
profilePicture: 1,
|
||||||
|
subscriptionExpirationDate: 1,
|
||||||
|
lastLogin: 1,
|
||||||
|
}),
|
||||||
|
]);
|
||||||
|
|
||||||
const linkedUsers = await getEntityUsers(entity.id)
|
const groupWithUser = convertToUsers(group, users);
|
||||||
const users = await getSpecificUsers([...group.participants, group.admin]);
|
|
||||||
const groupWithUser = convertToUsers(group, users);
|
|
||||||
|
|
||||||
return {
|
return {
|
||||||
props: serialize({ user, group: groupWithUser, users: linkedUsers.filter(x => isAdmin(user) ? true : !isAdmin(x)), entity }),
|
props: serialize({
|
||||||
};
|
user,
|
||||||
}, sessionOptions);
|
group: groupWithUser,
|
||||||
|
users: linkedUsers.filter((x) => (isAdmin(user) ? true : !isAdmin(x))),
|
||||||
|
entity,
|
||||||
|
}),
|
||||||
|
};
|
||||||
|
},
|
||||||
|
sessionOptions
|
||||||
|
);
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
user: User;
|
user: User;
|
||||||
group: GroupWithUsers;
|
group: GroupWithUsers;
|
||||||
users: User[];
|
users: User[];
|
||||||
entity: EntityWithRoles
|
entity: EntityWithRoles;
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Home({ user, group, users, entity }: Props) {
|
export default function Home({ user, group, users, entity }: Props) {
|
||||||
const [isAdding, setIsAdding] = useState(false);
|
const [isAdding, setIsAdding] = useState(false);
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [selectedUsers, setSelectedUsers] = useState<string[]>([]);
|
const [selectedUsers, setSelectedUsers] = useState<string[]>([]);
|
||||||
|
|
||||||
const canAddParticipants = useEntityPermission(user, entity, "add_to_classroom")
|
const canAddParticipants = useEntityPermission(
|
||||||
const canRemoveParticipants = useEntityPermission(user, entity, "remove_from_classroom")
|
user,
|
||||||
const canRenameClassroom = useEntityPermission(user, entity, "rename_classrooms")
|
entity,
|
||||||
const canDeleteClassroom = useEntityPermission(user, entity, "delete_classroom")
|
"add_to_classroom"
|
||||||
|
);
|
||||||
|
const canRemoveParticipants = useEntityPermission(
|
||||||
|
user,
|
||||||
|
entity,
|
||||||
|
"remove_from_classroom"
|
||||||
|
);
|
||||||
|
const canRenameClassroom = useEntityPermission(
|
||||||
|
user,
|
||||||
|
entity,
|
||||||
|
"rename_classrooms"
|
||||||
|
);
|
||||||
|
const canDeleteClassroom = useEntityPermission(
|
||||||
|
user,
|
||||||
|
entity,
|
||||||
|
"delete_classroom"
|
||||||
|
);
|
||||||
|
|
||||||
const nonParticipantUsers = useMemo(
|
const nonParticipantUsers = useMemo(
|
||||||
() => users.filter((x) => ![...group.participants.map((g) => g.id), group.admin.id, user.id].includes(x.id)),
|
() =>
|
||||||
[users, group.participants, group.admin.id, user.id],
|
users.filter(
|
||||||
);
|
(x) =>
|
||||||
|
![
|
||||||
|
...group.participants.map((g) => g.id),
|
||||||
|
group.admin.id,
|
||||||
|
user.id,
|
||||||
|
].includes(x.id)
|
||||||
|
),
|
||||||
|
[users, group.participants, group.admin.id, user.id]
|
||||||
|
);
|
||||||
|
|
||||||
const { rows, renderSearch } = useListSearch<User>(
|
const { rows, renderSearch } = useListSearch<User>(
|
||||||
[["name"], ["corporateInformation", "companyInformation", "name"]],
|
[["name"], ["corporateInformation", "companyInformation", "name"]],
|
||||||
isAdding ? nonParticipantUsers : group.participants,
|
isAdding ? nonParticipantUsers : group.participants
|
||||||
);
|
);
|
||||||
const { items, renderMinimal } = usePagination<User>(rows, 20);
|
const { items, renderMinimal } = usePagination<User>(rows, 20);
|
||||||
|
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
const toggleUser = (u: User) => setSelectedUsers((prev) => (prev.includes(u.id) ? prev.filter((p) => p !== u.id) : [...prev, u.id]));
|
const toggleUser = (u: User) =>
|
||||||
|
setSelectedUsers((prev) =>
|
||||||
|
prev.includes(u.id) ? prev.filter((p) => p !== u.id) : [...prev, u.id]
|
||||||
|
);
|
||||||
|
|
||||||
const removeParticipants = () => {
|
const toggleAllUsersInList = () =>
|
||||||
if (selectedUsers.length === 0) return;
|
setSelectedUsers((prev) =>
|
||||||
if (!canRemoveParticipants) return;
|
prev.length === rows.length
|
||||||
if (!confirm(`Are you sure you want to remove ${selectedUsers.length} participant${selectedUsers.length === 1 ? "" : "s"} from this group?`))
|
? []
|
||||||
return;
|
: [
|
||||||
|
...prev,
|
||||||
|
...items.reduce((acc, i) => {
|
||||||
|
if (!prev.find((item) => item === i.id)) {
|
||||||
|
(acc as string[]).push(i.id);
|
||||||
|
}
|
||||||
|
return acc;
|
||||||
|
}, [] as string[]),
|
||||||
|
]
|
||||||
|
);
|
||||||
|
|
||||||
setIsLoading(true);
|
const removeParticipants = () => {
|
||||||
|
if (selectedUsers.length === 0) return;
|
||||||
|
if (!canRemoveParticipants) return;
|
||||||
|
if (
|
||||||
|
!confirm(
|
||||||
|
`Are you sure you want to remove ${selectedUsers.length} participant${
|
||||||
|
selectedUsers.length === 1 ? "" : "s"
|
||||||
|
} from this group?`
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return;
|
||||||
|
|
||||||
axios
|
setIsLoading(true);
|
||||||
.patch(`/api/groups/${group.id}`, { participants: group.participants.map((x) => x.id).filter((x) => !selectedUsers.includes(x)) })
|
|
||||||
.then(() => {
|
|
||||||
toast.success("The group has been updated successfully!");
|
|
||||||
router.replace(router.asPath);
|
|
||||||
})
|
|
||||||
.catch((e) => {
|
|
||||||
console.error(e);
|
|
||||||
toast.error("Something went wrong!");
|
|
||||||
})
|
|
||||||
.finally(() => setIsLoading(false));
|
|
||||||
};
|
|
||||||
|
|
||||||
const addParticipants = () => {
|
axios
|
||||||
if (selectedUsers.length === 0) return;
|
.patch(`/api/groups/${group.id}`, {
|
||||||
if (!canAddParticipants || !isAdding) return;
|
participants: group.participants
|
||||||
if (!confirm(`Are you sure you want to add ${selectedUsers.length} participant${selectedUsers.length === 1 ? "" : "s"} to this group?`))
|
.map((x) => x.id)
|
||||||
return;
|
.filter((x) => !selectedUsers.includes(x)),
|
||||||
|
})
|
||||||
|
.then(() => {
|
||||||
|
toast.success("The group has been updated successfully!");
|
||||||
|
router.replace(router.asPath);
|
||||||
|
})
|
||||||
|
.catch((e) => {
|
||||||
|
console.error(e);
|
||||||
|
toast.error("Something went wrong!");
|
||||||
|
})
|
||||||
|
.finally(() => setIsLoading(false));
|
||||||
|
};
|
||||||
|
|
||||||
setIsLoading(true);
|
const addParticipants = () => {
|
||||||
|
if (selectedUsers.length === 0) return;
|
||||||
|
if (!canAddParticipants || !isAdding) return;
|
||||||
|
if (
|
||||||
|
!confirm(
|
||||||
|
`Are you sure you want to add ${selectedUsers.length} participant${
|
||||||
|
selectedUsers.length === 1 ? "" : "s"
|
||||||
|
} to this group?`
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return;
|
||||||
|
|
||||||
axios
|
setIsLoading(true);
|
||||||
.patch(`/api/groups/${group.id}`, { participants: [...group.participants.map((x) => x.id), ...selectedUsers] })
|
|
||||||
.then(() => {
|
|
||||||
toast.success("The group has been updated successfully!");
|
|
||||||
router.replace(router.asPath);
|
|
||||||
})
|
|
||||||
.catch((e) => {
|
|
||||||
console.error(e);
|
|
||||||
toast.error("Something went wrong!");
|
|
||||||
})
|
|
||||||
.finally(() => setIsLoading(false));
|
|
||||||
};
|
|
||||||
|
|
||||||
const renameGroup = () => {
|
axios
|
||||||
if (!canRenameClassroom) return;
|
.patch(`/api/groups/${group.id}`, {
|
||||||
|
participants: [
|
||||||
|
...group.participants.map((x) => x.id),
|
||||||
|
...selectedUsers,
|
||||||
|
],
|
||||||
|
})
|
||||||
|
.then(() => {
|
||||||
|
toast.success("The group has been updated successfully!");
|
||||||
|
router.replace(router.asPath);
|
||||||
|
})
|
||||||
|
.catch((e) => {
|
||||||
|
console.error(e);
|
||||||
|
toast.error("Something went wrong!");
|
||||||
|
})
|
||||||
|
.finally(() => setIsLoading(false));
|
||||||
|
};
|
||||||
|
|
||||||
const name = prompt("Rename this classroom:", group.name);
|
const renameGroup = () => {
|
||||||
if (!name) return;
|
if (!canRenameClassroom) return;
|
||||||
|
|
||||||
setIsLoading(true);
|
const name = prompt("Rename this classroom:", group.name);
|
||||||
axios
|
if (!name) return;
|
||||||
.patch(`/api/groups/${group.id}`, { name })
|
|
||||||
.then(() => {
|
|
||||||
toast.success("The classroom has been updated successfully!");
|
|
||||||
router.replace(router.asPath);
|
|
||||||
})
|
|
||||||
.catch((e) => {
|
|
||||||
console.error(e);
|
|
||||||
toast.error("Something went wrong!");
|
|
||||||
})
|
|
||||||
.finally(() => setIsLoading(false));
|
|
||||||
};
|
|
||||||
|
|
||||||
const deleteGroup = () => {
|
setIsLoading(true);
|
||||||
if (!canDeleteClassroom) return;
|
axios
|
||||||
if (!confirm("Are you sure you want to delete this classroom?")) return;
|
.patch(`/api/groups/${group.id}`, { name })
|
||||||
|
.then(() => {
|
||||||
|
toast.success("The classroom has been updated successfully!");
|
||||||
|
router.replace(router.asPath);
|
||||||
|
})
|
||||||
|
.catch((e) => {
|
||||||
|
console.error(e);
|
||||||
|
toast.error("Something went wrong!");
|
||||||
|
})
|
||||||
|
.finally(() => setIsLoading(false));
|
||||||
|
};
|
||||||
|
|
||||||
setIsLoading(true);
|
const deleteGroup = () => {
|
||||||
|
if (!canDeleteClassroom) return;
|
||||||
|
if (!confirm("Are you sure you want to delete this classroom?")) return;
|
||||||
|
|
||||||
axios
|
setIsLoading(true);
|
||||||
.delete(`/api/groups/${group.id}`)
|
|
||||||
.then(() => {
|
|
||||||
toast.success("This classroom has been successfully deleted!");
|
|
||||||
router.replace("/classrooms");
|
|
||||||
})
|
|
||||||
.catch((e) => {
|
|
||||||
console.error(e);
|
|
||||||
toast.error("Something went wrong!");
|
|
||||||
})
|
|
||||||
.finally(() => setIsLoading(false));
|
|
||||||
};
|
|
||||||
|
|
||||||
useEffect(() => setSelectedUsers([]), [isAdding]);
|
axios
|
||||||
|
.delete(`/api/groups/${group.id}`)
|
||||||
|
.then(() => {
|
||||||
|
toast.success("This classroom has been successfully deleted!");
|
||||||
|
router.replace("/classrooms");
|
||||||
|
})
|
||||||
|
.catch((e) => {
|
||||||
|
console.error(e);
|
||||||
|
toast.error("Something went wrong!");
|
||||||
|
})
|
||||||
|
.finally(() => setIsLoading(false));
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
useEffect(() => setSelectedUsers([]), [isAdding]);
|
||||||
<>
|
|
||||||
<Head>
|
|
||||||
<title>{group.name} | EnCoach</title>
|
|
||||||
<meta
|
|
||||||
name="description"
|
|
||||||
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
|
||||||
/>
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
|
||||||
<link rel="icon" href="/favicon.ico" />
|
|
||||||
</Head>
|
|
||||||
<ToastContainer />
|
|
||||||
{user && (
|
|
||||||
<Layout user={user}>
|
|
||||||
<section className="flex flex-col gap-0">
|
|
||||||
<div className="flex items-center justify-between">
|
|
||||||
<div className="flex flex-col gap-3">
|
|
||||||
<div className="flex items-center gap-2">
|
|
||||||
<Link
|
|
||||||
href="/classrooms"
|
|
||||||
className="text-mti-purple hover:text-mti-purple-dark transition ease-in-out duration-300 text-xl">
|
|
||||||
<BsChevronLeft />
|
|
||||||
</Link>
|
|
||||||
<h2 className="font-bold text-2xl">{group.name}</h2>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
{!isAdding && (
|
return (
|
||||||
<div className="flex items-center gap-2">
|
<>
|
||||||
<button
|
<Head>
|
||||||
onClick={renameGroup}
|
<title>{group.name} | EnCoach</title>
|
||||||
disabled={isLoading || !canRenameClassroom}
|
<meta
|
||||||
className="flex items-center gap-1 px-2 py-2 border rounded-full hover:bg-neutral-100 disabled:hover:bg-transparent disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300">
|
name="description"
|
||||||
<BsTag />
|
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
||||||
<span className="text-xs">Rename Classroom</span>
|
/>
|
||||||
</button>
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
<button
|
<link rel="icon" href="/favicon.ico" />
|
||||||
onClick={deleteGroup}
|
</Head>
|
||||||
disabled={isLoading || !canDeleteClassroom}
|
<ToastContainer />
|
||||||
className="flex items-center gap-1 px-2 py-2 border border-mti-rose rounded-full bg-mti-rose-light text-white hover:bg-mti-rose-dark disabled:hover:bg-mti-rose-light disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300">
|
{user && (
|
||||||
<BsTrash />
|
<>
|
||||||
<span className="text-xs">Delete Classroom</span>
|
<section className="flex flex-col gap-0">
|
||||||
</button>
|
<div className="flex items-center justify-between">
|
||||||
</div>
|
<div className="flex flex-col gap-3">
|
||||||
)}
|
<div className="flex items-center gap-2">
|
||||||
</div>
|
<Link
|
||||||
<div className="flex flex-col gap-2">
|
href="/classrooms"
|
||||||
<span className="flex items-center gap-2">
|
className="text-mti-purple hover:text-mti-purple-dark transition ease-in-out duration-300 text-xl"
|
||||||
<BsBuilding className="text-xl" /> {entity.label}
|
>
|
||||||
</span>
|
<BsChevronLeft />
|
||||||
<span className="flex items-center gap-2">
|
</Link>
|
||||||
<BsFillPersonVcardFill className="text-xl" /> {getUserName(group.admin)}
|
<h2 className="font-bold text-2xl">{group.name}</h2>
|
||||||
</span>
|
</div>
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<Divider />
|
|
||||||
<div className="flex items-center justify-between mb-4">
|
|
||||||
<span className="font-semibold text-xl">Participants</span>
|
|
||||||
{!isAdding && (
|
|
||||||
<div className="flex items-center gap-2">
|
|
||||||
<button
|
|
||||||
onClick={() => setIsAdding(true)}
|
|
||||||
disabled={isLoading || !canAddParticipants}
|
|
||||||
className="flex items-center gap-1 px-2 py-2 border rounded-full hover:bg-neutral-100 disabled:hover:bg-transparent disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300">
|
|
||||||
<BsPlus />
|
|
||||||
<span className="text-xs">Add Participants</span>
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
onClick={removeParticipants}
|
|
||||||
disabled={selectedUsers.length === 0 || isLoading || !canRemoveParticipants}
|
|
||||||
className="flex items-center gap-1 px-2 py-2 border border-mti-rose rounded-full bg-mti-rose-light text-white hover:bg-mti-rose-dark disabled:hover:bg-mti-rose-light disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300">
|
|
||||||
<BsTrash />
|
|
||||||
<span className="text-xs">Remove Participants</span>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
{isAdding && (
|
|
||||||
<div className="flex items-center gap-2">
|
|
||||||
<button
|
|
||||||
onClick={() => setIsAdding(false)}
|
|
||||||
disabled={isLoading}
|
|
||||||
className="flex items-center gap-1 px-2 py-2 border rounded-full border-mti-rose bg-mti-rose-light text-white hover:bg-mti-rose-dark disabled:hover:bg-mti-rose-light disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300">
|
|
||||||
<BsX />
|
|
||||||
<span className="text-xs">Discard Selection</span>
|
|
||||||
</button>
|
|
||||||
<button
|
|
||||||
onClick={addParticipants}
|
|
||||||
disabled={selectedUsers.length === 0 || isLoading || !canAddParticipants}
|
|
||||||
className="flex items-center gap-1 px-2 py-2 border rounded-full border-mti-green bg-mti-green-light text-white hover:bg-mti-green-dark disabled:hover:bg-mti-green-light disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300">
|
|
||||||
<BsPlus />
|
|
||||||
<span className="text-xs">Add Participants</span>
|
|
||||||
</button>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
<div className="w-full flex items-center gap-4">
|
|
||||||
{renderSearch()}
|
|
||||||
{renderMinimal()}
|
|
||||||
</div>
|
|
||||||
<div className="flex items-center gap-2 mt-4">
|
|
||||||
{['student', 'teacher', 'corporate'].map((type) => (
|
|
||||||
<button
|
|
||||||
key={type}
|
|
||||||
onClick={() => {
|
|
||||||
const typeUsers = mapBy(filterBy(isAdding ? nonParticipantUsers : group.participants, 'type', type), 'id')
|
|
||||||
if (typeUsers.every((u) => selectedUsers.includes(u))) {
|
|
||||||
setSelectedUsers((prev) => prev.filter((a) => !typeUsers.includes(a)));
|
|
||||||
} else {
|
|
||||||
setSelectedUsers((prev) => [...prev.filter((a) => !typeUsers.includes(a)), ...typeUsers]);
|
|
||||||
}
|
|
||||||
}}
|
|
||||||
disabled={filterBy(isAdding ? nonParticipantUsers : group.participants, 'type', type).length === 0}
|
|
||||||
className={clsx(
|
|
||||||
"bg-mti-purple-ultralight text-mti-purple px-4 py-2 rounded-full hover:text-white hover:bg-mti-purple-light",
|
|
||||||
"transition duration-300 ease-in-out",
|
|
||||||
"disabled:grayscale disabled:hover:bg-mti-purple-ultralight disabled:hover:text-mti-purple disabled:cursor-not-allowed",
|
|
||||||
filterBy(isAdding ? nonParticipantUsers : group.participants, 'type', type).length > 0 &&
|
|
||||||
filterBy(isAdding ? nonParticipantUsers : group.participants, 'type', type).every((u) => selectedUsers.includes(u.id)) &&
|
|
||||||
"!bg-mti-purple-light !text-white",
|
|
||||||
)}>
|
|
||||||
{capitalize(type)}
|
|
||||||
</button>
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
<section className="w-full h-full grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4">
|
{!isAdding && (
|
||||||
{items.map((u) => (
|
<div className="flex items-center gap-2">
|
||||||
<button
|
<button
|
||||||
onClick={() => toggleUser(u)}
|
onClick={renameGroup}
|
||||||
disabled={isAdding ? !canAddParticipants : !canRemoveParticipants}
|
disabled={isLoading || !canRenameClassroom}
|
||||||
key={u.id}
|
className="flex items-center gap-1 px-2 py-2 border rounded-full hover:bg-neutral-100 disabled:hover:bg-transparent disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300"
|
||||||
className={clsx(
|
>
|
||||||
"p-4 pr-6 h-48 relative border rounded-xl flex flex-col gap-3 justify-between text-left cursor-pointer",
|
<BsTag />
|
||||||
"hover:border-mti-purple transition ease-in-out duration-300",
|
<span className="text-xs">Rename Classroom</span>
|
||||||
selectedUsers.includes(u.id) && "border-mti-purple",
|
</button>
|
||||||
)}>
|
<button
|
||||||
<div className="flex items-center gap-2">
|
onClick={deleteGroup}
|
||||||
<div className="min-w-[3rem] min-h-[3rem] w-12 h-12 border flex items-center justify-center overflow-hidden rounded-full">
|
disabled={isLoading || !canDeleteClassroom}
|
||||||
<img src={u.profilePicture} alt={u.name} />
|
className="flex items-center gap-1 px-2 py-2 border border-mti-rose rounded-full bg-mti-rose-light text-white hover:bg-mti-rose-dark disabled:hover:bg-mti-rose-light disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300"
|
||||||
</div>
|
>
|
||||||
<div className="flex flex-col">
|
<BsTrash />
|
||||||
<span className="font-semibold">{getUserName(u)}</span>
|
<span className="text-xs">Delete Classroom</span>
|
||||||
<span className="opacity-80 text-sm">{USER_TYPE_LABELS[u.type]}</span>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="flex flex-col gap-2">
|
||||||
|
<span className="flex items-center gap-2">
|
||||||
|
<BsBuilding className="text-xl" /> {entity.label}
|
||||||
|
</span>
|
||||||
|
<span className="flex items-center gap-2">
|
||||||
|
<BsFillPersonVcardFill className="text-xl" />{" "}
|
||||||
|
{getUserName(group.admin)}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<Divider />
|
||||||
|
<div className="flex items-center justify-between mb-4">
|
||||||
|
<span className="font-semibold text-xl">Participants</span>
|
||||||
|
{!isAdding && (
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<button
|
||||||
|
onClick={() => setIsAdding(true)}
|
||||||
|
disabled={isLoading || !canAddParticipants}
|
||||||
|
className="flex items-center gap-1 px-2 py-2 border rounded-full hover:bg-neutral-100 disabled:hover:bg-transparent disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300"
|
||||||
|
>
|
||||||
|
<BsPlus />
|
||||||
|
<span className="text-xs">Add Participants</span>
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={removeParticipants}
|
||||||
|
disabled={
|
||||||
|
selectedUsers.length === 0 ||
|
||||||
|
isLoading ||
|
||||||
|
!canRemoveParticipants
|
||||||
|
}
|
||||||
|
className="flex items-center gap-1 px-2 py-2 border border-mti-rose rounded-full bg-mti-rose-light text-white hover:bg-mti-rose-dark disabled:hover:bg-mti-rose-light disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300"
|
||||||
|
>
|
||||||
|
<BsTrash />
|
||||||
|
<span className="text-xs">Remove Participants</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
{isAdding && (
|
||||||
|
<div className="flex items-center gap-2">
|
||||||
|
<button
|
||||||
|
onClick={() => setIsAdding(false)}
|
||||||
|
disabled={isLoading}
|
||||||
|
className="flex items-center gap-1 px-2 py-2 border rounded-full border-mti-rose bg-mti-rose-light text-white hover:bg-mti-rose-dark disabled:hover:bg-mti-rose-light disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300"
|
||||||
|
>
|
||||||
|
<BsX />
|
||||||
|
<span className="text-xs">Discard Selection</span>
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
onClick={addParticipants}
|
||||||
|
disabled={
|
||||||
|
selectedUsers.length === 0 ||
|
||||||
|
isLoading ||
|
||||||
|
!canAddParticipants
|
||||||
|
}
|
||||||
|
className="flex items-center gap-1 px-2 py-2 border rounded-full border-mti-green bg-mti-green-light text-white hover:bg-mti-green-dark disabled:hover:bg-mti-green-light disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300"
|
||||||
|
>
|
||||||
|
<BsPlus />
|
||||||
|
<span className="text-xs">Add Participants</span>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<div className="w-full flex items-center gap-4">
|
||||||
|
{renderSearch()}
|
||||||
|
{renderMinimal()}
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center gap-2 mt-4">
|
||||||
|
{["student", "teacher", "corporate"].map((type) => (
|
||||||
|
<button
|
||||||
|
key={type}
|
||||||
|
onClick={() => {
|
||||||
|
const typeUsers = mapBy(
|
||||||
|
filterBy(
|
||||||
|
isAdding ? nonParticipantUsers : group.participants,
|
||||||
|
"type",
|
||||||
|
type
|
||||||
|
),
|
||||||
|
"id"
|
||||||
|
);
|
||||||
|
if (typeUsers.every((u) => selectedUsers.includes(u))) {
|
||||||
|
setSelectedUsers((prev) =>
|
||||||
|
prev.filter((a) => !typeUsers.includes(a))
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
setSelectedUsers((prev) => [
|
||||||
|
...prev.filter((a) => !typeUsers.includes(a)),
|
||||||
|
...typeUsers,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
disabled={
|
||||||
|
filterBy(
|
||||||
|
isAdding ? nonParticipantUsers : group.participants,
|
||||||
|
"type",
|
||||||
|
type
|
||||||
|
).length === 0
|
||||||
|
}
|
||||||
|
className={clsx(
|
||||||
|
"bg-mti-purple-ultralight text-mti-purple px-4 py-2 rounded-full hover:text-white hover:bg-mti-purple-light",
|
||||||
|
"transition duration-300 ease-in-out",
|
||||||
|
"disabled:grayscale disabled:hover:bg-mti-purple-ultralight disabled:hover:text-mti-purple disabled:cursor-not-allowed",
|
||||||
|
filterBy(
|
||||||
|
isAdding ? nonParticipantUsers : group.participants,
|
||||||
|
"type",
|
||||||
|
type
|
||||||
|
).length > 0 &&
|
||||||
|
filterBy(
|
||||||
|
isAdding ? nonParticipantUsers : group.participants,
|
||||||
|
"type",
|
||||||
|
type
|
||||||
|
).every((u) => selectedUsers.includes(u.id)) &&
|
||||||
|
"!bg-mti-purple-light !text-white"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{capitalize(type)}
|
||||||
|
</button>
|
||||||
|
))}
|
||||||
|
<button
|
||||||
|
onClick={() => {
|
||||||
|
toggleAllUsersInList();
|
||||||
|
}}
|
||||||
|
disabled={rows.length === 0}
|
||||||
|
className={clsx(
|
||||||
|
"bg-mti-purple-ultralight text-mti-purple px-4 py-2 rounded-full hover:text-white hover:bg-mti-purple-light",
|
||||||
|
"transition duration-300 ease-in-out",
|
||||||
|
"disabled:grayscale disabled:hover:bg-mti-purple-ultralight disabled:hover:text-mti-purple disabled:cursor-not-allowed",
|
||||||
|
(isAdding ? nonParticipantUsers : group.participants)
|
||||||
|
.length === selectedUsers.length &&
|
||||||
|
"!bg-mti-purple-light !text-white"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{"De/Select All"}
|
||||||
|
</button>
|
||||||
|
<span className="opacity-80">
|
||||||
|
{selectedUsers.length} selected
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
<div className="flex flex-col gap-1">
|
<section className="w-full h-full grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4">
|
||||||
<span className="flex items-center gap-2">
|
{items.map((u) => (
|
||||||
<Tooltip tooltip="E-mail address">
|
<button
|
||||||
<BsEnvelopeFill />
|
onClick={() => toggleUser(u)}
|
||||||
</Tooltip>
|
disabled={
|
||||||
{u.email}
|
isAdding ? !canAddParticipants : !canRemoveParticipants
|
||||||
</span>
|
}
|
||||||
<span className="flex items-center gap-2">
|
key={u.id}
|
||||||
<Tooltip tooltip="Expiration Date">
|
className={clsx(
|
||||||
<BsStopwatchFill />
|
"p-4 pr-6 h-48 relative border rounded-xl flex flex-col gap-3 justify-between text-left cursor-pointer",
|
||||||
</Tooltip>
|
"hover:border-mti-purple transition ease-in-out duration-300",
|
||||||
{u.subscriptionExpirationDate ? moment(u.subscriptionExpirationDate).format("DD/MM/YYYY") : "Unlimited"}
|
selectedUsers.includes(u.id) && "border-mti-purple"
|
||||||
</span>
|
)}
|
||||||
<span className="flex items-center gap-2">
|
>
|
||||||
<Tooltip tooltip="Last Login">
|
<div className="flex items-center gap-2">
|
||||||
<BsClockFill />
|
<div className="min-w-[3rem] min-h-[3rem] w-12 h-12 border flex items-center justify-center overflow-hidden rounded-full">
|
||||||
</Tooltip>
|
<img src={u.profilePicture} alt={u.name} />
|
||||||
{u.lastLogin ? moment(u.lastLogin).format("DD/MM/YYYY - HH:mm") : "N/A"}
|
</div>
|
||||||
</span>
|
<div className="flex flex-col">
|
||||||
</div>
|
<span className="font-semibold">{getUserName(u)}</span>
|
||||||
</button>
|
<span className="opacity-80 text-sm">
|
||||||
))}
|
{USER_TYPE_LABELS[u.type]}
|
||||||
</section>
|
</span>
|
||||||
</Layout>
|
</div>
|
||||||
)}
|
</div>
|
||||||
</>
|
|
||||||
);
|
<div className="flex flex-col gap-1">
|
||||||
|
<span className="flex items-center gap-2">
|
||||||
|
<Tooltip tooltip="E-mail address">
|
||||||
|
<BsEnvelopeFill />
|
||||||
|
</Tooltip>
|
||||||
|
{u.email}
|
||||||
|
</span>
|
||||||
|
<span className="flex items-center gap-2">
|
||||||
|
<Tooltip tooltip="Expiration Date">
|
||||||
|
<BsStopwatchFill />
|
||||||
|
</Tooltip>
|
||||||
|
{u.subscriptionExpirationDate
|
||||||
|
? moment(u.subscriptionExpirationDate).format(
|
||||||
|
"DD/MM/YYYY"
|
||||||
|
)
|
||||||
|
: "Unlimited"}
|
||||||
|
</span>
|
||||||
|
<span className="flex items-center gap-2">
|
||||||
|
<Tooltip tooltip="Last Login">
|
||||||
|
<BsClockFill />
|
||||||
|
</Tooltip>
|
||||||
|
{u.lastLogin
|
||||||
|
? moment(u.lastLogin).format("DD/MM/YYYY - HH:mm")
|
||||||
|
: "N/A"}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</button>
|
||||||
|
))}
|
||||||
|
</section>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
</>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,223 +1,309 @@
|
|||||||
/* eslint-disable @next/next/no-img-element */
|
/* eslint-disable @next/next/no-img-element */
|
||||||
import Layout from "@/components/High/Layout";
|
|
||||||
import Input from "@/components/Low/Input";
|
import Input from "@/components/Low/Input";
|
||||||
import Select from "@/components/Low/Select";
|
import Select from "@/components/Low/Select";
|
||||||
import Tooltip from "@/components/Low/Tooltip";
|
import Tooltip from "@/components/Low/Tooltip";
|
||||||
import {useListSearch} from "@/hooks/useListSearch";
|
import { useListSearch } from "@/hooks/useListSearch";
|
||||||
import usePagination from "@/hooks/usePagination";
|
import usePagination from "@/hooks/usePagination";
|
||||||
import {Entity, EntityWithRoles} from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
import {User} from "@/interfaces/user";
|
import { User } from "@/interfaces/user";
|
||||||
import {sessionOptions} from "@/lib/session";
|
import { sessionOptions } from "@/lib/session";
|
||||||
import {USER_TYPE_LABELS} from "@/resources/user";
|
import { USER_TYPE_LABELS } from "@/resources/user";
|
||||||
import {filterBy, mapBy, redirect, serialize} from "@/utils";
|
import { filterBy, mapBy, redirect, serialize } from "@/utils";
|
||||||
import {getEntities, getEntitiesWithRoles} from "@/utils/entities.be";
|
import { getEntitiesWithRoles } from "@/utils/entities.be";
|
||||||
import {shouldRedirectHome} from "@/utils/navigation.disabled";
|
import { shouldRedirectHome } from "@/utils/navigation.disabled";
|
||||||
import {getUserName, isAdmin} from "@/utils/users";
|
import { getUserName, isAdmin } from "@/utils/users";
|
||||||
import {getEntitiesUsers, getLinkedUsers} from "@/utils/users.be";
|
import { getEntitiesUsers } from "@/utils/users.be";
|
||||||
import axios from "axios";
|
import axios from "axios";
|
||||||
import clsx from "clsx";
|
import clsx from "clsx";
|
||||||
import {withIronSessionSsr} from "iron-session/next";
|
import { withIronSessionSsr } from "iron-session/next";
|
||||||
import moment from "moment";
|
import moment from "moment";
|
||||||
import Head from "next/head";
|
import Head from "next/head";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import {useRouter} from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import {Divider} from "primereact/divider";
|
import { Divider } from "primereact/divider";
|
||||||
import {useEffect, useMemo, useState} from "react";
|
import { useEffect, useMemo, useState } from "react";
|
||||||
import {BsCheck, BsChevronLeft, BsClockFill, BsEnvelopeFill, BsStopwatchFill} from "react-icons/bs";
|
import {
|
||||||
import {toast, ToastContainer} from "react-toastify";
|
BsCheck,
|
||||||
|
BsChevronLeft,
|
||||||
|
BsClockFill,
|
||||||
|
BsEnvelopeFill,
|
||||||
|
BsStopwatchFill,
|
||||||
|
} from "react-icons/bs";
|
||||||
|
import { toast, ToastContainer } from "react-toastify";
|
||||||
import { requestUser } from "@/utils/api";
|
import { requestUser } from "@/utils/api";
|
||||||
import { findAllowedEntities } from "@/utils/permissions";
|
import { findAllowedEntities } from "@/utils/permissions";
|
||||||
import { capitalize } from "lodash";
|
import { capitalize } from "lodash";
|
||||||
|
|
||||||
export const getServerSideProps = withIronSessionSsr(async ({req, res}) => {
|
export const getServerSideProps = withIronSessionSsr(async ({ req, res }) => {
|
||||||
const user = await requestUser(req, res)
|
const user = await requestUser(req, res);
|
||||||
if (!user) return redirect("/login")
|
if (!user) return redirect("/login");
|
||||||
|
|
||||||
if (shouldRedirectHome(user)) return redirect("/")
|
if (shouldRedirectHome(user)) return redirect("/");
|
||||||
|
|
||||||
const entities = await getEntitiesWithRoles(isAdmin(user) ? undefined : mapBy(user.entities, "id"));
|
const entities = await getEntitiesWithRoles(
|
||||||
const users = await getEntitiesUsers(mapBy(entities, 'id'))
|
isAdmin(user) ? undefined : mapBy(user.entities, "id")
|
||||||
const allowedEntities = findAllowedEntities(user, entities, "create_classroom")
|
);
|
||||||
|
const users = await getEntitiesUsers(
|
||||||
|
mapBy(entities, "id"),
|
||||||
|
{
|
||||||
|
id: { $ne: user.id },
|
||||||
|
},
|
||||||
|
0,
|
||||||
|
{
|
||||||
|
_id: 0,
|
||||||
|
id: 1,
|
||||||
|
name: 1,
|
||||||
|
email: 1,
|
||||||
|
profilePicture: 1,
|
||||||
|
type: 1,
|
||||||
|
corporateInformation: 1,
|
||||||
|
lastLogin: 1,
|
||||||
|
subscriptionExpirationDate: 1,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
const allowedEntities = findAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"create_classroom"
|
||||||
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
props: serialize({user, entities: allowedEntities, users: users.filter((x) => x.id !== user.id)}),
|
props: serialize({
|
||||||
};
|
user,
|
||||||
|
entities: allowedEntities,
|
||||||
|
users: users,
|
||||||
|
}),
|
||||||
|
};
|
||||||
}, sessionOptions);
|
}, sessionOptions);
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
user: User;
|
user: User;
|
||||||
users: User[];
|
users: User[];
|
||||||
entities: EntityWithRoles[];
|
entities: EntityWithRoles[];
|
||||||
}
|
}
|
||||||
|
|
||||||
export default function Home({user, users, entities}: Props) {
|
export default function Home({ user, users, entities }: Props) {
|
||||||
const [isLoading, setIsLoading] = useState(false);
|
const [isLoading, setIsLoading] = useState(false);
|
||||||
const [selectedUsers, setSelectedUsers] = useState<string[]>([]);
|
const [selectedUsers, setSelectedUsers] = useState<string[]>([]);
|
||||||
const [name, setName] = useState("");
|
const [name, setName] = useState("");
|
||||||
const [entity, setEntity] = useState<string | undefined>(entities[0]?.id);
|
const [entity, setEntity] = useState<string | undefined>(entities[0]?.id);
|
||||||
|
|
||||||
const entityUsers = useMemo(() => !entity ? users : users.filter(u => mapBy(u.entities, 'id').includes(entity)), [entity, users])
|
const entityUsers = useMemo(
|
||||||
|
() =>
|
||||||
|
!entity
|
||||||
|
? users
|
||||||
|
: users.filter((u) => mapBy(u.entities, "id").includes(entity)),
|
||||||
|
[entity, users]
|
||||||
|
);
|
||||||
|
|
||||||
const {rows, renderSearch} = useListSearch<User>(
|
const { rows, renderSearch } = useListSearch<User>(
|
||||||
[["name"], ["type"], ["corporateInformation", "companyInformation", "name"]], entityUsers
|
[
|
||||||
);
|
["name"],
|
||||||
|
["type"],
|
||||||
|
["corporateInformation", "companyInformation", "name"],
|
||||||
|
],
|
||||||
|
entityUsers
|
||||||
|
);
|
||||||
|
|
||||||
const {items, renderMinimal} = usePagination<User>(rows, 16);
|
const { items, renderMinimal } = usePagination<User>(rows, 16);
|
||||||
|
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
useEffect(() => setSelectedUsers([]), [entity])
|
useEffect(() => setSelectedUsers([]), [entity]);
|
||||||
|
|
||||||
const createGroup = () => {
|
const createGroup = () => {
|
||||||
if (!name.trim()) return;
|
if (!name.trim()) return;
|
||||||
if (!entity) return;
|
if (!entity) return;
|
||||||
if (!confirm(`Are you sure you want to create this group with ${selectedUsers.length} participants?`)) return;
|
if (
|
||||||
|
!confirm(
|
||||||
|
`Are you sure you want to create this group with ${selectedUsers.length} participants?`
|
||||||
|
)
|
||||||
|
)
|
||||||
|
return;
|
||||||
|
|
||||||
setIsLoading(true);
|
setIsLoading(true);
|
||||||
|
|
||||||
axios
|
axios
|
||||||
.post<{id: string}>(`/api/groups`, {name, participants: selectedUsers, admin: user.id, entity})
|
.post<{ id: string }>(`/api/groups`, {
|
||||||
.then((result) => {
|
name,
|
||||||
toast.success("Your group has been created successfully!");
|
participants: selectedUsers,
|
||||||
router.replace(`/classrooms/${result.data.id}`);
|
admin: user.id,
|
||||||
})
|
entity,
|
||||||
.catch((e) => {
|
})
|
||||||
console.error(e);
|
.then((result) => {
|
||||||
toast.error("Something went wrong!");
|
toast.success("Your group has been created successfully!");
|
||||||
})
|
router.replace(`/classrooms/${result.data.id}`);
|
||||||
.finally(() => setIsLoading(false));
|
})
|
||||||
};
|
.catch((e) => {
|
||||||
|
console.error(e);
|
||||||
|
toast.error("Something went wrong!");
|
||||||
|
})
|
||||||
|
.finally(() => setIsLoading(false));
|
||||||
|
};
|
||||||
|
|
||||||
const toggleUser = (u: User) => setSelectedUsers((prev) => (prev.includes(u.id) ? prev.filter((p) => p !== u.id) : [...prev, u.id]));
|
const toggleUser = (u: User) =>
|
||||||
|
setSelectedUsers((prev) =>
|
||||||
|
prev.includes(u.id) ? prev.filter((p) => p !== u.id) : [...prev, u.id]
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Head>
|
<Head>
|
||||||
<title>Create Group | EnCoach</title>
|
<title>Create Group | EnCoach</title>
|
||||||
<meta
|
<meta
|
||||||
name="description"
|
name="description"
|
||||||
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
||||||
/>
|
/>
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
<link rel="icon" href="/favicon.ico" />
|
<link rel="icon" href="/favicon.ico" />
|
||||||
</Head>
|
</Head>
|
||||||
<ToastContainer />
|
<ToastContainer />
|
||||||
<Layout user={user}>
|
<>
|
||||||
<section className="flex flex-col gap-0">
|
<section className="flex flex-col gap-0">
|
||||||
<div className="flex gap-3 justify-between">
|
<div className="flex gap-3 justify-between">
|
||||||
<div className="flex items-center gap-2">
|
<div className="flex items-center gap-2">
|
||||||
<Link
|
<Link
|
||||||
href="/classrooms"
|
href="/classrooms"
|
||||||
className="text-mti-purple hover:text-mti-purple-dark transition ease-in-out duration-300 text-xl">
|
className="text-mti-purple hover:text-mti-purple-dark transition ease-in-out duration-300 text-xl"
|
||||||
<BsChevronLeft />
|
>
|
||||||
</Link>
|
<BsChevronLeft />
|
||||||
<h2 className="font-bold text-2xl">Create Classroom</h2>
|
</Link>
|
||||||
</div>
|
<h2 className="font-bold text-2xl">Create Classroom</h2>
|
||||||
<div className="flex items-center gap-4">
|
</div>
|
||||||
<button
|
<div className="flex items-center gap-4">
|
||||||
onClick={createGroup}
|
<button
|
||||||
disabled={!name.trim() || !entity || isLoading}
|
onClick={createGroup}
|
||||||
className="flex items-center gap-1 px-2 py-2 border rounded-full border-mti-green bg-mti-green-light text-white hover:bg-mti-green-dark disabled:hover:bg-mti-green-light disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300">
|
disabled={!name.trim() || !entity || isLoading}
|
||||||
<BsCheck />
|
className="flex items-center gap-1 px-2 py-2 border rounded-full border-mti-green bg-mti-green-light text-white hover:bg-mti-green-dark disabled:hover:bg-mti-green-light disabled:opacity-50 disabled:cursor-not-allowed cursor-pointer transition ease-in-out duration-300"
|
||||||
<span className="text-xs">Create Classroom</span>
|
>
|
||||||
</button>
|
<BsCheck />
|
||||||
</div>
|
<span className="text-xs">Create Classroom</span>
|
||||||
</div>
|
</button>
|
||||||
<Divider />
|
</div>
|
||||||
<div className="grid grid-cols-2 gap-4 place-items-end">
|
</div>
|
||||||
<div className="flex flex-col gap-4 w-full">
|
<Divider />
|
||||||
<span className="font-semibold text-xl">Classroom Name:</span>
|
<div className="grid grid-cols-2 gap-4 place-items-end">
|
||||||
<Input name="name" onChange={setName} type="text" placeholder="Classroom A" />
|
<div className="flex flex-col gap-4 w-full">
|
||||||
</div>
|
<span className="font-semibold text-xl">Classroom Name:</span>
|
||||||
<div className="flex flex-col gap-4 w-full">
|
<Input
|
||||||
<span className="font-semibold text-xl">Entity:</span>
|
name="name"
|
||||||
<Select
|
onChange={setName}
|
||||||
options={entities.map((e) => ({value: e.id, label: e.label}))}
|
type="text"
|
||||||
onChange={(v) => setEntity(v ? v.value! : undefined)}
|
placeholder="Classroom A"
|
||||||
defaultValue={{value: entities[0]?.id, label: entities[0]?.label}}
|
/>
|
||||||
/>
|
</div>
|
||||||
</div>
|
<div className="flex flex-col gap-4 w-full">
|
||||||
</div>
|
<span className="font-semibold text-xl">Entity:</span>
|
||||||
<Divider />
|
<Select
|
||||||
<div className="flex items-center justify-between mb-4">
|
options={entities.map((e) => ({ value: e.id, label: e.label }))}
|
||||||
<span className="font-semibold text-xl">Participants ({selectedUsers.length} selected):</span>
|
onChange={(v) => setEntity(v ? v.value! : undefined)}
|
||||||
</div>
|
defaultValue={{
|
||||||
<div className="w-full flex items-center gap-4">
|
value: entities[0]?.id,
|
||||||
{renderSearch()}
|
label: entities[0]?.label,
|
||||||
{renderMinimal()}
|
}}
|
||||||
</div>
|
/>
|
||||||
<div className="flex items-center gap-2 mt-4">
|
</div>
|
||||||
{['student', 'teacher', 'corporate'].map((type) => (
|
</div>
|
||||||
<button
|
<Divider />
|
||||||
key={type}
|
<div className="flex items-center justify-between mb-4">
|
||||||
onClick={() => {
|
<span className="font-semibold text-xl">
|
||||||
const typeUsers = mapBy(filterBy(entityUsers, 'type', type), 'id')
|
Participants ({selectedUsers.length} selected):
|
||||||
if (typeUsers.every((u) => selectedUsers.includes(u))) {
|
</span>
|
||||||
setSelectedUsers((prev) => prev.filter((a) => !typeUsers.includes(a)));
|
</div>
|
||||||
} else {
|
<div className="w-full flex items-center gap-4">
|
||||||
setSelectedUsers((prev) => [...prev.filter((a) => !typeUsers.includes(a)), ...typeUsers]);
|
{renderSearch()}
|
||||||
}
|
{renderMinimal()}
|
||||||
}}
|
</div>
|
||||||
disabled={filterBy(entityUsers, 'type', type).length === 0}
|
<div className="flex items-center gap-2 mt-4">
|
||||||
className={clsx(
|
{["student", "teacher", "corporate"].map((type) => (
|
||||||
"bg-mti-purple-ultralight text-mti-purple px-4 py-2 rounded-full hover:text-white hover:bg-mti-purple-light",
|
<button
|
||||||
"transition duration-300 ease-in-out",
|
key={type}
|
||||||
"disabled:grayscale disabled:hover:bg-mti-purple-ultralight disabled:hover:text-mti-purple disabled:cursor-not-allowed",
|
onClick={() => {
|
||||||
filterBy(entityUsers, 'type', type).length > 0 &&
|
const typeUsers = mapBy(
|
||||||
filterBy(entityUsers, 'type', type).every((u) => selectedUsers.includes(u.id)) &&
|
filterBy(entityUsers, "type", type),
|
||||||
"!bg-mti-purple-light !text-white",
|
"id"
|
||||||
)}>
|
);
|
||||||
{capitalize(type)}
|
if (typeUsers.every((u) => selectedUsers.includes(u))) {
|
||||||
</button>
|
setSelectedUsers((prev) =>
|
||||||
))}
|
prev.filter((a) => !typeUsers.includes(a))
|
||||||
</div>
|
);
|
||||||
</section>
|
} else {
|
||||||
|
setSelectedUsers((prev) => [
|
||||||
|
...prev.filter((a) => !typeUsers.includes(a)),
|
||||||
|
...typeUsers,
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
disabled={filterBy(entityUsers, "type", type).length === 0}
|
||||||
|
className={clsx(
|
||||||
|
"bg-mti-purple-ultralight text-mti-purple px-4 py-2 rounded-full hover:text-white hover:bg-mti-purple-light",
|
||||||
|
"transition duration-300 ease-in-out",
|
||||||
|
"disabled:grayscale disabled:hover:bg-mti-purple-ultralight disabled:hover:text-mti-purple disabled:cursor-not-allowed",
|
||||||
|
filterBy(entityUsers, "type", type).length > 0 &&
|
||||||
|
filterBy(entityUsers, "type", type).every((u) =>
|
||||||
|
selectedUsers.includes(u.id)
|
||||||
|
) &&
|
||||||
|
"!bg-mti-purple-light !text-white"
|
||||||
|
)}
|
||||||
|
>
|
||||||
|
{capitalize(type)}
|
||||||
|
</button>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
<section className="w-full h-full grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4">
|
<section className="w-full h-full grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4">
|
||||||
{items.map((u) => (
|
{items.map((u) => (
|
||||||
<button
|
<button
|
||||||
onClick={() => toggleUser(u)}
|
onClick={() => toggleUser(u)}
|
||||||
disabled={isLoading}
|
disabled={isLoading}
|
||||||
key={u.id}
|
key={u.id}
|
||||||
className={clsx(
|
className={clsx(
|
||||||
"p-4 pr-6 h-48 relative border rounded-xl flex flex-col gap-3 justify-between text-left cursor-pointer",
|
"p-4 pr-6 h-48 relative border rounded-xl flex flex-col gap-3 justify-between text-left cursor-pointer",
|
||||||
"hover:border-mti-purple transition ease-in-out duration-300",
|
"hover:border-mti-purple transition ease-in-out duration-300",
|
||||||
selectedUsers.includes(u.id) && "border-mti-purple",
|
selectedUsers.includes(u.id) && "border-mti-purple"
|
||||||
)}>
|
)}
|
||||||
<div className="flex items-center gap-2">
|
>
|
||||||
<div className="min-w-[3rem] min-h-[3rem] w-12 h-12 border flex items-center justify-center overflow-hidden rounded-full">
|
<div className="flex items-center gap-2">
|
||||||
<img src={u.profilePicture} alt={u.name} />
|
<div className="min-w-[3rem] min-h-[3rem] w-12 h-12 border flex items-center justify-center overflow-hidden rounded-full">
|
||||||
</div>
|
<img src={u.profilePicture} alt={u.name} />
|
||||||
<div className="flex flex-col">
|
</div>
|
||||||
<span className="font-semibold">{getUserName(u)}</span>
|
<div className="flex flex-col">
|
||||||
<span className="opacity-80 text-sm">{USER_TYPE_LABELS[u.type]}</span>
|
<span className="font-semibold">{getUserName(u)}</span>
|
||||||
</div>
|
<span className="opacity-80 text-sm">
|
||||||
</div>
|
{USER_TYPE_LABELS[u.type]}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div className="flex flex-col gap-1">
|
<div className="flex flex-col gap-1">
|
||||||
<span className="flex items-center gap-2">
|
<span className="flex items-center gap-2">
|
||||||
<Tooltip tooltip="E-mail address">
|
<Tooltip tooltip="E-mail address">
|
||||||
<BsEnvelopeFill />
|
<BsEnvelopeFill />
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
{u.email}
|
{u.email}
|
||||||
</span>
|
</span>
|
||||||
<span className="flex items-center gap-2">
|
<span className="flex items-center gap-2">
|
||||||
<Tooltip tooltip="Expiration Date">
|
<Tooltip tooltip="Expiration Date">
|
||||||
<BsStopwatchFill />
|
<BsStopwatchFill />
|
||||||
</Tooltip>
|
</Tooltip>
|
||||||
{u.subscriptionExpirationDate ? moment(u.subscriptionExpirationDate).format("DD/MM/YYYY") : "Unlimited"}
|
{u.subscriptionExpirationDate
|
||||||
</span>
|
? moment(u.subscriptionExpirationDate).format("DD/MM/YYYY")
|
||||||
<span className="flex items-center gap-2">
|
: "Unlimited"}
|
||||||
<Tooltip tooltip="Last Login">
|
</span>
|
||||||
<BsClockFill />
|
<span className="flex items-center gap-2">
|
||||||
</Tooltip>
|
<Tooltip tooltip="Last Login">
|
||||||
{u.lastLogin ? moment(u.lastLogin).format("DD/MM/YYYY - HH:mm") : "N/A"}
|
<BsClockFill />
|
||||||
</span>
|
</Tooltip>
|
||||||
</div>
|
{u.lastLogin
|
||||||
</button>
|
? moment(u.lastLogin).format("DD/MM/YYYY - HH:mm")
|
||||||
))}
|
: "N/A"}
|
||||||
</section>
|
</span>
|
||||||
</Layout>
|
</div>
|
||||||
</>
|
</button>
|
||||||
);
|
))}
|
||||||
|
</section>
|
||||||
|
</>
|
||||||
|
</>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ import Head from "next/head";
|
|||||||
import { withIronSessionSsr } from "iron-session/next";
|
import { withIronSessionSsr } from "iron-session/next";
|
||||||
import { sessionOptions } from "@/lib/session";
|
import { sessionOptions } from "@/lib/session";
|
||||||
import { ToastContainer } from "react-toastify";
|
import { ToastContainer } from "react-toastify";
|
||||||
import Layout from "@/components/High/Layout";
|
|
||||||
import { GroupWithUsers, User } from "@/interfaces/user";
|
import { GroupWithUsers, User } from "@/interfaces/user";
|
||||||
import { shouldRedirectHome } from "@/utils/navigation.disabled";
|
import { shouldRedirectHome } from "@/utils/navigation.disabled";
|
||||||
import { getUserName, isAdmin } from "@/utils/users";
|
import { getUserName, isAdmin } from "@/utils/users";
|
||||||
@@ -11,13 +10,13 @@ import { convertToUsers, getGroupsForEntities } from "@/utils/groups.be";
|
|||||||
import { getSpecificUsers } from "@/utils/users.be";
|
import { getSpecificUsers } from "@/utils/users.be";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { uniq } from "lodash";
|
import { uniq } from "lodash";
|
||||||
import { BsFillMortarboardFill, BsPlus } from "react-icons/bs";
|
import { BsPlus } from "react-icons/bs";
|
||||||
import CardList from "@/components/High/CardList";
|
import CardList from "@/components/High/CardList";
|
||||||
import Separator from "@/components/Low/Separator";
|
import Separator from "@/components/Low/Separator";
|
||||||
import { findBy, mapBy, redirect, serialize } from "@/utils";
|
import { findBy, mapBy, redirect, serialize } from "@/utils";
|
||||||
import { requestUser } from "@/utils/api";
|
import { requestUser } from "@/utils/api";
|
||||||
import { findAllowedEntities } from "@/utils/permissions";
|
import { findAllowedEntities } from "@/utils/permissions";
|
||||||
import { getEntities, getEntitiesWithRoles } from "@/utils/entities.be";
|
import { getEntitiesWithRoles } from "@/utils/entities.be";
|
||||||
import { useAllowedEntities } from "@/hooks/useEntityPermissions";
|
import { useAllowedEntities } from "@/hooks/useEntityPermissions";
|
||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
import { FaPersonChalkboard } from "react-icons/fa6";
|
import { FaPersonChalkboard } from "react-icons/fa6";
|
||||||
@@ -28,132 +27,182 @@ import StudentClassroomTransfer from "@/components/Imports/StudentClassroomTrans
|
|||||||
import Modal from "@/components/Modal";
|
import Modal from "@/components/Modal";
|
||||||
|
|
||||||
export const getServerSideProps = withIronSessionSsr(async ({ req, res }) => {
|
export const getServerSideProps = withIronSessionSsr(async ({ req, res }) => {
|
||||||
const user = await requestUser(req, res)
|
const user = await requestUser(req, res);
|
||||||
if (!user) return redirect("/login")
|
if (!user) return redirect("/login");
|
||||||
|
|
||||||
if (shouldRedirectHome(user)) return redirect("/")
|
if (shouldRedirectHome(user)) return redirect("/");
|
||||||
|
|
||||||
const entityIDS = mapBy(user.entities, "id");
|
const entityIDS = mapBy(user.entities, "id");
|
||||||
const entities = await getEntitiesWithRoles(isAdmin(user) ? undefined : entityIDS)
|
const entities = await getEntitiesWithRoles(
|
||||||
const allowedEntities = findAllowedEntities(user, entities, "view_classrooms")
|
isAdmin(user) ? undefined : entityIDS
|
||||||
|
);
|
||||||
|
|
||||||
const groups = await getGroupsForEntities(mapBy(allowedEntities, 'id'));
|
const allowedEntities = findAllowedEntities(
|
||||||
|
user,
|
||||||
|
entities,
|
||||||
|
"view_classrooms"
|
||||||
|
);
|
||||||
|
|
||||||
const users = await getSpecificUsers(uniq(groups.flatMap((g) => [...g.participants, g.admin])));
|
const groups = await getGroupsForEntities(mapBy(allowedEntities, "id"));
|
||||||
const groupsWithUsers: GroupWithUsers[] = groups.map((g) => convertToUsers(g, users.filter(x => isAdmin(user) ? true : !isAdmin(x))));
|
|
||||||
|
|
||||||
return {
|
const users = await getSpecificUsers(
|
||||||
props: serialize({ user, groups: groupsWithUsers, entities: allowedEntities }),
|
uniq(groups.flatMap((g) => [...g.participants, g.admin])),
|
||||||
};
|
{ _id: 0, id: 1, name: 1, email: 1, corporateInformation: 1, type: 1 }
|
||||||
|
);
|
||||||
|
const groupsWithUsers: GroupWithUsers[] = groups.map((g) =>
|
||||||
|
convertToUsers(
|
||||||
|
g,
|
||||||
|
users.filter((x) => (isAdmin(user) ? true : !isAdmin(x)))
|
||||||
|
)
|
||||||
|
);
|
||||||
|
|
||||||
|
return {
|
||||||
|
props: serialize({
|
||||||
|
user,
|
||||||
|
groups: groupsWithUsers,
|
||||||
|
entities: allowedEntities,
|
||||||
|
}),
|
||||||
|
};
|
||||||
}, sessionOptions);
|
}, sessionOptions);
|
||||||
|
|
||||||
const SEARCH_FIELDS = [
|
const SEARCH_FIELDS = [
|
||||||
["name"],
|
["name"],
|
||||||
["admin", "name"],
|
["admin", "name"],
|
||||||
["admin", "email"],
|
["admin", "email"],
|
||||||
["admin", "corporateInformation", "companyInformation", "name"],
|
["admin", "corporateInformation", "companyInformation", "name"],
|
||||||
["participants", "name"],
|
["participants", "name"],
|
||||||
["participants", "email"],
|
["participants", "email"],
|
||||||
["participants", "corporateInformation", "companyInformation", "name"],
|
["participants", "corporateInformation", "companyInformation", "name"],
|
||||||
];
|
];
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
user: User;
|
user: User;
|
||||||
groups: GroupWithUsers[];
|
groups: GroupWithUsers[];
|
||||||
entities: EntityWithRoles[]
|
entities: EntityWithRoles[];
|
||||||
}
|
}
|
||||||
export default function Home({ user, groups, entities }: Props) {
|
export default function Home({ user, groups, entities }: Props) {
|
||||||
const entitiesAllowCreate = useAllowedEntities(user, entities, 'create_classroom');
|
const entitiesAllowCreate = useAllowedEntities(
|
||||||
const [showImport, setShowImport] = useState(false);
|
user,
|
||||||
|
entities,
|
||||||
|
"create_classroom"
|
||||||
|
);
|
||||||
|
const [showImport, setShowImport] = useState(false);
|
||||||
|
|
||||||
const renderCard = (group: GroupWithUsers) => (
|
const renderCard = (group: GroupWithUsers) => (
|
||||||
<Link
|
<Link
|
||||||
href={`/classrooms/${group.id}`}
|
href={`/classrooms/${group.id}`}
|
||||||
key={group.id}
|
key={group.id}
|
||||||
className="p-4 border-2 border-mti-purple-light/20 rounded-xl flex gap-2 justify-between hover:border-mti-purple group transition ease-in-out duration-300 text-left cursor-pointer">
|
className="p-4 border-2 border-mti-purple-light/20 rounded-xl flex gap-2 justify-between hover:border-mti-purple group transition ease-in-out duration-300 text-left cursor-pointer"
|
||||||
<div className="flex flex-col gap-2 w-full">
|
>
|
||||||
<span className="flex items-center gap-1">
|
<div className="flex flex-col gap-2 w-full">
|
||||||
<span className="bg-mti-purple text-white font-semibold px-2">Classroom</span>
|
<span className="flex items-center gap-1">
|
||||||
{group.name}
|
<span className="bg-mti-purple text-white font-semibold px-2">
|
||||||
</span>
|
Classroom
|
||||||
<span className="flex items-center gap-1">
|
</span>
|
||||||
<span className="bg-mti-purple text-white font-semibold px-2">Admin</span>
|
{group.name}
|
||||||
{getUserName(group.admin)}
|
</span>
|
||||||
</span>
|
<span className="flex items-center gap-1">
|
||||||
{!!group.entity && (
|
<span className="bg-mti-purple text-white font-semibold px-2">
|
||||||
<span className="flex items-center gap-1">
|
Admin
|
||||||
<span className="bg-mti-purple text-white font-semibold px-2">Entity</span>
|
</span>
|
||||||
{findBy(entities, 'id', group.entity)?.label}
|
{getUserName(group.admin)}
|
||||||
</span>
|
</span>
|
||||||
)}
|
{!!group.entity && (
|
||||||
<span className="flex items-center gap-1">
|
<span className="flex items-center gap-1">
|
||||||
<span className="bg-mti-purple text-white font-semibold px-2">Participants</span>
|
<span className="bg-mti-purple text-white font-semibold px-2">
|
||||||
<span className="bg-mti-purple-light/50 px-2">{group.participants.length}</span>
|
Entity
|
||||||
</span>
|
</span>
|
||||||
<span>
|
{findBy(entities, "id", group.entity)?.label}
|
||||||
{group.participants.slice(0, 3).map(getUserName).join(", ")}{' '}
|
</span>
|
||||||
{group.participants.length > 3 ? <span className="opacity-50 bg-mti-purple-light/50 px-1 text-sm">and {group.participants.length - 3} more</span> : ""}
|
)}
|
||||||
</span>
|
<span className="flex items-center gap-1">
|
||||||
</div>
|
<span className="bg-mti-purple text-white font-semibold px-2">
|
||||||
<div className="w-fit">
|
Participants
|
||||||
<FaPersonChalkboard className="w-full h-20 -translate-y-[15%] group-hover:text-mti-purple transition ease-in-out duration-300" />
|
</span>
|
||||||
</div>
|
<span className="bg-mti-purple-light/50 px-2">
|
||||||
</Link>
|
{group.participants.length}
|
||||||
);
|
</span>
|
||||||
|
</span>
|
||||||
|
<span>
|
||||||
|
{group.participants.slice(0, 3).map(getUserName).join(", ")}{" "}
|
||||||
|
{group.participants.length > 3 ? (
|
||||||
|
<span className="opacity-50 bg-mti-purple-light/50 px-1 text-sm">
|
||||||
|
and {group.participants.length - 3} more
|
||||||
|
</span>
|
||||||
|
) : (
|
||||||
|
""
|
||||||
|
)}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div className="w-fit">
|
||||||
|
<FaPersonChalkboard className="w-full h-20 -translate-y-[15%] group-hover:text-mti-purple transition ease-in-out duration-300" />
|
||||||
|
</div>
|
||||||
|
</Link>
|
||||||
|
);
|
||||||
|
|
||||||
const firstCard = () => (
|
const firstCard = () => (
|
||||||
<Link
|
<Link
|
||||||
href={`/classrooms/create`}
|
href={`/classrooms/create`}
|
||||||
className="p-4 border-2 hover:text-mti-purple rounded-xl flex flex-col items-center justify-center gap-0 hover:border-mti-purple transition ease-in-out duration-300 text-left cursor-pointer">
|
className="p-4 border-2 hover:text-mti-purple rounded-xl flex flex-col items-center justify-center gap-0 hover:border-mti-purple transition ease-in-out duration-300 text-left cursor-pointer"
|
||||||
<BsPlus size={40} />
|
>
|
||||||
<span className="font-semibold">Create Classroom</span>
|
<BsPlus size={40} />
|
||||||
</Link>
|
<span className="font-semibold">Create Classroom</span>
|
||||||
);
|
</Link>
|
||||||
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Head>
|
<Head>
|
||||||
<title>Classrooms | EnCoach</title>
|
<title>Classrooms | EnCoach</title>
|
||||||
<meta
|
<meta
|
||||||
name="description"
|
name="description"
|
||||||
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
||||||
/>
|
/>
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
<link rel="icon" href="/favicon.ico" />
|
<link rel="icon" href="/favicon.ico" />
|
||||||
</Head>
|
</Head>
|
||||||
<ToastContainer />
|
<ToastContainer />
|
||||||
<Layout user={user} className="!gap-4">
|
<>
|
||||||
<section className="flex flex-col gap-4 w-full h-full">
|
<section className="flex flex-col gap-4 w-full h-full">
|
||||||
<Modal isOpen={showImport} onClose={() => setShowImport(false)} maxWidth="max-w-[85%]">
|
<Modal
|
||||||
<StudentClassroomTransfer user={user} entities={entities} onFinish={() => setShowImport(false)} />
|
isOpen={showImport}
|
||||||
</Modal>
|
onClose={() => setShowImport(false)}
|
||||||
<div className="flex flex-col gap-4">
|
maxWidth="max-w-[85%]"
|
||||||
<div className="flex justify-between">
|
>
|
||||||
<h2 className="font-bold text-2xl">Classrooms</h2>
|
<StudentClassroomTransfer
|
||||||
{entitiesAllowCreate.length !== 0 && <button
|
user={user}
|
||||||
className={clsx(
|
entities={entities}
|
||||||
"flex flex-row gap-3 items-center py-1.5 px-4 text-lg",
|
onFinish={() => setShowImport(false)}
|
||||||
"bg-mti-purple-light border border-mti-purple-light rounded-xl text-white",
|
/>
|
||||||
"hover:bg-white hover:text-mti-purple-light transition duration-300 ease-in-out",
|
</Modal>
|
||||||
)}
|
<div className="flex flex-col gap-4">
|
||||||
onClick={() => setShowImport(true)}
|
<div className="flex justify-between">
|
||||||
>
|
<h2 className="font-bold text-2xl">Classrooms</h2>
|
||||||
<FaFileUpload className="w-5 h-5" />
|
{entitiesAllowCreate.length !== 0 && (
|
||||||
Transfer Students
|
<button
|
||||||
</button>
|
className={clsx(
|
||||||
}
|
"flex flex-row gap-3 items-center py-1.5 px-4 text-lg",
|
||||||
</div>
|
"bg-mti-purple-light border border-mti-purple-light rounded-xl text-white",
|
||||||
<Separator />
|
"hover:bg-white hover:text-mti-purple-light transition duration-300 ease-in-out"
|
||||||
</div>
|
)}
|
||||||
|
onClick={() => setShowImport(true)}
|
||||||
|
>
|
||||||
|
<FaFileUpload className="w-5 h-5" />
|
||||||
|
Transfer Students
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
<Separator />
|
||||||
|
</div>
|
||||||
|
|
||||||
<CardList<GroupWithUsers>
|
<CardList<GroupWithUsers>
|
||||||
list={groups}
|
list={groups}
|
||||||
searchFields={SEARCH_FIELDS}
|
searchFields={SEARCH_FIELDS}
|
||||||
renderCard={renderCard}
|
renderCard={renderCard}
|
||||||
firstCard={entitiesAllowCreate.length === 0 ? undefined : firstCard}
|
firstCard={entitiesAllowCreate.length === 0 ? undefined : firstCard}
|
||||||
/>
|
/>
|
||||||
</section>
|
</section>
|
||||||
</Layout>
|
</>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,191 +1,213 @@
|
|||||||
/* eslint-disable @next/next/no-img-element */
|
/* eslint-disable @next/next/no-img-element */
|
||||||
import Layout from "@/components/High/Layout";
|
|
||||||
import UserDisplayList from "@/components/UserDisplayList";
|
import UserDisplayList from "@/components/UserDisplayList";
|
||||||
import IconCard from "@/components/IconCard";
|
import IconCard from "@/components/IconCard";
|
||||||
import { EntityWithRoles } from "@/interfaces/entity";
|
import { EntityWithRoles } from "@/interfaces/entity";
|
||||||
import { Assignment } from "@/interfaces/results";
|
import { Stat, Type, User } from "@/interfaces/user";
|
||||||
import { Group, Stat, Type, User } from "@/interfaces/user";
|
|
||||||
import { sessionOptions } from "@/lib/session";
|
import { sessionOptions } from "@/lib/session";
|
||||||
import { dateSorter, filterBy, mapBy, redirect, serialize } from "@/utils";
|
import { filterBy, mapBy, redirect, serialize } from "@/utils";
|
||||||
import { requestUser } from "@/utils/api";
|
import { requestUser } from "@/utils/api";
|
||||||
import { countEntitiesAssignments, getAssignments } from "@/utils/assignments.be";
|
import { countEntitiesAssignments } from "@/utils/assignments.be";
|
||||||
import { getEntitiesWithRoles } from "@/utils/entities.be";
|
import { getEntities } from "@/utils/entities.be";
|
||||||
import { countGroups, getGroups } from "@/utils/groups.be";
|
import { countGroups } from "@/utils/groups.be";
|
||||||
import { checkAccess } from "@/utils/permissions";
|
import { checkAccess } from "@/utils/permissions";
|
||||||
import { calculateAverageLevel, calculateBandScore } from "@/utils/score";
|
|
||||||
import { groupByExam } from "@/utils/stats";
|
import { groupByExam } from "@/utils/stats";
|
||||||
import { getStatsByUsers } from "@/utils/stats.be";
|
import { getStatsByUsers } from "@/utils/stats.be";
|
||||||
import { countUsers, getUser, getUsers } from "@/utils/users.be";
|
import { countUsersByTypes, getUsers } from "@/utils/users.be";
|
||||||
import { withIronSessionSsr } from "iron-session/next";
|
import { withIronSessionSsr } from "iron-session/next";
|
||||||
import { uniqBy } from "lodash";
|
|
||||||
import Head from "next/head";
|
import Head from "next/head";
|
||||||
import { useRouter } from "next/router";
|
import { useRouter } from "next/router";
|
||||||
import { useMemo } from "react";
|
|
||||||
import {
|
import {
|
||||||
BsBank,
|
BsBank,
|
||||||
BsClipboard2Data,
|
BsEnvelopePaper,
|
||||||
BsEnvelopePaper,
|
BsPencilSquare,
|
||||||
BsPencilSquare,
|
BsPeople,
|
||||||
BsPeople,
|
BsPeopleFill,
|
||||||
BsPeopleFill,
|
BsPersonFill,
|
||||||
BsPersonFill,
|
BsPersonFillGear,
|
||||||
BsPersonFillGear,
|
|
||||||
} from "react-icons/bs";
|
} from "react-icons/bs";
|
||||||
import { ToastContainer } from "react-toastify";
|
import { ToastContainer } from "react-toastify";
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
user: User;
|
user: User;
|
||||||
students: User[];
|
students: User[];
|
||||||
latestStudents: User[]
|
latestStudents: User[];
|
||||||
latestTeachers: User[]
|
latestTeachers: User[];
|
||||||
entities: EntityWithRoles[];
|
entities: EntityWithRoles[];
|
||||||
usersCount: { [key in Type]: number }
|
usersCount: { [key in Type]: number };
|
||||||
assignmentsCount: number;
|
assignmentsCount: number;
|
||||||
stats: Stat[];
|
stats: Stat[];
|
||||||
groupsCount: number;
|
groupsCount: number;
|
||||||
}
|
}
|
||||||
|
|
||||||
export const getServerSideProps = withIronSessionSsr(async ({ req, res }) => {
|
export const getServerSideProps = withIronSessionSsr(async ({ req, res }) => {
|
||||||
const user = await requestUser(req, res)
|
const user = await requestUser(req, res);
|
||||||
if (!user || !user.isVerified) return redirect("/login")
|
if (!user || !user.isVerified) return redirect("/login");
|
||||||
|
|
||||||
if (!checkAccess(user, ["admin", "developer"])) return redirect("/")
|
if (!checkAccess(user, ["admin", "developer"])) return redirect("/");
|
||||||
|
const [
|
||||||
|
entities,
|
||||||
|
usersCount,
|
||||||
|
groupsCount,
|
||||||
|
students,
|
||||||
|
latestStudents,
|
||||||
|
latestTeachers,
|
||||||
|
] = await Promise.all([
|
||||||
|
getEntities(undefined, { _id: 0, id: 1, label: 1 }),
|
||||||
|
countUsersByTypes(["student", "teacher", "corporate", "mastercorporate"]),
|
||||||
|
countGroups(),
|
||||||
|
getUsers(
|
||||||
|
{ type: "student" },
|
||||||
|
10,
|
||||||
|
{
|
||||||
|
averageLevel: -1,
|
||||||
|
},
|
||||||
|
{ _id: 0, id: 1, name: 1, email: 1, profilePicture: 1 }
|
||||||
|
),
|
||||||
|
getUsers(
|
||||||
|
{ type: "student" },
|
||||||
|
10,
|
||||||
|
{
|
||||||
|
registrationDate: -1,
|
||||||
|
},
|
||||||
|
{ _id: 0, id: 1, name: 1, email: 1, profilePicture: 1 }
|
||||||
|
),
|
||||||
|
getUsers(
|
||||||
|
{ type: "teacher" },
|
||||||
|
10,
|
||||||
|
{
|
||||||
|
registrationDate: -1,
|
||||||
|
},
|
||||||
|
{ _id: 0, id: 1, name: 1, email: 1, profilePicture: 1 }
|
||||||
|
),
|
||||||
|
]);
|
||||||
|
const [assignmentsCount, stats] = await Promise.all([
|
||||||
|
countEntitiesAssignments(mapBy(entities, "id"), {
|
||||||
|
archived: { $ne: true },
|
||||||
|
}),
|
||||||
|
getStatsByUsers(mapBy(students, "id")),
|
||||||
|
]);
|
||||||
|
|
||||||
const students = await getUsers({ type: 'student' });
|
return {
|
||||||
const usersCount = {
|
props: serialize({
|
||||||
student: await countUsers({ type: "student" }),
|
user,
|
||||||
teacher: await countUsers({ type: "teacher" }),
|
students,
|
||||||
corporate: await countUsers({ type: "corporate" }),
|
latestStudents,
|
||||||
mastercorporate: await countUsers({ type: "mastercorporate" }),
|
latestTeachers,
|
||||||
}
|
usersCount,
|
||||||
|
entities,
|
||||||
const latestStudents = await getUsers({ type: 'student' }, 10, { registrationDate: -1 })
|
assignmentsCount,
|
||||||
const latestTeachers = await getUsers({ type: 'teacher' }, 10, { registrationDate: -1 })
|
stats,
|
||||||
|
groupsCount,
|
||||||
const entities = await getEntitiesWithRoles();
|
}),
|
||||||
const assignmentsCount = await countEntitiesAssignments(mapBy(entities, 'id'), { archived: { $ne: true } });
|
};
|
||||||
const groupsCount = await countGroups();
|
|
||||||
|
|
||||||
const stats = await getStatsByUsers(mapBy(students, 'id'));
|
|
||||||
|
|
||||||
return { props: serialize({ user, students, latestStudents, latestTeachers, usersCount, entities, assignmentsCount, stats, groupsCount }) };
|
|
||||||
}, sessionOptions);
|
}, sessionOptions);
|
||||||
|
|
||||||
export default function Dashboard({
|
export default function Dashboard({
|
||||||
user,
|
user,
|
||||||
students,
|
students,
|
||||||
latestStudents,
|
latestStudents,
|
||||||
latestTeachers,
|
latestTeachers,
|
||||||
usersCount,
|
usersCount,
|
||||||
entities,
|
entities,
|
||||||
assignmentsCount,
|
assignmentsCount,
|
||||||
stats,
|
stats,
|
||||||
groupsCount
|
groupsCount,
|
||||||
}: Props) {
|
}: Props) {
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<Head>
|
<Head>
|
||||||
<title>EnCoach</title>
|
<title>EnCoach</title>
|
||||||
<meta
|
<meta
|
||||||
name="description"
|
name="description"
|
||||||
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
content="A training platform for the IELTS exam provided by the Muscat Training Institute and developed by eCrop."
|
||||||
/>
|
/>
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
||||||
<link rel="icon" href="/favicon.ico" />
|
<link rel="icon" href="/favicon.ico" />
|
||||||
</Head>
|
</Head>
|
||||||
<ToastContainer />
|
<ToastContainer />
|
||||||
<Layout user={user}>
|
<>
|
||||||
<section className="grid grid-cols-5 place-items-center -md:grid-cols-2 gap-4 text-center">
|
<section className="grid grid-cols-5 place-items-center -md:grid-cols-2 gap-4 text-center">
|
||||||
<IconCard
|
<IconCard
|
||||||
onClick={() => router.push("/users?type=student")}
|
onClick={() => router.push("/users?type=student")}
|
||||||
Icon={BsPersonFill}
|
Icon={BsPersonFill}
|
||||||
label="Students"
|
label="Students"
|
||||||
value={usersCount.student}
|
value={usersCount.student}
|
||||||
color="purple"
|
color="purple"
|
||||||
/>
|
/>
|
||||||
<IconCard
|
<IconCard
|
||||||
onClick={() => router.push("/users?type=teacher")}
|
onClick={() => router.push("/users?type=teacher")}
|
||||||
Icon={BsPencilSquare}
|
Icon={BsPencilSquare}
|
||||||
label="Teachers"
|
label="Teachers"
|
||||||
value={usersCount.teacher}
|
value={usersCount.teacher}
|
||||||
color="purple"
|
color="purple"
|
||||||
/>
|
/>
|
||||||
<IconCard
|
<IconCard
|
||||||
Icon={BsBank}
|
Icon={BsBank}
|
||||||
onClick={() => router.push("/users?type=corporate")}
|
onClick={() => router.push("/users?type=corporate")}
|
||||||
label="Corporates"
|
label="Corporates"
|
||||||
value={usersCount.corporate}
|
value={usersCount.corporate}
|
||||||
color="purple"
|
color="purple"
|
||||||
/>
|
/>
|
||||||
<IconCard
|
<IconCard
|
||||||
Icon={BsBank}
|
Icon={BsBank}
|
||||||
onClick={() => router.push("/users?type=mastercorporate")}
|
onClick={() => router.push("/users?type=mastercorporate")}
|
||||||
label="Master Corporates"
|
label="Master Corporates"
|
||||||
value={usersCount.mastercorporate}
|
value={usersCount.mastercorporate}
|
||||||
color="purple"
|
color="purple"
|
||||||
/>
|
/>
|
||||||
<IconCard
|
<IconCard
|
||||||
Icon={BsPeople}
|
Icon={BsPeople}
|
||||||
onClick={() => router.push("/classrooms")}
|
onClick={() => router.push("/classrooms")}
|
||||||
label="Classrooms"
|
label="Classrooms"
|
||||||
value={groupsCount}
|
value={groupsCount}
|
||||||
color="purple"
|
color="purple"
|
||||||
/>
|
/>
|
||||||
<IconCard Icon={BsPeopleFill}
|
<IconCard
|
||||||
onClick={() => router.push("/entities")}
|
Icon={BsPeopleFill}
|
||||||
label="Entities"
|
onClick={() => router.push("/entities")}
|
||||||
value={entities.length}
|
label="Entities"
|
||||||
color="purple"
|
value={entities.length}
|
||||||
/>
|
color="purple"
|
||||||
<IconCard Icon={BsPersonFillGear}
|
/>
|
||||||
onClick={() => router.push("/statistical")}
|
<IconCard
|
||||||
label="Entity Statistics"
|
Icon={BsPersonFillGear}
|
||||||
value={entities.length}
|
onClick={() => router.push("/statistical")}
|
||||||
color="purple"
|
label="Entity Statistics"
|
||||||
/>
|
value={entities.length}
|
||||||
<IconCard Icon={BsPersonFillGear}
|
color="purple"
|
||||||
onClick={() => router.push("/users/performance")}
|
/>
|
||||||
label="Student Performance"
|
<IconCard
|
||||||
value={usersCount.student}
|
Icon={BsPersonFillGear}
|
||||||
color="purple"
|
onClick={() => router.push("/users/performance")}
|
||||||
/>
|
label="Student Performance"
|
||||||
<IconCard
|
value={usersCount.student}
|
||||||
Icon={BsEnvelopePaper}
|
color="purple"
|
||||||
onClick={() => router.push("/assignments")}
|
/>
|
||||||
label="Assignments"
|
<IconCard
|
||||||
value={assignmentsCount}
|
Icon={BsEnvelopePaper}
|
||||||
color="purple"
|
onClick={() => router.push("/assignments")}
|
||||||
/>
|
label="Assignments"
|
||||||
</section>
|
value={assignmentsCount}
|
||||||
|
color="purple"
|
||||||
|
/>
|
||||||
|
</section>
|
||||||
|
|
||||||
<section className="grid grid-cols-1 md:grid-cols-2 gap-4 w-full justify-between">
|
<section className="grid grid-cols-1 md:grid-cols-2 gap-4 w-full justify-between">
|
||||||
<UserDisplayList
|
<UserDisplayList users={latestStudents} title="Latest Students" />
|
||||||
users={latestStudents}
|
<UserDisplayList users={latestTeachers} title="Latest Teachers" />
|
||||||
title="Latest Students"
|
<UserDisplayList users={students} title="Highest level students" />
|
||||||
/>
|
<UserDisplayList
|
||||||
<UserDisplayList
|
users={students.sort(
|
||||||
users={latestTeachers}
|
(a, b) =>
|
||||||
title="Latest Teachers"
|
Object.keys(groupByExam(filterBy(stats, "user", b))).length -
|
||||||
/>
|
Object.keys(groupByExam(filterBy(stats, "user", a))).length
|
||||||
<UserDisplayList
|
)}
|
||||||
users={students.sort((a, b) => calculateAverageLevel(b.levels) - calculateAverageLevel(a.levels))}
|
title="Highest exam count students"
|
||||||
title="Highest level students"
|
/>
|
||||||
/>
|
</section>
|
||||||
<UserDisplayList
|
</>
|
||||||
users={
|
</>
|
||||||
students
|
);
|
||||||
.sort(
|
|
||||||
(a, b) =>
|
|
||||||
Object.keys(groupByExam(filterBy(stats, "user", b))).length -
|
|
||||||
Object.keys(groupByExam(filterBy(stats, "user", a))).length,
|
|
||||||
)
|
|
||||||
}
|
|
||||||
title="Highest exam count students"
|
|
||||||
/>
|
|
||||||
</section>
|
|
||||||
</Layout>
|
|
||||||
</>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user