Showing
6 changed files
with
306 additions
and
56 deletions
| @@ -44,6 +44,7 @@ | @@ -44,6 +44,7 @@ | ||
| 44 | }, | 44 | }, |
| 45 | "dependencies": { | 45 | "dependencies": { |
| 46 | "@ant-design/icons": "^5.2.5", | 46 | "@ant-design/icons": "^5.2.5", |
| 47 | + "@qx/flow": "^1.0.0-alpha.23", | ||
| 47 | "@qx/ui": "0.0.3-beta.1", | 48 | "@qx/ui": "0.0.3-beta.1", |
| 48 | "@qx/utils": "0.0.58", | 49 | "@qx/utils": "0.0.58", |
| 49 | "ahooks": "^3.7.5", | 50 | "ahooks": "^3.7.5", |
| @@ -5,13 +5,13 @@ nav: | @@ -5,13 +5,13 @@ nav: | ||
| 5 | order: 1 | 5 | order: 1 |
| 6 | group: | 6 | group: |
| 7 | path: /common | 7 | path: /common |
| 8 | - title: 基础条件配置 | 8 | + title: 条件配置 |
| 9 | order: 0 | 9 | order: 0 |
| 10 | --- | 10 | --- |
| 11 | 11 | ||
| 12 | -## QxBaseCondition 条件配置 | 12 | +## QxBaseCondition 基础条件配置 |
| 13 | 13 | ||
| 14 | -### 条件配置 | 14 | +### 基础条件配置 |
| 15 | 15 | ||
| 16 | ```tsx | 16 | ```tsx |
| 17 | import { QxBaseCondition } from '@qx/common'; | 17 | import { QxBaseCondition } from '@qx/common'; |
| @@ -2,17 +2,11 @@ | @@ -2,17 +2,11 @@ | ||
| 2 | 2 | ||
| 3 | .qx-condition { | 3 | .qx-condition { |
| 4 | width: 100%; | 4 | width: 100%; |
| 5 | - | ||
| 6 | - &-item { | 5 | + |
| 6 | + &-header { | ||
| 7 | display: flex; | 7 | display: flex; |
| 8 | align-items: center; | 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 | import { | 3 | import { |
| 3 | QxBaseCondition, | 4 | QxBaseCondition, |
| 4 | QxBaseConditionValueType, | 5 | QxBaseConditionValueType, |
| 5 | } from '../qx-base-condition'; | 6 | } from '../qx-base-condition'; |
| 7 | +import { QxBaseIcon } from '../qx-base-icon'; | ||
| 6 | import { QxConditionSql, SqlType } from '../qx-condition-sql'; | 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 | export const QxCondition: React.FC<QxConditionProps> = ({ | 22 | export const QxCondition: React.FC<QxConditionProps> = ({ |
| 10 | value, | 23 | value, |
| @@ -14,14 +27,115 @@ export const QxCondition: React.FC<QxConditionProps> = ({ | @@ -14,14 +27,115 @@ export const QxCondition: React.FC<QxConditionProps> = ({ | ||
| 14 | showAssignment = true, | 27 | showAssignment = true, |
| 15 | onChange, | 28 | onChange, |
| 16 | node, | 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 | const handleChange = (opt: Partial<QxConditionValueType>) => { | 42 | const handleChange = (opt: Partial<QxConditionValueType>) => { |
| 20 | onChange?.(Object.assign({}, value, opt, { enabled })); | 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 | return ( | 136 | return ( |
| 24 | - <div className={`qx-condition ${className}`} style={style}> | 137 | + <div className={`qx-condition ${className && className}`} style={style}> |
| 138 | + {showHeader && RenderHeader} | ||
| 25 | <QxBaseCondition | 139 | <QxBaseCondition |
| 26 | mode="condition" | 140 | mode="condition" |
| 27 | nodes={nodes} | 141 | nodes={nodes} |
| @@ -69,7 +183,13 @@ export interface QxConditionProps { | @@ -69,7 +183,13 @@ export interface QxConditionProps { | ||
| 69 | onChange?: (value: QxConditionValueType) => void; | 183 | onChange?: (value: QxConditionValueType) => void; |
| 70 | className?: string; | 184 | className?: string; |
| 71 | style?: React.CSSProperties; | 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,8 +7,6 @@ import { FieldBaseType } from '../qx-base-condition'; | ||
| 7 | import { QxBaseIcon } from '../qx-base-icon'; | 7 | import { QxBaseIcon } from '../qx-base-icon'; |
| 8 | import { request } from '../utils'; | 8 | import { request } from '../utils'; |
| 9 | import './index.less'; | 9 | import './index.less'; |
| 10 | -// import type { FiledType } from '@/interface'; | ||
| 11 | -// import type { INode } from '@qx/flow'; | ||
| 12 | 10 | ||
| 13 | const getAppsFields = (params: string[], appId = 'default') => { | 11 | const getAppsFields = (params: string[], appId = 'default') => { |
| 14 | return request.post(`/qx-apaas-lowcode/app/${appId}/fields`, { | 12 | return request.post(`/qx-apaas-lowcode/app/${appId}/fields`, { |
| @@ -217,13 +215,16 @@ export const useNodeFieldDisplay = ({ | @@ -217,13 +215,16 @@ export const useNodeFieldDisplay = ({ | ||
| 217 | /** | 215 | /** |
| 218 | * 查询所有 formId 的字段,并给 result 添加 child | 216 | * 查询所有 formId 的字段,并给 result 添加 child |
| 219 | */ | 217 | */ |
| 220 | - const handleGetAppsFields = async () => { | 218 | + const handleFormTypeAddChild = async () => { |
| 221 | const forms = findResultByFormId(sourceParentNodes); | 219 | const forms = findResultByFormId(sourceParentNodes); |
| 220 | + if (!forms.length) return sourceParentNodes; | ||
| 221 | + | ||
| 222 | const ids = forms.map((item) => item.qxProps?.formId); | 222 | const ids = forms.map((item) => item.qxProps?.formId); |
| 223 | if (Array.isArray(ids) && ids.length && subset) { | 223 | if (Array.isArray(ids) && ids.length && subset) { |
| 224 | try { | 224 | try { |
| 225 | const data = await getAppsFields(ids as any[]); | 225 | const data = await getAppsFields(ids as any[]); |
| 226 | Object.keys(data).forEach((id) => { | 226 | Object.keys(data).forEach((id) => { |
| 227 | + // 此处是引用类型,会间接修改 sourceParentNodes | ||
| 227 | const form = forms.find((item) => item.qxProps?.formId === id); | 228 | const form = forms.find((item) => item.qxProps?.formId === id); |
| 228 | if (!form) return; | 229 | if (!form) return; |
| 229 | 230 | ||
| @@ -272,12 +273,12 @@ export const useNodeFieldDisplay = ({ | @@ -272,12 +273,12 @@ export const useNodeFieldDisplay = ({ | ||
| 272 | name: fields.title, | 273 | name: fields.title, |
| 273 | code: | 274 | code: |
| 274 | parent && ['ORG', 'FORM', 'USER'].includes(parent.type) | 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 | extract: { | 278 | extract: { |
| 278 | ...(fields.extract || {}), | 279 | ...(fields.extract || {}), |
| 279 | fieldType: fields.type, | 280 | fieldType: fields.type, |
| 280 | - fieldKey: fields.code, | 281 | + fieldKey: fields.id, |
| 281 | }, | 282 | }, |
| 282 | }); | 283 | }); |
| 283 | fields.child = correctionNodeField( | 284 | fields.child = correctionNodeField( |
| @@ -290,43 +291,31 @@ export const useNodeFieldDisplay = ({ | @@ -290,43 +291,31 @@ export const useNodeFieldDisplay = ({ | ||
| 290 | return fields; | 291 | return fields; |
| 291 | }; | 292 | }; |
| 292 | 293 | ||
| 294 | + /** | ||
| 295 | + * 获取可被选择的节点和结果 | ||
| 296 | + */ | ||
| 293 | const getOptionalNodes = async () => { | 297 | const getOptionalNodes = async () => { |
| 294 | - const targetParentNodes = await handleGetAppsFields(); | 298 | + const targetParentNodes = await handleFormTypeAddChild(); // 给 form 类型的字段添加 child |
| 295 | 299 | ||
| 296 | for (let i = 0; i < targetParentNodes.length; i++) { | 300 | for (let i = 0; i < targetParentNodes.length; i++) { |
| 301 | + // TODO: 统一节点和 result 格式 | ||
| 297 | correctionNodeField( | 302 | correctionNodeField( |
| 298 | targetParentNodes[i].data?.result || [], | 303 | targetParentNodes[i].data?.result || [], |
| 299 | targetParentNodes[i].id, | 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 | * 根据 limitType 获取可选择的字段 | 309 | * 根据 limitType 获取可选择的字段 |
| 311 | */ | 310 | */ |
| 312 | function getEffectiveResult(result: FiledType[]) { | 311 | function getEffectiveResult(result: FiledType[]) { |
| 313 | const newResult = []; | 312 | const newResult = []; |
| 313 | + | ||
| 314 | for (let i = 0; i < result.length; i++) { | 314 | for (let i = 0; i < result.length; i++) { |
| 315 | const resultItem = result[i] || {}; | 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 | limitTypes?.includes( | 319 | limitTypes?.includes( |
| 331 | FieldBaseType[resultItem.type as keyof typeof FieldBaseType] || | 320 | FieldBaseType[resultItem.type as keyof typeof FieldBaseType] || |
| 332 | resultItem.type, | 321 | resultItem.type, |
| @@ -334,6 +323,10 @@ export const useNodeFieldDisplay = ({ | @@ -334,6 +323,10 @@ export const useNodeFieldDisplay = ({ | ||
| 334 | ) { | 323 | ) { |
| 335 | newResult.push(resultItem); | 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 | return newResult; | 332 | return newResult; |
| @@ -351,9 +344,15 @@ export const useNodeFieldDisplay = ({ | @@ -351,9 +344,15 @@ export const useNodeFieldDisplay = ({ | ||
| 351 | return nodes; | 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 | const getResultFieldMaps = (optionalNodes: INode[]) => { | 358 | const getResultFieldMaps = (optionalNodes: INode[]) => { |
| @@ -470,13 +469,18 @@ const SelectItem = (props: any) => { | @@ -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 | const [visible, setVisible] = useState(false); | 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 | const getOptions = () => { | 485 | const getOptions = () => { |
| 482 | return optionalNodes.map((node) => ({ | 486 | return optionalNodes.map((node) => ({ |
| @@ -528,8 +532,23 @@ export const QxFlowNodeFieldSelector: React.FC<NodeFieldSelectProps> = ( | @@ -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 | return ( | 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 | <Dropdown | 552 | <Dropdown |
| 534 | destroyPopupOnHide | 553 | destroyPopupOnHide |
| 535 | trigger={['click']} | 554 | trigger={['click']} |
| @@ -556,7 +575,7 @@ export const QxFlowNodeFieldSelector: React.FC<NodeFieldSelectProps> = ( | @@ -556,7 +575,7 @@ export const QxFlowNodeFieldSelector: React.FC<NodeFieldSelectProps> = ( | ||
| 556 | </Dropdown> | 575 | </Dropdown> |
| 557 | </div> | 576 | </div> |
| 558 | ); | 577 | ); |
| 559 | -}; | 578 | +}); |
| 560 | 579 | ||
| 561 | export interface NodeFieldSelectProps { | 580 | export interface NodeFieldSelectProps { |
| 562 | node: INode; | 581 | node: INode; |