2023-03-14 09:56:03 +01:00
|
|
|
import { useCallback, useMemo } from 'react';
|
2022-06-03 12:32:30 +02:00
|
|
|
|
2023-03-14 09:56:03 +01:00
|
|
|
export type IGetSearchContextOutput<T extends any = any> = {
|
|
|
|
data: T[];
|
2022-06-03 12:32:30 +02:00
|
|
|
columns: any[];
|
|
|
|
searchValue: string;
|
2023-03-14 09:56:03 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
type IUseSearchOutput<T extends any> = {
|
|
|
|
getSearchText: (input: string) => string;
|
|
|
|
data: T[];
|
|
|
|
getSearchContext: () => IGetSearchContextOutput<T>;
|
|
|
|
};
|
2022-06-03 12:32:30 +02:00
|
|
|
|
2023-03-14 09:56:03 +01:00
|
|
|
export const useSearch = <T extends any>(
|
2022-06-03 12:32:30 +02:00
|
|
|
columns: any[],
|
|
|
|
searchValue: string,
|
2023-03-14 09:56:03 +01:00
|
|
|
data: T[]
|
|
|
|
): IUseSearchOutput<T> => {
|
|
|
|
const getSearchText = useCallback(
|
|
|
|
(value: string) => getSearchTextGenerator(columns)(value),
|
|
|
|
[columns]
|
|
|
|
);
|
2022-06-03 12:32:30 +02:00
|
|
|
|
2023-03-14 09:56:03 +01:00
|
|
|
const getSearchContext = useCallback(() => {
|
2022-06-03 12:32:30 +02:00
|
|
|
return { data, searchValue, columns };
|
2023-03-14 09:56:03 +01:00
|
|
|
}, [data, searchValue, columns]);
|
2022-06-03 12:32:30 +02:00
|
|
|
|
2023-03-14 09:56:03 +01:00
|
|
|
const search = useMemo(() => {
|
|
|
|
if (!searchValue) return data;
|
2022-06-03 12:32:30 +02:00
|
|
|
|
|
|
|
const filteredData = filter(columns, searchValue, data);
|
|
|
|
const searchedData = searchInFilteredData(
|
|
|
|
columns,
|
|
|
|
getSearchText(searchValue),
|
|
|
|
filteredData
|
|
|
|
);
|
|
|
|
|
|
|
|
return searchedData;
|
2023-03-14 09:56:03 +01:00
|
|
|
}, [columns, searchValue, data, getSearchText]);
|
2022-06-03 12:32:30 +02:00
|
|
|
|
2023-03-14 09:56:03 +01:00
|
|
|
return { data: search, getSearchText, getSearchContext };
|
2022-06-03 12:32:30 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
export const filter = (columns: any[], searchValue: string, data: any[]) => {
|
|
|
|
let filteredDataSet = data;
|
|
|
|
|
|
|
|
getFilterableColumns(columns)
|
|
|
|
.filter(column => isValidFilter(searchValue, column.filterName))
|
|
|
|
.forEach(column => {
|
|
|
|
const values = getFilterValues(column.filterName, searchValue);
|
|
|
|
|
|
|
|
filteredDataSet = filteredDataSet.filter(row => {
|
|
|
|
if (column.filterBy) {
|
|
|
|
return column.filterBy(row, values);
|
|
|
|
}
|
|
|
|
|
|
|
|
return defaultFilter(getColumnValues(column, row), values);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
return filteredDataSet;
|
|
|
|
};
|
|
|
|
|
2023-03-14 09:56:03 +01:00
|
|
|
export const searchInFilteredData = <T extends any>(
|
2022-06-03 12:32:30 +02:00
|
|
|
columns: any[],
|
|
|
|
searchValue: string,
|
2023-03-14 09:56:03 +01:00
|
|
|
filteredData: T[]
|
2022-06-03 12:32:30 +02:00
|
|
|
) => {
|
|
|
|
const searchableColumns = columns.filter(
|
|
|
|
column => column.searchable && column.accessor
|
|
|
|
);
|
|
|
|
|
|
|
|
return filteredData.filter(row => {
|
|
|
|
return searchableColumns.some(column => {
|
|
|
|
if (column.searchBy) {
|
|
|
|
return column.searchBy(row, searchValue);
|
|
|
|
}
|
|
|
|
|
|
|
|
return defaultSearch(getColumnValues(column, row), searchValue);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const defaultFilter = (fieldValue: string, values: string[]) =>
|
|
|
|
values.some(value => fieldValue?.toLowerCase() === value?.toLowerCase());
|
|
|
|
|
|
|
|
const defaultSearch = (fieldValue: string, value: string) =>
|
|
|
|
fieldValue?.toLowerCase().includes(value?.toLowerCase());
|
|
|
|
|
|
|
|
export const getSearchTextGenerator = (columns: any[]) => {
|
|
|
|
const filters = columns
|
|
|
|
.filter(column => column.filterName)
|
|
|
|
.map(column => column.filterName);
|
|
|
|
|
|
|
|
const isValidSearch = (fragment: string) => {
|
|
|
|
return filters.some(filter => isValidFilter(fragment, filter));
|
|
|
|
};
|
|
|
|
|
|
|
|
return (searchValue: string) =>
|
|
|
|
searchValue
|
|
|
|
.split(' ')
|
|
|
|
.filter(fragment => !isValidSearch(fragment))
|
|
|
|
.join(' ');
|
|
|
|
};
|
|
|
|
|
|
|
|
export const isValidFilter = (input: string, match: string) =>
|
|
|
|
new RegExp(`${match}:\\w+`).test(input);
|
|
|
|
|
|
|
|
export const getFilterableColumns = (columns: any[]) =>
|
|
|
|
columns.filter(column => column.filterName && column.accessor);
|
|
|
|
|
|
|
|
export const getColumnValues = (column: any, row: any) => {
|
|
|
|
const value =
|
|
|
|
typeof column.accessor === 'function'
|
|
|
|
? column.accessor(row)
|
|
|
|
: column.accessor.includes('.')
|
|
|
|
? column.accessor
|
|
|
|
.split('.')
|
2022-07-13 16:35:43 +02:00
|
|
|
.reduce((object: any, key: string) => object?.[key], row)
|
2022-06-03 12:32:30 +02:00
|
|
|
: row[column.accessor];
|
|
|
|
|
|
|
|
if (column.filterParsing) {
|
|
|
|
return column.filterParsing(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
return value;
|
|
|
|
};
|
|
|
|
|
|
|
|
export const getFilterValues = (filterName: string, searchValue: string) =>
|
|
|
|
searchValue
|
|
|
|
?.split(`${filterName}:`)[1]
|
|
|
|
?.split(' ')[0]
|
|
|
|
?.split(',')
|
|
|
|
.filter(value => value) ?? [];
|