feat: search input
This commit is contained in:
@@ -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);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
@@ -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,
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -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 => {
|
||||||
|
|||||||
@@ -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;
|
||||||
|
|||||||
@@ -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 {
|
||||||
|
|||||||
@@ -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>
|
||||||
|
|||||||
Reference in New Issue
Block a user