mirror of
https://github.com/infiniflow/ragflow.git
synced 2025-06-26 22:19:57 +00:00

### What problem does this PR solve? Feat: Add child nodes and their connecting lines by clicking #3221 ### Type of change - [x] New Feature (non-breaking change which adds functionality)
111 lines
3.7 KiB
TypeScript
111 lines
3.7 KiB
TypeScript
import { IconFont } from '@/components/icon-font';
|
|
import { Card, CardContent } from '@/components/ui/card';
|
|
import { ISwitchCondition, ISwitchNode } from '@/interfaces/database/flow';
|
|
import { NodeProps, Position } from '@xyflow/react';
|
|
import { memo, useCallback } from 'react';
|
|
import { NodeHandleId, SwitchOperatorOptions } from '../../constant';
|
|
import { useGetComponentLabelByValue } from '../../hooks/use-get-begin-query';
|
|
import { CommonHandle } from './handle';
|
|
import { RightHandleStyle } from './handle-icon';
|
|
import NodeHeader from './node-header';
|
|
import { NodeWrapper } from './node-wrapper';
|
|
import { ToolBar } from './toolbar';
|
|
import { useBuildSwitchHandlePositions } from './use-build-switch-handle-positions';
|
|
|
|
const getConditionKey = (idx: number, length: number) => {
|
|
if (idx === 0 && length !== 1) {
|
|
return 'If';
|
|
} else if (idx === length - 1) {
|
|
return 'Else';
|
|
}
|
|
|
|
return 'ElseIf';
|
|
};
|
|
|
|
const ConditionBlock = ({
|
|
condition,
|
|
nodeId,
|
|
}: {
|
|
condition: ISwitchCondition;
|
|
nodeId: string;
|
|
}) => {
|
|
const items = condition?.items ?? [];
|
|
const getLabel = useGetComponentLabelByValue(nodeId);
|
|
|
|
const renderOperatorIcon = useCallback((operator?: string) => {
|
|
const name = SwitchOperatorOptions.find((x) => x.value === operator)?.icon;
|
|
return <IconFont name={name!}></IconFont>;
|
|
}, []);
|
|
|
|
return (
|
|
<Card>
|
|
<CardContent className="p-0 divide-y divide-background-card">
|
|
{items.map((x, idx) => (
|
|
<div key={idx}>
|
|
<section className="flex justify-between gap-2 items-center text-xs p-1">
|
|
<div className="flex-1 truncate text-background-checked">
|
|
{getLabel(x?.cpn_id)}
|
|
</div>
|
|
<span>{renderOperatorIcon(x?.operator)}</span>
|
|
<div className="flex-1 truncate">{x?.value}</div>
|
|
</section>
|
|
</div>
|
|
))}
|
|
</CardContent>
|
|
</Card>
|
|
);
|
|
};
|
|
|
|
function InnerSwitchNode({ id, data, selected }: NodeProps<ISwitchNode>) {
|
|
const { positions } = useBuildSwitchHandlePositions({ data, id });
|
|
return (
|
|
<ToolBar selected={selected} id={id} label={data.label}>
|
|
<NodeWrapper>
|
|
<CommonHandle
|
|
type="target"
|
|
position={Position.Left}
|
|
isConnectable
|
|
nodeId={id}
|
|
id={NodeHandleId.End}
|
|
></CommonHandle>
|
|
<NodeHeader id={id} name={data.name} label={data.label}></NodeHeader>
|
|
<section className="gap-2.5 flex flex-col">
|
|
{positions.map((position, idx) => {
|
|
return (
|
|
<div key={idx}>
|
|
<section className="flex flex-col">
|
|
<div className="flex justify-between">
|
|
<span className="text-text-sub-title text-xs translate-y-2">
|
|
{idx < positions.length - 1 &&
|
|
position.condition?.logical_operator?.toUpperCase()}
|
|
</span>
|
|
<span>{getConditionKey(idx, positions.length)}</span>
|
|
</div>
|
|
{position.condition && (
|
|
<ConditionBlock
|
|
nodeId={id}
|
|
condition={position.condition}
|
|
></ConditionBlock>
|
|
)}
|
|
</section>
|
|
<CommonHandle
|
|
key={position.text}
|
|
id={position.text}
|
|
type="source"
|
|
position={Position.Right}
|
|
isConnectable
|
|
style={{ ...RightHandleStyle, top: position.top }}
|
|
nodeId={id}
|
|
isConnectableEnd={false}
|
|
></CommonHandle>
|
|
</div>
|
|
);
|
|
})}
|
|
</section>
|
|
</NodeWrapper>
|
|
</ToolBar>
|
|
);
|
|
}
|
|
|
|
export const SwitchNode = memo(InnerSwitchNode);
|