feat: search input

This commit is contained in:
2024-08-24 05:16:43 +03:00
parent 30a96811bb
commit 40769d3664
6 changed files with 30 additions and 13 deletions

View File

@@ -7,6 +7,6 @@ export type AuthLoginRequest = {
first_name: string; first_name: string;
hash: string; hash: string;
id: number; id: number;
photo_url: string; photo_url?: (string | null);
}; };

View File

@@ -92,10 +92,12 @@ export class ProductService {
*/ */
public static getProductsByClientId({ public static getProductsByClientId({
clientId, clientId,
searchInput,
page, page,
itemsPerPage, itemsPerPage,
}: { }: {
clientId: number, clientId: number,
searchInput: string,
page?: (number | null), page?: (number | null),
itemsPerPage?: (number | null), itemsPerPage?: (number | null),
}): CancelablePromise<ProductGetResponse> { }): CancelablePromise<ProductGetResponse> {
@@ -104,6 +106,7 @@ export class ProductService {
url: '/product/get', url: '/product/get',
query: { query: {
'client_id': clientId, 'client_id': clientId,
'search_input': searchInput,
'page': page, 'page': page,
'items_per_page': itemsPerPage, 'items_per_page': itemsPerPage,
}, },

View File

@@ -14,7 +14,7 @@ const ClientSelect: FC<Props> = ({value, onChange, withLabel = false}) => {
const options = clients.map(client => ({label: client.name, value: client.id.toString()})) const options = clients.map(client => ({label: client.name, value: client.id.toString()}))
return ( return (
<Select <Select
searchable
placeholder={"Выберите клиента"} placeholder={"Выберите клиента"}
value={value && options.find(client => client.value == value.id.toString())?.value} value={value && options.find(client => client.value == value.id.toString())?.value}
onChange={event => { onChange={event => {

View File

@@ -5,12 +5,13 @@ type Props = {
clientId: number, clientId: number,
page?: number, page?: number,
itemsPerPage?: number, itemsPerPage?: number,
searchInput: string
} }
const useProductsList = (props: Props) => { const useProductsList = (props: Props) => {
const {clientId, page, itemsPerPage} = props; const {clientId, page, itemsPerPage, searchInput} = props;
const {data, refetch} = useQuery({ const {data, refetch} = useQuery({
queryKey: ['getAllServices', clientId, page, itemsPerPage], queryKey: ['getAllServices', clientId, page, itemsPerPage, searchInput],
queryFn: () => ProductService.getProductsByClientId({clientId, page, itemsPerPage}) queryFn: () => ProductService.getProductsByClientId(props)
}); });
const products = !data ? [] : data.products; const products = !data ? [] : data.products;
const paginationInfo = data?.paginationInfo; const paginationInfo = data?.paginationInfo;

View File

@@ -13,6 +13,7 @@
padding: rem(5); padding: rem(5);
gap: rem(10); gap: rem(10);
display: flex; display: flex;
justify-content: space-between;
} }
.table-container { .table-container {

View File

@@ -1,7 +1,7 @@
import PageBlock from "../../../components/PageBlock/PageBlock.tsx"; import PageBlock from "../../../components/PageBlock/PageBlock.tsx";
import {FC, useEffect, useState} from "react"; import {FC, useEffect, useState} from "react";
import styles from './ProductsPage.module.css'; import styles from './ProductsPage.module.css';
import {Button, Pagination, Text} from "@mantine/core"; import {Button, Flex, Pagination, rem, Text, TextInput} from "@mantine/core";
import ClientSelect from "../../../components/Selects/ClientSelect/ClientSelect.tsx"; import ClientSelect from "../../../components/Selects/ClientSelect/ClientSelect.tsx";
import ProductsTable from "../components/ProductsTable/ProductsTable.tsx"; import ProductsTable from "../components/ProductsTable/ProductsTable.tsx";
import {modals} from "@mantine/modals"; import {modals} from "@mantine/modals";
@@ -9,16 +9,19 @@ import {notifications} from "../../../shared/lib/notifications.ts";
import {CreateProductRequest} from "../types.ts"; import {CreateProductRequest} from "../types.ts";
import {ProductSchema, ProductService} from "../../../client"; import {ProductSchema, ProductService} from "../../../client";
import useProductsList from "../hooks/useProductsList.tsx"; import useProductsList from "../hooks/useProductsList.tsx";
import {useDebouncedValue} from "@mantine/hooks";
export const ProductsPage: FC = () => { export const ProductsPage: FC = () => {
const [clientId, setClientId] = useState(-1); const [clientId, setClientId] = useState(-1);
const [totalPages, setTotalPages] = useState(1); const [totalPages, setTotalPages] = useState(1);
const [currentPage, setCurrentPage] = useState(1); const [currentPage, setCurrentPage] = useState(1);
const [searchInputRaw, setSearchInputRaw] = useState<string>("");
const [searchInput] = useDebouncedValue(searchInputRaw, 500);
const {products, paginationInfo, refetch} = useProductsList({ const {products, paginationInfo, refetch} = useProductsList({
clientId, clientId,
page: currentPage - 1, page: currentPage - 1,
itemsPerPage: 10 itemsPerPage: 10,
searchInput
}); });
const onProductCreate = (request: CreateProductRequest) => { const onProductCreate = (request: CreateProductRequest) => {
ProductService.createProduct({requestBody: request}) ProductService.createProduct({requestBody: request})
@@ -88,12 +91,21 @@ export const ProductsPage: FC = () => {
<div className={styles['container']}> <div className={styles['container']}>
<PageBlock> <PageBlock>
<div className={styles['top-panel']}> <div className={styles['top-panel']}>
<ClientSelect onChange={event => setClientId(event.id)}/> <Flex gap={rem(10)}>
<Button
onClick={() => onCreateProductClick()}
variant={"default"}
>Создать</Button>
<ClientSelect onChange={event => setClientId(event.id)}/>
<Button
onClick={() => onCreateProductClick()}
variant={"default"}
>Создать</Button>
</Flex>
<Flex>
<TextInput
placeholder={"Артикул, название, шк"}
onChange={event => setSearchInputRaw(event.currentTarget.value)}
value={searchInputRaw}
/>
</Flex>
</div> </div>
</PageBlock> </PageBlock>
<PageBlock> <PageBlock>