You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 

117 lines
3.0 KiB

  1. "use client";
  2. import React, { useCallback, useEffect, useMemo, useState } from "react";
  3. import SearchBox, { Criterion } from "../SearchBox/index";
  4. import { useTranslation } from "react-i18next";
  5. import SearchResults, { Column } from "../SearchResults/index";
  6. import EditNote from "@mui/icons-material/EditNote";
  7. import DeleteIcon from "@mui/icons-material/Delete";
  8. import { useRouter } from "next/navigation";
  9. import { deleteDialog, successDialog } from "../Swal/CustomAlerts";
  10. import { SkillResult } from "@/app/api/skill";
  11. import { deleteSkill } from "@/app/api/skill/actions";
  12. import { MAINTAIN_SKILL } from "@/middleware";
  13. interface Props {
  14. skill: SkillResult[];
  15. abilities: String[];
  16. }
  17. type SearchQuery = Partial<Omit<SkillResult, "id">>;
  18. type SearchParamNames = keyof SearchQuery;
  19. const SkillSearch: React.FC<Props> = ({ skill, abilities }) => {
  20. const { t } = useTranslation();
  21. const [filteredSkill, setFilteredSkill] = useState(skill);
  22. const router = useRouter();
  23. // translation
  24. const skillName = t("Skill Name")
  25. const skillCode = t("Skill Code")
  26. const deleteSuccess = t("Delete Success")
  27. const edit = t("Actions")
  28. const name = t("name")
  29. const code = t("code")
  30. const description = t("description")
  31. const maintainSkill = abilities.includes(MAINTAIN_SKILL)
  32. const searchCriteria: Criterion<SearchParamNames>[] = useMemo(
  33. () => [
  34. {
  35. label: skillName,
  36. paramName: "name",
  37. type: "text",
  38. },
  39. {
  40. label: skillCode,
  41. paramName: "code",
  42. type: "text",
  43. },
  44. ],
  45. [t]
  46. );
  47. const onSkillClick = useCallback(
  48. (skill: SkillResult) => {
  49. console.log(skill);
  50. const id = skill.id;
  51. router.push(`/settings/skill/edit?id=${id}`);
  52. },
  53. [router, t]
  54. );
  55. const deleteClick = useCallback((skill: SkillResult) => {
  56. deleteDialog(async () => {
  57. await deleteSkill(skill.id);
  58. successDialog(deleteSuccess, t);
  59. }, t);
  60. }, []);
  61. useEffect(() => {
  62. setFilteredSkill(skill)
  63. }, [skill]);
  64. const columns = useMemo<Column<SkillResult>[]>(
  65. () => [
  66. {
  67. name: "action",
  68. label: edit,
  69. onClick: onSkillClick,
  70. buttonIcon: <EditNote />,
  71. isHidden: !maintainSkill,
  72. },
  73. { name: "name", label: name },
  74. { name: "code", label: code },
  75. { name: "description", label: description },
  76. {
  77. name: "delete",
  78. label: t("Delete"),
  79. onClick: deleteClick,
  80. buttonIcon: <DeleteIcon />,
  81. color: "error",
  82. isHidden: !maintainSkill,
  83. },
  84. ],
  85. [t, onSkillClick, deleteClick]
  86. );
  87. return (
  88. <>
  89. <SearchBox
  90. criteria={searchCriteria}
  91. onSearch={(query) => {
  92. setFilteredSkill(
  93. skill.filter(
  94. (s) =>
  95. s.name.toLowerCase().includes(query.name.toLowerCase()) &&
  96. s.code.toLowerCase().includes(query.code.toLowerCase())
  97. )
  98. );
  99. }}
  100. />
  101. <SearchResults<SkillResult> items={filteredSkill} columns={columns} />
  102. </>
  103. );
  104. };
  105. export default SkillSearch;