25'ten fazla konu seçemezsiniz Konular bir harf veya rakamla başlamalı, kısa çizgiler ('-') içerebilir ve en fazla 35 karakter uzunluğunda olabilir.
 
 

117 satır
3.4 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. label: t("Status"),
  44. paramName: "status",
  45. type: "select",
  46. options: uniq(projects.map((project) => project.status)),
  47. },
  48. ],
  49. [t, projectCategories, projects],
  50. );
  51. const onReset = useCallback(() => {
  52. setFilteredProjects(projects);
  53. }, [projects]);
  54. const onProjectClick = useCallback(
  55. (project: ProjectResult) => {
  56. if (Boolean(project.mainProject)) {
  57. router.push(`/projects/editSub?id=${project.id}`);
  58. } else router.push(`/projects/edit?id=${project.id}`);
  59. },
  60. [router],
  61. );
  62. const columns = useMemo<Column<ProjectResult>[]>(
  63. () => [
  64. {
  65. name: "id",
  66. label: t("Details"),
  67. onClick: onProjectClick,
  68. buttonIcon: <EditNote />,
  69. },
  70. { name: "code", label: t("Project Code") },
  71. { name: "name", label: t("Project Name") },
  72. { name: "category", label: t("Project Category") },
  73. { name: "team", label: t("Team") },
  74. { name: "client", label: t("Client") },
  75. { name: "status", label: t("Status") },
  76. ],
  77. [t, onProjectClick],
  78. );
  79. return (
  80. <>
  81. <SearchBox
  82. criteria={searchCriteria}
  83. onSearch={(query) => {
  84. setFilteredProjects(
  85. projects.filter(
  86. (p) =>
  87. p.code.toLowerCase().includes(query.code.toLowerCase()) &&
  88. p.name.toLowerCase().includes(query.name.toLowerCase()) &&
  89. (query.client === "All" || p.client === query.client) &&
  90. (query.category === "All" || p.category === query.category) &&
  91. (query.team === "All" || p.team === query.team) &&
  92. (query.status === "All" || p.status === query.status),
  93. ),
  94. );
  95. }}
  96. onReset={onReset}
  97. />
  98. <SearchResults<ProjectResult>
  99. items={filteredProjects}
  100. columns={columns}
  101. />
  102. </>
  103. );
  104. };
  105. export default ProjectSearch;