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.
 
 

107 regels
3.0 KiB

  1. "use client";
  2. import { ProjectCategory, ProjectResult } from "@/app/api/projects";
  3. import React, { useCallback, useMemo, useState } from "react";
  4. import SearchBox, { Criterion } from "../SearchBox";
  5. import { useTranslation } from "react-i18next";
  6. import SearchResults, { Column } from "../SearchResults";
  7. import EditNote from "@mui/icons-material/EditNote";
  8. import uniq from "lodash/uniq";
  9. import { useRouter } from "next/navigation";
  10. interface Props {
  11. projects: ProjectResult[];
  12. projectCategories: ProjectCategory[];
  13. }
  14. type SearchQuery = Partial<Omit<ProjectResult, "id">>;
  15. type SearchParamNames = keyof SearchQuery;
  16. const ProjectSearch: React.FC<Props> = ({ projects, projectCategories }) => {
  17. const router = useRouter();
  18. const { t } = useTranslation("projects");
  19. const [filteredProjects, setFilteredProjects] = useState(projects);
  20. const searchCriteria: Criterion<SearchParamNames>[] = useMemo(
  21. () => [
  22. { label: t("Project code"), paramName: "code", type: "text" },
  23. { label: t("Project name"), paramName: "name", type: "text" },
  24. {
  25. label: t("Client name"),
  26. paramName: "client",
  27. type: "select",
  28. options: uniq(projects.map((project) => project.client)),
  29. },
  30. {
  31. label: t("Project category"),
  32. paramName: "category",
  33. type: "select",
  34. options: projectCategories.map((category) => category.name),
  35. },
  36. {
  37. label: t("Team"),
  38. paramName: "team",
  39. type: "select",
  40. options: uniq(projects.map((project) => project.team)),
  41. },
  42. ],
  43. [t, projectCategories, projects],
  44. );
  45. const onReset = useCallback(() => {
  46. setFilteredProjects(projects);
  47. }, [projects]);
  48. const onProjectClick = useCallback(
  49. (project: ProjectResult) => {
  50. router.push(`/projects/edit?id=${project.id}`);
  51. },
  52. [router],
  53. );
  54. const columns = useMemo<Column<ProjectResult>[]>(
  55. () => [
  56. {
  57. name: "id",
  58. label: t("Details"),
  59. onClick: onProjectClick,
  60. buttonIcon: <EditNote />,
  61. },
  62. { name: "code", label: t("Project Code") },
  63. { name: "name", label: t("Project Name") },
  64. { name: "category", label: t("Project Category") },
  65. { name: "team", label: t("Team") },
  66. { name: "client", label: t("Client") },
  67. ],
  68. [t, onProjectClick],
  69. );
  70. return (
  71. <>
  72. <SearchBox
  73. criteria={searchCriteria}
  74. onSearch={(query) => {
  75. setFilteredProjects(
  76. projects.filter(
  77. (p) =>
  78. p.code.toLowerCase().includes(query.code.toLowerCase()) &&
  79. p.name.toLowerCase().includes(query.name.toLowerCase()) &&
  80. (query.client === "All" || p.client === query.client) &&
  81. (query.category === "All" || p.category === query.category) &&
  82. (query.team === "All" || p.team === query.team),
  83. ),
  84. );
  85. }}
  86. onReset={onReset}
  87. />
  88. <SearchResults<ProjectResult>
  89. items={filteredProjects}
  90. columns={columns}
  91. />
  92. </>
  93. );
  94. };
  95. export default ProjectSearch;