Showing
6 changed files
with
306 additions
and
56 deletions
... | ... | @@ -5,13 +5,13 @@ nav: |
5 | 5 | order: 1 |
6 | 6 | group: |
7 | 7 | path: /common |
8 | - title: 基础条件配置 | |
8 | + title: 条件配置 | |
9 | 9 | order: 0 |
10 | 10 | --- |
11 | 11 | |
12 | -## QxBaseCondition 条件配置 | |
12 | +## QxBaseCondition 基础条件配置 | |
13 | 13 | |
14 | -### 条件配置 | |
14 | +### 基础条件配置 | |
15 | 15 | |
16 | 16 | ```tsx |
17 | 17 | import { QxBaseCondition } from '@qx/common'; | ... | ... |
... | ... | @@ -2,17 +2,11 @@ |
2 | 2 | |
3 | 3 | .qx-condition { |
4 | 4 | width: 100%; |
5 | - | |
6 | - &-item { | |
5 | + | |
6 | + &-header { | |
7 | 7 | display: flex; |
8 | 8 | align-items: center; |
9 | - | |
10 | - | |
11 | - &__idx { | |
12 | - color: @N9; | |
13 | - margin-right: 4px; | |
14 | - width: 24px; | |
15 | - display: inline-block; | |
16 | - } | |
9 | + justify-content: space-between; | |
10 | + margin-bottom: 16px; | |
17 | 11 | } |
18 | 12 | } | ... | ... |
1 | -import React, { useEffect } from 'react'; | |
1 | +import { Button } from 'antd'; | |
2 | +import React, { useEffect, useRef, useState } from 'react'; | |
2 | 3 | import { |
3 | 4 | QxBaseCondition, |
4 | 5 | QxBaseConditionValueType, |
5 | 6 | } from '../qx-base-condition'; |
7 | +import { QxBaseIcon } from '../qx-base-icon'; | |
6 | 8 | import { QxConditionSql, SqlType } from '../qx-condition-sql'; |
7 | -import { INode } from '../qx-flow-node-selector'; | |
9 | +import QxFieldPopover from '../qx-field/src/popover'; | |
10 | +import FieldsCheckboxGroup from '../qx-field/src/popover/checkbox'; | |
11 | +import { INode, QxFlowNodeFieldSelector } from '../qx-flow-node-selector'; | |
12 | +import { request } from '../utils'; | |
13 | + | |
14 | +import './index.less'; | |
15 | + | |
16 | +const excludeCodes = ['id']; | |
17 | + | |
18 | +export function getFieldsByFunId(funId: string, params?: { useKey: boolean }) { | |
19 | + return request.get(`/qx-apaas-lowcode/app/form/${funId}/field`, { params }); | |
20 | +} | |
8 | 21 | |
9 | 22 | export const QxCondition: React.FC<QxConditionProps> = ({ |
10 | 23 | value, |
... | ... | @@ -14,14 +27,115 @@ export const QxCondition: React.FC<QxConditionProps> = ({ |
14 | 27 | showAssignment = true, |
15 | 28 | onChange, |
16 | 29 | node, |
17 | - nodes | |
30 | + nodes, | |
31 | + header, | |
32 | + headerLeft, | |
33 | + headerRight, | |
34 | + formId, | |
35 | + multiple, | |
36 | + showHeader = false | |
18 | 37 | }) => { |
38 | + const [fields, setFields] = useState([]); | |
39 | + | |
40 | + const fieldSelectorRef = useRef(); | |
41 | + | |
19 | 42 | const handleChange = (opt: Partial<QxConditionValueType>) => { |
20 | 43 | onChange?.(Object.assign({}, value, opt, { enabled })); |
21 | 44 | }; |
22 | 45 | |
46 | + const handleGetFieldsList = async (formId: string) => { | |
47 | + try { | |
48 | + const data = await getFieldsByFunId(formId); | |
49 | + setFields(data.filter((i) => !excludeCodes.includes(i.code))); | |
50 | + } catch (error) { | |
51 | + setFields([]); | |
52 | + } | |
53 | + }; | |
54 | + | |
55 | + useEffect(() => { | |
56 | + if (formId) handleGetFieldsList(formId); | |
57 | + }, [formId]); | |
58 | + | |
59 | + const PopoverComponent = multiple ? FieldsCheckboxGroup : QxFieldPopover; | |
60 | + | |
61 | + const handleAddField = (val: any, opt: any) => { | |
62 | + handleChange({ | |
63 | + operators: [ | |
64 | + ...(value?.operators || []), | |
65 | + { | |
66 | + field: opt, | |
67 | + code: opt.code, | |
68 | + name: opt.name, | |
69 | + type: opt.extract?.fieldType, | |
70 | + opt: 'IS', | |
71 | + mappingValues: [], | |
72 | + }, | |
73 | + ], | |
74 | + }); | |
75 | + // setFieldValue( | |
76 | + // 'condition', | |
77 | + // Object.assign({}, condition, { | |
78 | + // operators: [ | |
79 | + // ...(condition.operators || []), | |
80 | + // { field: opt, code: val }, | |
81 | + // ], | |
82 | + // }), | |
83 | + // ); | |
84 | + }; | |
85 | + | |
86 | + // const fieldOpt = useMemo(() => { | |
87 | + // return fieldSelectorRef.current?.fieldMap || {}; | |
88 | + // }, []); | |
89 | + | |
90 | + const RenderHeader = header ? ( | |
91 | + header | |
92 | + ) : ( | |
93 | + <div className="qx-condition-header"> | |
94 | + {!headerLeft ? ( | |
95 | + <div className="qx-condition-header__title">条件配置</div> | |
96 | + ) : ( | |
97 | + headerLeft | |
98 | + )} | |
99 | + {!headerRight ? ( | |
100 | + <div className="qx-condition-header__right"> | |
101 | + {node ? ( | |
102 | + <QxFlowNodeFieldSelector | |
103 | + ref={fieldSelectorRef} | |
104 | + node={node} | |
105 | + nodes={nodes!} | |
106 | + mode="variable" | |
107 | + onChange={handleAddField} | |
108 | + > | |
109 | + <Button size="small" type="link"> | |
110 | + <QxBaseIcon style={{ fontSize: 16 }} type="qx-icon-plus" /> | |
111 | + 添加字段 | |
112 | + </Button> | |
113 | + </QxFlowNodeFieldSelector> | |
114 | + ) : ( | |
115 | + <PopoverComponent | |
116 | + ref={fieldSelectorRef} | |
117 | + data={fields} | |
118 | + onSelect={handleAddField} | |
119 | + onChange={handleAddField} | |
120 | + > | |
121 | + <Button size="small" type="link"> | |
122 | + <QxBaseIcon style={{ fontSize: 16 }} type="qx-icon-plus" /> | |
123 | + 添加字段 | |
124 | + </Button> | |
125 | + </PopoverComponent> | |
126 | + )} | |
127 | + </div> | |
128 | + ) : ( | |
129 | + headerRight | |
130 | + )} | |
131 | + </div> | |
132 | + ); | |
133 | + | |
134 | + console.log('operators', fieldSelectorRef); | |
135 | + | |
23 | 136 | return ( |
24 | - <div className={`qx-condition ${className}`} style={style}> | |
137 | + <div className={`qx-condition ${className && className}`} style={style}> | |
138 | + {showHeader && RenderHeader} | |
25 | 139 | <QxBaseCondition |
26 | 140 | mode="condition" |
27 | 141 | nodes={nodes} |
... | ... | @@ -69,7 +183,13 @@ export interface QxConditionProps { |
69 | 183 | onChange?: (value: QxConditionValueType) => void; |
70 | 184 | className?: string; |
71 | 185 | style?: React.CSSProperties; |
72 | - node?: INode | |
73 | - nodes?: INode[] | |
74 | - showAssignment?: boolean | |
186 | + node?: INode; | |
187 | + nodes?: INode[]; | |
188 | + showAssignment?: boolean; | |
189 | + header?: React.ReactNode; | |
190 | + headerRight?: React.ReactNode; | |
191 | + headerLeft?: React.ReactNode; | |
192 | + multiple?: boolean; | |
193 | + formId?: string; | |
194 | + showHeader?: boolean; | |
75 | 195 | } | ... | ... |
src/qx-flow-node-selector/index.md
0 → 100644
1 | +--- | |
2 | +nav: | |
3 | + path: /component | |
4 | + title: 组件 | |
5 | + order: 1 | |
6 | +group: | |
7 | + path: /flow | |
8 | + title: 流程 | |
9 | + order: 0 | |
10 | +--- | |
11 | + | |
12 | +## QxFlowNodeFieldSelector 流程结果集选择器 | |
13 | + | |
14 | +### 普通 | |
15 | + | |
16 | +```tsx | |
17 | +import { QxFlowNodeFieldSelector } from '@qx/common'; | |
18 | + | |
19 | +const node = { | |
20 | + id: 'dfc29d5b64fa42489a65b3cfaeb999da', | |
21 | + type: 'default_DF_CONDITION', | |
22 | + name: '条件', | |
23 | + iconColor: '#F77234', | |
24 | + data: {}, | |
25 | + children: [], | |
26 | + previousId: 'b0a2c7925a7b45e884fe8f82a0f39e9b', | |
27 | +}; | |
28 | + | |
29 | +const nodes = [ | |
30 | + { | |
31 | + id: '4c4fc5213db149808c57d093b15e6295', | |
32 | + name: '开始', | |
33 | + type: 'default_DF_START', | |
34 | + data: { | |
35 | + nodeVersion: '3.0.0', | |
36 | + data: { | |
37 | + enablePropagation: false, | |
38 | + propagation: 'REQUIRED', | |
39 | + isolation: 'REPEATABLE_READ', | |
40 | + }, | |
41 | + result: [ | |
42 | + { | |
43 | + id: '9911c21704ba4d8da8651185f441f9d3', | |
44 | + code: '5sfuiz', | |
45 | + type: 'OBJECT', | |
46 | + title: '5sfuiz', | |
47 | + qxProps: {}, | |
48 | + pid: '', | |
49 | + description: '', | |
50 | + valueMapping: { mappingValues: [] }, | |
51 | + valuesObj: [], | |
52 | + child: [ | |
53 | + { | |
54 | + id: '022c007b4c304c58850ed592ef5c1774', | |
55 | + type: 'STRING', | |
56 | + pid: '9911c21704ba4d8da8651185f441f9d3', | |
57 | + code: 'gcc8qd', | |
58 | + title: 'gcc8qd', | |
59 | + }, | |
60 | + ], | |
61 | + }, | |
62 | + { | |
63 | + id: '6289083b52474567aba6d3f5fe9eda90', | |
64 | + code: '3coizb', | |
65 | + type: 'FORM', | |
66 | + title: '3coizb', | |
67 | + qxProps: { | |
68 | + appId: 'HQIXKC0dxbuYENalZzP', | |
69 | + formId: 'VX1TdanWSgYrKYn3vT8', | |
70 | + isTree: false, | |
71 | + }, | |
72 | + pid: '', | |
73 | + }, | |
74 | + ], | |
75 | + }, | |
76 | + children: [], | |
77 | + }, | |
78 | + { | |
79 | + id: 'b0a2c7925a7b45e884fe8f82a0f39e9b', | |
80 | + name: '分支', | |
81 | + type: 'default_DF_BRANCH', | |
82 | + previousId: '4c4fc5213db149808c57d093b15e6295', | |
83 | + children: [ | |
84 | + { | |
85 | + id: 'dfc29d5b64fa42489a65b3cfaeb999da', | |
86 | + type: 'default_DF_CONDITION', | |
87 | + previousId: 'b0a2c7925a7b45e884fe8f82a0f39e9b', | |
88 | + name: '条件', | |
89 | + data: {}, | |
90 | + children: [], | |
91 | + }, | |
92 | + { | |
93 | + id: '003ca991f70548a3ba8c4f9b8d0daad2', | |
94 | + name: '条件', | |
95 | + type: 'default_DF_CONDITION', | |
96 | + previousId: 'b0a2c7925a7b45e884fe8f82a0f39e9b', | |
97 | + children: [], | |
98 | + }, | |
99 | + ], | |
100 | + }, | |
101 | + { | |
102 | + id: '576f817ce67846318d1132f231128f05', | |
103 | + name: '结束', | |
104 | + previousId: 'b0a2c7925a7b45e884fe8f82a0f39e9b', | |
105 | + type: 'default_DF_END', | |
106 | + data: { nodeVersion: '3.0.0' }, | |
107 | + children: [], | |
108 | + }, | |
109 | +]; | |
110 | + | |
111 | +export default () => { | |
112 | + return <QxFlowNodeFieldSelector node={node} nodes={nodes} />; | |
113 | +}; | |
114 | +``` | |
115 | + | |
116 | +<API id="NodeFieldSelectProps"></API> | ... | ... |
... | ... | @@ -7,8 +7,6 @@ import { FieldBaseType } from '../qx-base-condition'; |
7 | 7 | import { QxBaseIcon } from '../qx-base-icon'; |
8 | 8 | import { request } from '../utils'; |
9 | 9 | import './index.less'; |
10 | -// import type { FiledType } from '@/interface'; | |
11 | -// import type { INode } from '@qx/flow'; | |
12 | 10 | |
13 | 11 | const getAppsFields = (params: string[], appId = 'default') => { |
14 | 12 | return request.post(`/qx-apaas-lowcode/app/${appId}/fields`, { |
... | ... | @@ -217,13 +215,16 @@ export const useNodeFieldDisplay = ({ |
217 | 215 | /** |
218 | 216 | * 查询所有 formId 的字段,并给 result 添加 child |
219 | 217 | */ |
220 | - const handleGetAppsFields = async () => { | |
218 | + const handleFormTypeAddChild = async () => { | |
221 | 219 | const forms = findResultByFormId(sourceParentNodes); |
220 | + if (!forms.length) return sourceParentNodes; | |
221 | + | |
222 | 222 | const ids = forms.map((item) => item.qxProps?.formId); |
223 | 223 | if (Array.isArray(ids) && ids.length && subset) { |
224 | 224 | try { |
225 | 225 | const data = await getAppsFields(ids as any[]); |
226 | 226 | Object.keys(data).forEach((id) => { |
227 | + // 此处是引用类型,会间接修改 sourceParentNodes | |
227 | 228 | const form = forms.find((item) => item.qxProps?.formId === id); |
228 | 229 | if (!form) return; |
229 | 230 | |
... | ... | @@ -272,12 +273,12 @@ export const useNodeFieldDisplay = ({ |
272 | 273 | name: fields.title, |
273 | 274 | code: |
274 | 275 | parent && ['ORG', 'FORM', 'USER'].includes(parent.type) |
275 | - ? `${parent.code}.${fields.code}` | |
276 | - : `${nodeId}|${fields.code}`, | |
276 | + ? `${parent.code}.${fields.id}` | |
277 | + : `${nodeId}|${fields.id}`, | |
277 | 278 | extract: { |
278 | 279 | ...(fields.extract || {}), |
279 | 280 | fieldType: fields.type, |
280 | - fieldKey: fields.code, | |
281 | + fieldKey: fields.id, | |
281 | 282 | }, |
282 | 283 | }); |
283 | 284 | fields.child = correctionNodeField( |
... | ... | @@ -290,43 +291,31 @@ export const useNodeFieldDisplay = ({ |
290 | 291 | return fields; |
291 | 292 | }; |
292 | 293 | |
294 | + /** | |
295 | + * 获取可被选择的节点和结果 | |
296 | + */ | |
293 | 297 | const getOptionalNodes = async () => { |
294 | - const targetParentNodes = await handleGetAppsFields(); | |
298 | + const targetParentNodes = await handleFormTypeAddChild(); // 给 form 类型的字段添加 child | |
295 | 299 | |
296 | 300 | for (let i = 0; i < targetParentNodes.length; i++) { |
301 | + // TODO: 统一节点和 result 格式 | |
297 | 302 | correctionNodeField( |
298 | 303 | targetParentNodes[i].data?.result || [], |
299 | 304 | targetParentNodes[i].id, |
300 | 305 | ); |
301 | 306 | } |
302 | 307 | |
303 | - if (!limitTypes || !limitTypes.length) { | |
304 | - setOptionalNodes(targetParentNodes); | |
305 | - renderInputDisplay(targetParentNodes); | |
306 | - return; | |
307 | - } | |
308 | - | |
309 | 308 | /** |
310 | 309 | * 根据 limitType 获取可选择的字段 |
311 | 310 | */ |
312 | 311 | function getEffectiveResult(result: FiledType[]) { |
313 | 312 | const newResult = []; |
313 | + | |
314 | 314 | for (let i = 0; i < result.length; i++) { |
315 | 315 | const resultItem = result[i] || {}; |
316 | - // correctionNodeField(resultItem, nodeId, parent); | |
317 | 316 | |
318 | - if (resultItem.child) { | |
319 | - resultItem.child = getEffectiveResult(resultItem.child); | |
320 | - if ( | |
321 | - (Array.isArray(resultItem.child) && resultItem.child.length) || | |
322 | - limitTypes?.includes( | |
323 | - FieldBaseType[resultItem.type as keyof typeof FieldBaseType] || | |
324 | - resultItem.type, | |
325 | - ) | |
326 | - ) { | |
327 | - newResult.push(resultItem); | |
328 | - } | |
329 | - } else if ( | |
317 | + if ( | |
318 | + // 先将表单字段类型或流程参数类型转换为基础类型再做过滤 | |
330 | 319 | limitTypes?.includes( |
331 | 320 | FieldBaseType[resultItem.type as keyof typeof FieldBaseType] || |
332 | 321 | resultItem.type, |
... | ... | @@ -334,6 +323,10 @@ export const useNodeFieldDisplay = ({ |
334 | 323 | ) { |
335 | 324 | newResult.push(resultItem); |
336 | 325 | } |
326 | + | |
327 | + if (Array.isArray(resultItem.child) && resultItem.child.length) { | |
328 | + resultItem.child = getEffectiveResult(resultItem.child); | |
329 | + } | |
337 | 330 | } |
338 | 331 | |
339 | 332 | return newResult; |
... | ... | @@ -351,9 +344,15 @@ export const useNodeFieldDisplay = ({ |
351 | 344 | return nodes; |
352 | 345 | } |
353 | 346 | |
354 | - const newNodes = getEffectiveNodes(targetParentNodes); | |
355 | - setOptionalNodes([...newNodes]); | |
356 | - renderInputDisplay([...newNodes]); | |
347 | + let newNodes = targetParentNodes; | |
348 | + | |
349 | + // 有类型限制根据 limitType 筛选出可选的节点和 result | |
350 | + if (limitTypes && Array.isArray(limitTypes) && limitTypes.length) { | |
351 | + newNodes = getEffectiveNodes(targetParentNodes); | |
352 | + } | |
353 | + | |
354 | + setOptionalNodes(newNodes); | |
355 | + renderInputDisplay(newNodes); | |
357 | 356 | }; |
358 | 357 | |
359 | 358 | const getResultFieldMaps = (optionalNodes: INode[]) => { |
... | ... | @@ -470,13 +469,18 @@ const SelectItem = (props: any) => { |
470 | 469 | ); |
471 | 470 | }; |
472 | 471 | |
473 | -export const QxFlowNodeFieldSelector: React.FC<NodeFieldSelectProps> = ( | |
474 | - props, | |
475 | -) => { | |
472 | +export const QxFlowNodeFieldSelector = React.forwardRef< | |
473 | + any, | |
474 | + NodeFieldSelectProps | |
475 | +>((props, ref) => { | |
476 | 476 | const [visible, setVisible] = useState(false); |
477 | 477 | |
478 | - const { optionalNodes, renderInputDisplay, inputDisplay } = | |
479 | - useNodeFieldDisplay(props); | |
478 | + const { | |
479 | + optionalNodes, | |
480 | + renderInputDisplay, | |
481 | + inputDisplay, | |
482 | + } = useNodeFieldDisplay(props); | |
483 | + // console.log('111111', props, optionalNodes); | |
480 | 484 | |
481 | 485 | const getOptions = () => { |
482 | 486 | return optionalNodes.map((node) => ({ |
... | ... | @@ -528,8 +532,23 @@ export const QxFlowNodeFieldSelector: React.FC<NodeFieldSelectProps> = ( |
528 | 532 | } |
529 | 533 | }; |
530 | 534 | |
535 | + useEffect(() => { | |
536 | + const closeDropdown = () => { | |
537 | + setVisible(false); | |
538 | + }; | |
539 | + document.body.addEventListener('click', closeDropdown); | |
540 | + return () => { | |
541 | + document.body.removeEventListener('click', closeDropdown); | |
542 | + }; | |
543 | + }, []); | |
544 | + | |
531 | 545 | return ( |
532 | - <div className={cls('qx-node-select')}> | |
546 | + <div | |
547 | + className={cls('qx-node-select')} | |
548 | + onClick={(e) => { | |
549 | + e.stopPropagation(); | |
550 | + }} | |
551 | + > | |
533 | 552 | <Dropdown |
534 | 553 | destroyPopupOnHide |
535 | 554 | trigger={['click']} |
... | ... | @@ -556,7 +575,7 @@ export const QxFlowNodeFieldSelector: React.FC<NodeFieldSelectProps> = ( |
556 | 575 | </Dropdown> |
557 | 576 | </div> |
558 | 577 | ); |
559 | -}; | |
578 | +}); | |
560 | 579 | |
561 | 580 | export interface NodeFieldSelectProps { |
562 | 581 | node: INode; | ... | ... |