feishin/src/renderer/components/query-builder/index.tsx

219 lines
6.3 KiB
TypeScript
Raw Normal View History

2023-01-04 04:09:24 -08:00
import { Group, Stack } from '@mantine/core';
import { Select } from '/@/renderer/components/select';
import { AnimatePresence, motion } from 'framer-motion';
import { RiAddFill, RiAddLine, RiDeleteBinFill, RiMore2Line, RiRestartLine } from 'react-icons/ri';
2023-01-04 04:09:24 -08:00
import { Button } from '/@/renderer/components/button';
import { DropdownMenu } from '/@/renderer/components/dropdown-menu';
import { QueryBuilderOption } from '/@/renderer/components/query-builder/query-builder-option';
2023-01-04 15:54:25 -08:00
import { QueryBuilderGroup, QueryBuilderRule } from '/@/renderer/types';
2023-01-04 04:09:24 -08:00
const FILTER_GROUP_OPTIONS_DATA = [
{
label: 'Match all',
value: 'all',
},
{
label: 'Match any',
value: 'any',
},
];
type AddArgs = {
groupIndex: number[];
level: number;
};
type DeleteArgs = {
groupIndex: number[];
level: number;
uniqueId: string;
};
interface QueryBuilderProps {
data: Record<string, any>;
filters: { label: string; type: string; value: string }[];
2023-01-04 04:09:24 -08:00
groupIndex: number[];
level: number;
onAddRule: (args: AddArgs) => void;
onAddRuleGroup: (args: AddArgs) => void;
onChangeField: (args: any) => void;
onChangeOperator: (args: any) => void;
onChangeType: (args: any) => void;
onChangeValue: (args: any) => void;
onClearFilters: () => void;
2023-01-04 04:09:24 -08:00
onDeleteRule: (args: DeleteArgs) => void;
onDeleteRuleGroup: (args: DeleteArgs) => void;
onResetFilters: () => void;
operators: {
boolean: { label: string; value: string }[];
date: { label: string; value: string }[];
number: { label: string; value: string }[];
string: { label: string; value: string }[];
};
2023-01-04 04:09:24 -08:00
uniqueId: string;
}
export const QueryBuilder = ({
data,
level,
onAddRule,
onDeleteRuleGroup,
onDeleteRule,
onAddRuleGroup,
onChangeType,
onChangeField,
operators,
2023-01-04 04:09:24 -08:00
onChangeOperator,
onChangeValue,
onClearFilters,
onResetFilters,
2023-01-04 04:09:24 -08:00
groupIndex,
uniqueId,
filters,
}: QueryBuilderProps) => {
const handleAddRule = () => {
2023-01-04 15:54:25 -08:00
onAddRule({ groupIndex, level });
2023-01-04 04:09:24 -08:00
};
const handleAddRuleGroup = () => {
2023-01-04 15:54:25 -08:00
onAddRuleGroup({ groupIndex, level });
2023-01-04 04:09:24 -08:00
};
const handleDeleteRuleGroup = () => {
2023-01-04 15:54:25 -08:00
onDeleteRuleGroup({ groupIndex, level, uniqueId });
2023-01-04 04:09:24 -08:00
};
const handleChangeType = (value: string | null) => {
onChangeType({ groupIndex, level, value });
};
return (
<Stack
ml={`${level * 10}px`}
spacing="sm"
>
<Group spacing="sm">
2023-01-04 04:09:24 -08:00
<Select
data={FILTER_GROUP_OPTIONS_DATA}
maxWidth={175}
2023-01-04 15:54:25 -08:00
size="sm"
value={data.type}
2023-01-04 04:09:24 -08:00
width="20%"
onChange={handleChangeType}
/>
<Button
px={5}
2023-01-04 15:54:25 -08:00
size="sm"
2023-01-04 04:09:24 -08:00
tooltip={{ label: 'Add rule' }}
variant="default"
onClick={handleAddRule}
>
<RiAddLine size={20} />
</Button>
<DropdownMenu position="bottom-start">
2023-01-04 04:09:24 -08:00
<DropdownMenu.Target>
<Button
p={0}
2023-01-04 15:54:25 -08:00
size="sm"
2023-01-04 04:09:24 -08:00
variant="subtle"
>
<RiMore2Line size={20} />
</Button>
</DropdownMenu.Target>
<DropdownMenu.Dropdown>
<DropdownMenu.Item
icon={<RiAddFill />}
onClick={handleAddRuleGroup}
>
Add rule group
</DropdownMenu.Item>
2023-01-04 04:09:24 -08:00
{level > 0 && (
<DropdownMenu.Item
icon={<RiDeleteBinFill />}
onClick={handleDeleteRuleGroup}
>
2023-01-04 04:09:24 -08:00
Remove rule group
</DropdownMenu.Item>
)}
{level === 0 && (
<>
<DropdownMenu.Divider />
<DropdownMenu.Item
$danger
icon={<RiRestartLine color="var(--danger-color)" />}
onClick={onResetFilters}
>
Reset to default
</DropdownMenu.Item>
<DropdownMenu.Item
$danger
icon={<RiDeleteBinFill color="var(--danger-color)" />}
onClick={onClearFilters}
>
Clear filters
</DropdownMenu.Item>
</>
)}
2023-01-04 04:09:24 -08:00
</DropdownMenu.Dropdown>
</DropdownMenu>
</Group>
<AnimatePresence initial={false}>
2023-01-04 15:54:25 -08:00
{data?.rules?.map((rule: QueryBuilderRule) => (
2023-01-04 04:09:24 -08:00
<motion.div
2023-01-04 15:54:25 -08:00
key={rule.uniqueId}
2023-01-04 04:09:24 -08:00
animate={{ opacity: 1, x: 0 }}
exit={{ opacity: 0, x: -25 }}
initial={{ opacity: 0, x: -25 }}
transition={{ duration: 0.2, ease: 'easeInOut' }}
>
<QueryBuilderOption
data={rule}
filters={filters}
groupIndex={groupIndex || []}
level={level}
noRemove={data?.rules?.length === 1}
operators={operators}
2023-01-04 04:09:24 -08:00
onChangeField={onChangeField}
onChangeOperator={onChangeOperator}
onChangeValue={onChangeValue}
onDeleteRule={onDeleteRule}
/>
</motion.div>
))}
</AnimatePresence>
2023-01-04 15:54:25 -08:00
{data?.group && (
<AnimatePresence initial={false}>
2023-01-04 15:54:25 -08:00
{data.group?.map((group: QueryBuilderGroup, index: number) => (
2023-01-04 04:09:24 -08:00
<motion.div
2023-01-04 15:54:25 -08:00
key={group.uniqueId}
2023-01-04 04:09:24 -08:00
animate={{ opacity: 1, x: 0 }}
exit={{ opacity: 0, x: -25 }}
initial={{ opacity: 0, x: -25 }}
transition={{ duration: 0.2, ease: 'easeInOut' }}
>
<QueryBuilder
data={group}
filters={filters}
groupIndex={[...(groupIndex || []), index]}
level={level + 1}
operators={operators}
2023-01-04 04:09:24 -08:00
uniqueId={group.uniqueId}
onAddRule={onAddRule}
onAddRuleGroup={onAddRuleGroup}
onChangeField={onChangeField}
onChangeOperator={onChangeOperator}
onChangeType={onChangeType}
onChangeValue={onChangeValue}
onClearFilters={onClearFilters}
2023-01-04 04:09:24 -08:00
onDeleteRule={onDeleteRule}
onDeleteRuleGroup={onDeleteRuleGroup}
onResetFilters={onResetFilters}
2023-01-04 04:09:24 -08:00
/>
</motion.div>
))}
</AnimatePresence>
)}
</Stack>
);
};