提交 | 用户 | 时间
|
3e359e
|
1 |
<template> |
H |
2 |
<el-drawer |
|
3 |
:append-to-body="true" |
|
4 |
v-model="settingVisible" |
|
5 |
:show-close="false" |
|
6 |
:size="550" |
|
7 |
:before-close="saveConfig" |
|
8 |
class="justify-start" |
|
9 |
> |
|
10 |
<template #header> |
|
11 |
<div class="config-header"> |
|
12 |
<input |
|
13 |
v-if="showInput" |
|
14 |
type="text" |
|
15 |
class="config-editable-input" |
|
16 |
@blur="blurEvent()" |
|
17 |
v-mountedFocus |
|
18 |
v-model="nodeName" |
|
19 |
:placeholder="nodeName" |
|
20 |
/> |
|
21 |
<div v-else class="node-name"> |
|
22 |
{{ nodeName }} <Icon class="ml-1" icon="ep:edit-pen" :size="16" @click="clickIcon()" /> |
|
23 |
</div> |
|
24 |
<div class="divide-line"></div> |
|
25 |
</div> |
|
26 |
</template> |
|
27 |
<div class="flex flex-items-center mb-3"> |
|
28 |
<span class="font-size-16px mr-3">审批类型 :</span> |
|
29 |
<el-radio-group v-model="approveType"> |
|
30 |
<el-radio |
|
31 |
v-for="(item, index) in APPROVE_TYPE" |
|
32 |
:key="index" |
|
33 |
:value="item.value" |
|
34 |
:label="item.value" |
|
35 |
> |
|
36 |
{{ item.label }} |
|
37 |
</el-radio> |
|
38 |
</el-radio-group> |
|
39 |
</div> |
|
40 |
<el-tabs type="border-card" v-model="activeTabName" v-if="approveType === ApproveType.USER"> |
|
41 |
<el-tab-pane label="审批人" name="user"> |
|
42 |
<div> |
|
43 |
<el-form ref="formRef" :model="configForm" label-position="top" :rules="formRules"> |
|
44 |
<el-form-item label="审批人设置" prop="candidateStrategy"> |
|
45 |
<el-radio-group |
|
46 |
v-model="configForm.candidateStrategy" |
|
47 |
@change="changeCandidateStrategy" |
|
48 |
> |
|
49 |
<el-radio |
|
50 |
v-for="(dict, index) in CANDIDATE_STRATEGY" |
|
51 |
:key="index" |
|
52 |
:value="dict.value" |
|
53 |
:label="dict.value" |
|
54 |
> |
|
55 |
{{ dict.label }} |
|
56 |
</el-radio> |
|
57 |
</el-radio-group> |
|
58 |
</el-form-item> |
|
59 |
<el-form-item |
|
60 |
v-if="configForm.candidateStrategy == CandidateStrategy.ROLE" |
|
61 |
label="指定角色" |
|
62 |
prop="roleIds" |
|
63 |
> |
|
64 |
<el-select v-model="configForm.roleIds" clearable multiple style="width: 100%"> |
|
65 |
<el-option |
|
66 |
v-for="item in roleOptions" |
|
67 |
:key="item.id" |
|
68 |
:label="item.name" |
|
69 |
:value="item.id" |
|
70 |
/> |
|
71 |
</el-select> |
|
72 |
</el-form-item> |
|
73 |
<el-form-item |
|
74 |
v-if=" |
|
75 |
configForm.candidateStrategy == CandidateStrategy.DEPT_MEMBER || |
|
76 |
configForm.candidateStrategy == CandidateStrategy.DEPT_LEADER || |
|
77 |
configForm.candidateStrategy == CandidateStrategy.MULTI_LEVEL_DEPT_LEADER |
|
78 |
" |
|
79 |
label="指定部门" |
|
80 |
prop="deptIds" |
|
81 |
span="24" |
|
82 |
> |
|
83 |
<el-tree-select |
|
84 |
ref="treeRef" |
|
85 |
v-model="configForm.deptIds" |
|
86 |
:data="deptTreeOptions" |
|
87 |
:props="defaultProps" |
|
88 |
empty-text="加载中,请稍后" |
|
89 |
multiple |
|
90 |
node-key="id" |
|
91 |
:check-strictly="true" |
|
92 |
style="width: 100%" |
|
93 |
show-checkbox |
|
94 |
/> |
|
95 |
</el-form-item> |
|
96 |
<el-form-item |
|
97 |
v-if="configForm.candidateStrategy == CandidateStrategy.POST" |
|
98 |
label="指定岗位" |
|
99 |
prop="postIds" |
|
100 |
span="24" |
|
101 |
> |
|
102 |
<el-select v-model="configForm.postIds" clearable multiple style="width: 100%"> |
|
103 |
<el-option |
|
104 |
v-for="item in postOptions" |
|
105 |
:key="item.id" |
|
106 |
:label="item.name" |
|
107 |
:value="item.id!" |
|
108 |
/> |
|
109 |
</el-select> |
|
110 |
</el-form-item> |
|
111 |
<el-form-item |
|
112 |
v-if="configForm.candidateStrategy == CandidateStrategy.USER" |
|
113 |
label="指定用户" |
|
114 |
prop="userIds" |
|
115 |
span="24" |
|
116 |
> |
|
117 |
<el-select v-model="configForm.userIds" clearable multiple style="width: 100%"> |
|
118 |
<el-option |
|
119 |
v-for="item in userOptions" |
|
120 |
:key="item.id" |
|
121 |
:label="item.nickname" |
|
122 |
:value="item.id" |
|
123 |
/> |
|
124 |
</el-select> |
|
125 |
</el-form-item> |
|
126 |
<el-form-item |
|
127 |
v-if="configForm.candidateStrategy === CandidateStrategy.USER_GROUP" |
|
128 |
label="指定用户组" |
|
129 |
prop="userGroups" |
|
130 |
> |
|
131 |
<el-select v-model="configForm.userGroups" clearable multiple style="width: 100%"> |
|
132 |
<el-option |
|
133 |
v-for="item in userGroupOptions" |
|
134 |
:key="item.id" |
|
135 |
:label="item.name" |
|
136 |
:value="item.id" |
|
137 |
/> |
|
138 |
</el-select> |
|
139 |
</el-form-item> |
|
140 |
<el-form-item |
|
141 |
v-if="configForm.candidateStrategy === CandidateStrategy.FORM_USER" |
|
142 |
label="表单内用户字段" |
|
143 |
prop="formUser" |
|
144 |
> |
|
145 |
<el-select v-model="configForm.formUser" clearable style="width: 100%"> |
|
146 |
<el-option |
|
147 |
v-for="(item, idx) in userFieldOnFormOptions" |
|
148 |
:key="idx" |
|
149 |
:label="item.title" |
|
150 |
:value="item.field" |
|
151 |
:disabled ="!item.required" |
|
152 |
/> |
|
153 |
</el-select> |
|
154 |
</el-form-item> |
|
155 |
<el-form-item |
|
156 |
v-if="configForm.candidateStrategy === CandidateStrategy.FORM_DEPT_LEADER" |
|
157 |
label="表单内部门字段" |
|
158 |
prop="formDept" |
|
159 |
> |
|
160 |
<el-select v-model="configForm.formDept" clearable style="width: 100%"> |
|
161 |
<el-option |
|
162 |
v-for="(item, idx) in deptFieldOnFormOptions" |
|
163 |
:key="idx" |
|
164 |
:label="item.title" |
|
165 |
:value="item.field" |
|
166 |
:disabled ="!item.required" |
|
167 |
/> |
|
168 |
</el-select> |
|
169 |
</el-form-item> |
|
170 |
<el-form-item |
|
171 |
v-if=" |
|
172 |
configForm.candidateStrategy == CandidateStrategy.MULTI_LEVEL_DEPT_LEADER || |
|
173 |
configForm.candidateStrategy == CandidateStrategy.START_USER_DEPT_LEADER || |
|
174 |
configForm.candidateStrategy == |
|
175 |
CandidateStrategy.START_USER_MULTI_LEVEL_DEPT_LEADER || |
|
176 |
configForm.candidateStrategy == CandidateStrategy.FORM_DEPT_LEADER |
|
177 |
" |
|
178 |
:label="deptLevelLabel!" |
|
179 |
prop="deptLevel" |
|
180 |
span="24" |
|
181 |
> |
|
182 |
<el-select v-model="configForm.deptLevel" clearable> |
|
183 |
<el-option |
|
184 |
v-for="(item, index) in MULTI_LEVEL_DEPT" |
|
185 |
:key="index" |
|
186 |
:label="item.label" |
|
187 |
:value="item.value" |
|
188 |
/> |
|
189 |
</el-select> |
|
190 |
</el-form-item> |
|
191 |
<!-- TODO @jason:后续要支持选择已经存好的表达式 --> |
|
192 |
<el-form-item |
|
193 |
v-if="configForm.candidateStrategy === CandidateStrategy.EXPRESSION" |
|
194 |
label="流程表达式" |
|
195 |
prop="expression" |
|
196 |
> |
|
197 |
<el-input |
|
198 |
type="textarea" |
|
199 |
v-model="configForm.expression" |
|
200 |
clearable |
|
201 |
style="width: 100%" |
|
202 |
/> |
|
203 |
</el-form-item> |
|
204 |
<el-form-item label="多人审批方式" prop="approveMethod"> |
|
205 |
<el-radio-group v-model="configForm.approveMethod" @change="approveMethodChanged"> |
|
206 |
<div class="flex-col"> |
|
207 |
<div |
|
208 |
v-for="(item, index) in APPROVE_METHODS" |
|
209 |
:key="index" |
|
210 |
class="flex items-center" |
|
211 |
> |
|
212 |
<el-radio :value="item.value" :label="item.value"> |
|
213 |
{{ item.label }} |
|
214 |
</el-radio> |
|
215 |
<el-form-item prop="approveRatio"> |
|
216 |
<el-input-number |
|
217 |
v-model="configForm.approveRatio" |
|
218 |
:min="10" |
|
219 |
:max="100" |
|
220 |
:step="10" |
|
221 |
size="small" |
|
222 |
v-if=" |
|
223 |
item.value === ApproveMethodType.APPROVE_BY_RATIO && |
|
224 |
configForm.approveMethod === ApproveMethodType.APPROVE_BY_RATIO |
|
225 |
" |
|
226 |
/> |
|
227 |
</el-form-item> |
|
228 |
</div> |
|
229 |
</div> |
|
230 |
</el-radio-group> |
|
231 |
</el-form-item> |
|
232 |
|
|
233 |
<el-divider content-position="left">审批人拒绝时</el-divider> |
|
234 |
<el-form-item prop="rejectHandlerType"> |
|
235 |
<el-radio-group v-model="configForm.rejectHandlerType"> |
|
236 |
<div class="flex-col"> |
|
237 |
<div v-for="(item, index) in REJECT_HANDLER_TYPES" :key="index"> |
|
238 |
<el-radio :key="item.value" :value="item.value" :label="item.label" /> |
|
239 |
</div> |
|
240 |
</div> |
|
241 |
</el-radio-group> |
|
242 |
</el-form-item> |
|
243 |
<el-form-item |
|
244 |
v-if="configForm.rejectHandlerType == RejectHandlerType.RETURN_USER_TASK" |
|
245 |
label="驳回节点" |
|
246 |
prop="returnNodeId" |
|
247 |
> |
|
248 |
<el-select v-model="configForm.returnNodeId" clearable style="width: 100%"> |
|
249 |
<el-option |
|
250 |
v-for="item in returnTaskList" |
|
251 |
:key="item.id" |
|
252 |
:label="item.name" |
|
253 |
:value="item.id" |
|
254 |
/> |
|
255 |
</el-select> |
|
256 |
</el-form-item> |
|
257 |
|
|
258 |
<el-divider content-position="left">审批人超时未处理时</el-divider> |
|
259 |
<el-form-item label="启用开关" prop="timeoutHandlerEnable"> |
|
260 |
<el-switch |
|
261 |
v-model="configForm.timeoutHandlerEnable" |
|
262 |
active-text="开启" |
|
263 |
inactive-text="关闭" |
|
264 |
@change="timeoutHandlerChange" |
|
265 |
/> |
|
266 |
</el-form-item> |
|
267 |
<el-form-item |
|
268 |
label="执行动作" |
|
269 |
prop="timeoutHandlerType" |
|
270 |
v-if="configForm.timeoutHandlerEnable" |
|
271 |
> |
|
272 |
<el-radio-group |
|
273 |
v-model="configForm.timeoutHandlerType" |
|
274 |
@change="timeoutHandlerTypeChanged" |
|
275 |
> |
|
276 |
<el-radio-button |
|
277 |
v-for="item in TIMEOUT_HANDLER_TYPES" |
|
278 |
:key="item.value" |
|
279 |
:value="item.value" |
|
280 |
:label="item.label" |
|
281 |
/> |
|
282 |
</el-radio-group> |
|
283 |
</el-form-item> |
|
284 |
<el-form-item label="超时时间设置" v-if="configForm.timeoutHandlerEnable"> |
|
285 |
<span class="mr-2">当超过</span> |
|
286 |
<el-form-item prop="timeDuration"> |
|
287 |
<el-input-number |
|
288 |
class="mr-2" |
|
289 |
:style="{ width: '100px' }" |
|
290 |
v-model="configForm.timeDuration" |
|
291 |
:min="1" |
|
292 |
controls-position="right" |
|
293 |
/> |
|
294 |
</el-form-item> |
|
295 |
<el-select |
|
296 |
v-model="timeUnit" |
|
297 |
class="mr-2" |
|
298 |
:style="{ width: '100px' }" |
|
299 |
@change="timeUnitChange" |
|
300 |
> |
|
301 |
<el-option |
|
302 |
v-for="item in TIME_UNIT_TYPES" |
|
303 |
:key="item.value" |
|
304 |
:label="item.label" |
|
305 |
:value="item.value" |
|
306 |
/> |
|
307 |
</el-select> |
|
308 |
未处理 |
|
309 |
</el-form-item> |
|
310 |
<el-form-item |
|
311 |
label="最大提醒次数" |
|
312 |
prop="maxRemindCount" |
|
313 |
v-if="configForm.timeoutHandlerEnable && configForm.timeoutHandlerType === 1" |
|
314 |
> |
|
315 |
<el-input-number v-model="configForm.maxRemindCount" :min="1" :max="10" /> |
|
316 |
</el-form-item> |
|
317 |
|
|
318 |
<el-divider content-position="left">审批人为空时</el-divider> |
|
319 |
<el-form-item prop="assignEmptyHandlerType"> |
|
320 |
<el-radio-group v-model="configForm.assignEmptyHandlerType"> |
|
321 |
<div class="flex-col"> |
|
322 |
<div v-for="(item, index) in ASSIGN_EMPTY_HANDLER_TYPES" :key="index"> |
|
323 |
<el-radio :key="item.value" :value="item.value" :label="item.label" /> |
|
324 |
</div> |
|
325 |
</div> |
|
326 |
</el-radio-group> |
|
327 |
</el-form-item> |
|
328 |
<el-form-item |
|
329 |
v-if="configForm.assignEmptyHandlerType == AssignEmptyHandlerType.ASSIGN_USER" |
|
330 |
label="指定用户" |
|
331 |
prop="assignEmptyHandlerUserIds" |
|
332 |
span="24" |
|
333 |
> |
|
334 |
<el-select |
|
335 |
v-model="configForm.assignEmptyHandlerUserIds" |
|
336 |
clearable |
|
337 |
multiple |
|
338 |
style="width: 100%" |
|
339 |
> |
|
340 |
<el-option |
|
341 |
v-for="item in userOptions" |
|
342 |
:key="item.id" |
|
343 |
:label="item.nickname" |
|
344 |
:value="item.id" |
|
345 |
/> |
|
346 |
</el-select> |
|
347 |
</el-form-item> |
|
348 |
|
|
349 |
<el-divider content-position="left">审批人与提交人为同一人时</el-divider> |
|
350 |
<el-form-item prop="assignStartUserHandlerType"> |
|
351 |
<el-radio-group v-model="configForm.assignStartUserHandlerType"> |
|
352 |
<div class="flex-col"> |
|
353 |
<div v-for="(item, index) in ASSIGN_START_USER_HANDLER_TYPES" :key="index"> |
|
354 |
<el-radio :key="item.value" :value="item.value" :label="item.label" /> |
|
355 |
</div> |
|
356 |
</div> |
|
357 |
</el-radio-group> |
|
358 |
</el-form-item> |
|
359 |
</el-form> |
|
360 |
</div> |
|
361 |
</el-tab-pane> |
|
362 |
<el-tab-pane label="操作按钮设置" name="buttons"> |
|
363 |
<div class="button-setting-pane"> |
|
364 |
<div class="button-setting-desc">操作按钮</div> |
|
365 |
<div class="button-setting-title"> |
|
366 |
<div class="button-title-label">操作按钮</div> |
|
367 |
<div class="pl-4 button-title-label">显示名称</div> |
|
368 |
<div class="button-title-label">启用</div> |
|
369 |
</div> |
|
370 |
<div class="button-setting-item" v-for="(item, index) in buttonsSetting" :key="index"> |
|
371 |
<div class="button-setting-item-label"> {{ OPERATION_BUTTON_NAME.get(item.id) }} </div> |
|
372 |
<div class="button-setting-item-label"> |
|
373 |
<input |
|
374 |
type="text" |
|
375 |
class="editable-title-input" |
|
376 |
@blur="btnDisplayNameBlurEvent(index)" |
|
377 |
v-mountedFocus |
|
378 |
v-model="item.displayName" |
|
379 |
:placeholder="item.displayName" |
|
380 |
v-if="btnDisplayNameEdit[index]" |
|
381 |
/> |
|
382 |
<el-button v-else text @click="changeBtnDisplayName(index)" |
|
383 |
>{{ item.displayName }} <Icon icon="ep:edit" |
|
384 |
/></el-button> |
|
385 |
</div> |
|
386 |
<div class="button-setting-item-label"> |
|
387 |
<el-switch v-model="item.enable" /> |
|
388 |
</div> |
|
389 |
</div> |
|
390 |
</div> |
|
391 |
</el-tab-pane> |
|
392 |
<el-tab-pane label="表单字段权限" name="fields" v-if="formType === 10"> |
|
393 |
<div class="field-setting-pane"> |
|
394 |
<div class="field-setting-desc">字段权限</div> |
|
395 |
<div class="field-permit-title"> |
|
396 |
<div class="setting-title-label first-title"> 字段名称 </div> |
|
397 |
<div class="other-titles"> |
|
398 |
<span class="setting-title-label">只读</span> |
|
399 |
<span class="setting-title-label">可编辑</span> |
|
400 |
<span class="setting-title-label">隐藏</span> |
|
401 |
</div> |
|
402 |
</div> |
|
403 |
<div |
|
404 |
class="field-setting-item" |
|
405 |
v-for="(item, index) in fieldsPermissionConfig" |
|
406 |
:key="index" |
|
407 |
> |
|
408 |
<div class="field-setting-item-label"> {{ item.title }} </div> |
|
409 |
<el-radio-group class="field-setting-item-group" v-model="item.permission"> |
|
410 |
<div class="item-radio-wrap"> |
|
411 |
<el-radio |
|
412 |
:value="FieldPermissionType.READ" |
|
413 |
size="large" |
|
414 |
:label="FieldPermissionType.READ" |
|
415 |
><span></span |
|
416 |
></el-radio> |
|
417 |
</div> |
|
418 |
<div class="item-radio-wrap"> |
|
419 |
<el-radio |
|
420 |
:value="FieldPermissionType.WRITE" |
|
421 |
size="large" |
|
422 |
:label="FieldPermissionType.WRITE" |
|
423 |
><span></span |
|
424 |
></el-radio> |
|
425 |
</div> |
|
426 |
<div class="item-radio-wrap"> |
|
427 |
<el-radio |
|
428 |
:value="FieldPermissionType.NONE" |
|
429 |
size="large" |
|
430 |
:label="FieldPermissionType.NONE" |
|
431 |
><span></span |
|
432 |
></el-radio> |
|
433 |
</div> |
|
434 |
</el-radio-group> |
|
435 |
</div> |
|
436 |
</div> |
|
437 |
</el-tab-pane> |
|
438 |
</el-tabs> |
|
439 |
<template #footer> |
|
440 |
<el-divider /> |
|
441 |
<div> |
|
442 |
<el-button type="primary" @click="saveConfig">确 定</el-button> |
|
443 |
<el-button @click="closeDrawer">取 消</el-button> |
|
444 |
</div> |
|
445 |
</template> |
|
446 |
</el-drawer> |
|
447 |
</template> |
|
448 |
|
|
449 |
<script setup lang="ts"> |
|
450 |
import { |
|
451 |
SimpleFlowNode, |
|
452 |
APPROVE_TYPE, |
|
453 |
ApproveType, |
|
454 |
APPROVE_METHODS, |
|
455 |
CandidateStrategy, |
|
456 |
NodeType, |
|
457 |
ApproveMethodType, |
|
458 |
TimeUnitType, |
|
459 |
RejectHandlerType, |
|
460 |
TIMEOUT_HANDLER_TYPES, |
|
461 |
TIME_UNIT_TYPES, |
|
462 |
REJECT_HANDLER_TYPES, |
|
463 |
DEFAULT_BUTTON_SETTING, |
|
464 |
OPERATION_BUTTON_NAME, |
|
465 |
ButtonSetting, |
|
466 |
MULTI_LEVEL_DEPT, |
|
467 |
CANDIDATE_STRATEGY, |
|
468 |
ASSIGN_START_USER_HANDLER_TYPES, |
|
469 |
TimeoutHandlerType, |
|
470 |
ASSIGN_EMPTY_HANDLER_TYPES, |
|
471 |
AssignEmptyHandlerType, |
9259c2
|
472 |
FieldPermissionType, |
H |
473 |
ProcessVariableEnum |
3e359e
|
474 |
} from '../consts' |
H |
475 |
|
|
476 |
import { |
|
477 |
useWatchNode, |
|
478 |
useNodeName, |
|
479 |
useFormFieldsPermission, |
|
480 |
useNodeForm, |
|
481 |
UserTaskFormType, |
|
482 |
useDrawer |
|
483 |
} from '../node' |
|
484 |
import { defaultProps } from '@/utils/tree' |
|
485 |
import { cloneDeep } from 'lodash-es' |
|
486 |
import { convertTimeUnit, getApproveTypeText } from '../utils' |
|
487 |
defineOptions({ |
|
488 |
name: 'UserTaskNodeConfig' |
|
489 |
}) |
|
490 |
const props = defineProps({ |
|
491 |
flowNode: { |
|
492 |
type: Object as () => SimpleFlowNode, |
|
493 |
required: true |
|
494 |
} |
|
495 |
}) |
|
496 |
const emits = defineEmits<{ |
|
497 |
'find:returnTaskNodes': [nodeList: SimpleFlowNode[]] |
|
498 |
}>() |
|
499 |
const deptLevelLabel = computed(() => { |
|
500 |
let label = '部门负责人来源' |
|
501 |
if (configForm.value.candidateStrategy == CandidateStrategy.MULTI_LEVEL_DEPT_LEADER) { |
|
502 |
label = label + '(指定部门向上)' |
|
503 |
} else if (configForm.value.candidateStrategy == CandidateStrategy.FORM_DEPT_LEADER) { |
|
504 |
label = label + '(表单内部门向上)' |
|
505 |
} else { |
|
506 |
label = label + '(发起人部门向上)' |
|
507 |
} |
|
508 |
return label |
|
509 |
}) |
|
510 |
// 监控节点的变化 |
|
511 |
const currentNode = useWatchNode(props) |
|
512 |
// 抽屉配置 |
|
513 |
const { settingVisible, closeDrawer, openDrawer } = useDrawer() |
|
514 |
// 节点名称配置 |
|
515 |
const { nodeName, showInput, clickIcon, blurEvent } = useNodeName(NodeType.USER_TASK_NODE) |
|
516 |
// 激活的 Tab 标签页 |
|
517 |
const activeTabName = ref('user') |
|
518 |
// 表单字段权限设置 |
|
519 |
const { formType, fieldsPermissionConfig, formFieldOptions, getNodeConfigFormFields } = |
|
520 |
useFormFieldsPermission(FieldPermissionType.READ) |
|
521 |
// 表单内用户字段选项, 必须是必填和用户选择器 |
|
522 |
const userFieldOnFormOptions = computed(() => { |
9259c2
|
523 |
// 固定添加发起人 ID 字段 |
H |
524 |
formFieldOptions.unshift({ |
|
525 |
field: ProcessVariableEnum.START_USER_ID, |
|
526 |
title: '发起人', |
|
527 |
type: 'UserSelect', |
|
528 |
required: true |
|
529 |
}) |
3e359e
|
530 |
return formFieldOptions.filter((item) => item.type === 'UserSelect') |
H |
531 |
}) |
|
532 |
// 表单内部门字段选项, 必须是必填和部门选择器 |
|
533 |
const deptFieldOnFormOptions = computed(() => { |
|
534 |
return formFieldOptions.filter((item) => item.type === 'DeptSelect') |
|
535 |
}) |
|
536 |
// 操作按钮设置 |
|
537 |
const { buttonsSetting, btnDisplayNameEdit, changeBtnDisplayName, btnDisplayNameBlurEvent } = |
|
538 |
useButtonsSetting() |
|
539 |
const approveType = ref(ApproveType.USER) |
|
540 |
// 审批人表单设置 |
|
541 |
const formRef = ref() // 表单 Ref |
|
542 |
// 表单校验规则 |
|
543 |
const formRules = reactive({ |
|
544 |
candidateStrategy: [{ required: true, message: '审批人设置不能为空', trigger: 'change' }], |
|
545 |
userIds: [{ required: true, message: '用户不能为空', trigger: 'change' }], |
|
546 |
roleIds: [{ required: true, message: '角色不能为空', trigger: 'change' }], |
|
547 |
deptIds: [{ required: true, message: '部门不能为空', trigger: 'change' }], |
|
548 |
userGroups: [{ required: true, message: '用户组不能为空', trigger: 'change' }], |
|
549 |
formUser: [{ required: true, message: '表单内用户字段不能为空', trigger: 'change' }], |
|
550 |
formDept: [{ required: true, message: '表单内部门字段不能为空', trigger: 'change' }], |
|
551 |
postIds: [{ required: true, message: '岗位不能为空', trigger: 'change' }], |
|
552 |
expression: [{ required: true, message: '流程表达式不能为空', trigger: 'blur' }], |
|
553 |
approveMethod: [{ required: true, message: '多人审批方式不能为空', trigger: 'change' }], |
|
554 |
approveRatio: [{ required: true, message: '通过比例不能为空', trigger: 'blur' }], |
|
555 |
returnNodeId: [{ required: true, message: '驳回节点不能为空', trigger: 'change' }], |
|
556 |
timeoutHandlerEnable: [{ required: true }], |
|
557 |
timeoutHandlerType: [{ required: true }], |
|
558 |
timeDuration: [{ required: true, message: '超时时间不能为空', trigger: 'blur' }], |
|
559 |
maxRemindCount: [{ required: true, message: '提醒次数不能为空', trigger: 'blur' }], |
|
560 |
assignEmptyHandlerType: [{ required: true }], |
|
561 |
assignEmptyHandlerUserIds: [{ required: true, message: '用户不能为空', trigger: 'change' }], |
|
562 |
assignStartUserHandlerType: [{ required: true }] |
|
563 |
}) |
|
564 |
|
|
565 |
const { |
|
566 |
configForm: tempConfigForm, |
|
567 |
roleOptions, |
|
568 |
postOptions, |
|
569 |
userOptions, |
|
570 |
userGroupOptions, |
|
571 |
deptTreeOptions, |
|
572 |
handleCandidateParam, |
|
573 |
parseCandidateParam, |
|
574 |
getShowText |
|
575 |
} = useNodeForm(NodeType.USER_TASK_NODE) |
|
576 |
const configForm = tempConfigForm as Ref<UserTaskFormType> |
|
577 |
|
|
578 |
// 改变审批人设置策略 |
|
579 |
const changeCandidateStrategy = () => { |
|
580 |
configForm.value.userIds = [] |
|
581 |
configForm.value.deptIds = [] |
|
582 |
configForm.value.roleIds = [] |
|
583 |
configForm.value.postIds = [] |
|
584 |
configForm.value.userGroups = [] |
|
585 |
configForm.value.deptLevel = 1 |
|
586 |
configForm.value.formUser = '' |
|
587 |
configForm.value.formDept = '' |
|
588 |
configForm.value.approveMethod = ApproveMethodType.SEQUENTIAL_APPROVE |
|
589 |
} |
|
590 |
|
|
591 |
// 审批方式改变 |
|
592 |
const approveMethodChanged = () => { |
|
593 |
configForm.value.rejectHandlerType = RejectHandlerType.FINISH_PROCESS |
|
594 |
if (configForm.value.approveMethod === ApproveMethodType.APPROVE_BY_RATIO) { |
|
595 |
configForm.value.approveRatio = 100 |
|
596 |
} |
|
597 |
formRef.value.clearValidate('approveRatio') |
|
598 |
} |
|
599 |
// 审批拒绝 可退回的节点 |
|
600 |
const returnTaskList = ref<SimpleFlowNode[]>([]) |
|
601 |
// 审批人超时未处理设置 |
|
602 |
const { |
|
603 |
timeoutHandlerChange, |
|
604 |
cTimeoutType, |
|
605 |
timeoutHandlerTypeChanged, |
|
606 |
timeUnit, |
|
607 |
timeUnitChange, |
|
608 |
isoTimeDuration, |
|
609 |
cTimeoutMaxRemindCount |
|
610 |
} = useTimeoutHandler() |
|
611 |
|
|
612 |
// 保存配置 |
|
613 |
const saveConfig = async () => { |
|
614 |
activeTabName.value = 'user' |
|
615 |
// 设置审批节点名称 |
|
616 |
currentNode.value.name = nodeName.value! |
|
617 |
// 设置审批类型 |
|
618 |
currentNode.value.approveType = approveType.value |
|
619 |
// 如果不是人工审批。返回 |
|
620 |
if (approveType.value !== ApproveType.USER) { |
|
621 |
currentNode.value.showText = getApproveTypeText(approveType.value) |
|
622 |
settingVisible.value = false |
|
623 |
return true |
|
624 |
} |
|
625 |
|
|
626 |
if (!formRef) return false |
|
627 |
const valid = await formRef.value.validate() |
|
628 |
if (!valid) return false |
|
629 |
const showText = getShowText() |
|
630 |
if (!showText) return false |
|
631 |
|
|
632 |
currentNode.value.candidateStrategy = configForm.value.candidateStrategy |
|
633 |
// 处理 candidateParam 参数 |
|
634 |
currentNode.value.candidateParam = handleCandidateParam() |
|
635 |
// 设置审批方式 |
|
636 |
currentNode.value.approveMethod = configForm.value.approveMethod |
|
637 |
if (configForm.value.approveMethod === ApproveMethodType.APPROVE_BY_RATIO) { |
|
638 |
currentNode.value.approveRatio = configForm.value.approveRatio |
|
639 |
} |
|
640 |
// 设置拒绝处理 |
|
641 |
currentNode.value.rejectHandler = { |
|
642 |
type: configForm.value.rejectHandlerType!, |
|
643 |
returnNodeId: configForm.value.returnNodeId |
|
644 |
} |
|
645 |
// 设置超时处理 |
|
646 |
currentNode.value.timeoutHandler = { |
|
647 |
enable: configForm.value.timeoutHandlerEnable!, |
|
648 |
type: cTimeoutType.value, |
|
649 |
timeDuration: isoTimeDuration.value, |
|
650 |
maxRemindCount: cTimeoutMaxRemindCount.value |
|
651 |
} |
|
652 |
// 设置审批人为空时 |
|
653 |
currentNode.value.assignEmptyHandler = { |
|
654 |
type: configForm.value.assignEmptyHandlerType!, |
|
655 |
userIds: |
|
656 |
configForm.value.assignEmptyHandlerType === AssignEmptyHandlerType.ASSIGN_USER |
|
657 |
? configForm.value.assignEmptyHandlerUserIds |
|
658 |
: undefined |
|
659 |
} |
|
660 |
// 设置审批人与发起人相同时 |
|
661 |
currentNode.value.assignStartUserHandlerType = configForm.value.assignStartUserHandlerType |
|
662 |
// 设置表单权限 |
|
663 |
currentNode.value.fieldsPermission = fieldsPermissionConfig.value |
|
664 |
// 设置按钮权限 |
|
665 |
currentNode.value.buttonsSetting = buttonsSetting.value |
|
666 |
|
|
667 |
currentNode.value.showText = showText |
|
668 |
settingVisible.value = false |
|
669 |
return true |
|
670 |
} |
|
671 |
|
|
672 |
// 显示审批节点配置, 由父组件传过来 |
|
673 |
const showUserTaskNodeConfig = (node: SimpleFlowNode) => { |
|
674 |
nodeName.value = node.name |
|
675 |
// 1 审批类型 |
|
676 |
approveType.value = node.approveType ? node.approveType : ApproveType.USER |
|
677 |
// 如果审批类型不是人工审批返回 |
|
678 |
if (approveType.value !== ApproveType.USER) { |
|
679 |
return |
|
680 |
} |
|
681 |
|
|
682 |
//2.1 审批人设置 |
|
683 |
configForm.value.candidateStrategy = node.candidateStrategy! |
|
684 |
// 解析候选人参数 |
|
685 |
parseCandidateParam(node.candidateStrategy!, node?.candidateParam) |
|
686 |
// 2.2 设置审批方式 |
|
687 |
configForm.value.approveMethod = node.approveMethod! |
|
688 |
if (node.approveMethod == ApproveMethodType.APPROVE_BY_RATIO) { |
|
689 |
configForm.value.approveRatio = node.approveRatio! |
|
690 |
} |
|
691 |
// 2.3 设置审批拒绝处理 |
|
692 |
configForm.value.rejectHandlerType = node.rejectHandler!.type |
|
693 |
configForm.value.returnNodeId = node.rejectHandler?.returnNodeId |
|
694 |
const matchNodeList = [] |
|
695 |
emits('find:returnTaskNodes', matchNodeList) |
|
696 |
returnTaskList.value = matchNodeList |
|
697 |
// 2.4 设置审批超时处理 |
|
698 |
configForm.value.timeoutHandlerEnable = node.timeoutHandler!.enable |
|
699 |
if (node.timeoutHandler?.enable && node.timeoutHandler?.timeDuration) { |
|
700 |
const strTimeDuration = node.timeoutHandler.timeDuration |
|
701 |
let parseTime = strTimeDuration.slice(2, strTimeDuration.length - 1) |
|
702 |
let parseTimeUnit = strTimeDuration.slice(strTimeDuration.length - 1) |
|
703 |
configForm.value.timeDuration = parseInt(parseTime) |
|
704 |
timeUnit.value = convertTimeUnit(parseTimeUnit) |
|
705 |
} |
|
706 |
configForm.value.timeoutHandlerType = node.timeoutHandler?.type |
|
707 |
configForm.value.maxRemindCount = node.timeoutHandler?.maxRemindCount |
|
708 |
// 2.5 设置审批人为空时 |
|
709 |
configForm.value.assignEmptyHandlerType = node.assignEmptyHandler?.type |
|
710 |
configForm.value.assignEmptyHandlerUserIds = node.assignEmptyHandler?.userIds |
|
711 |
// 2.6 设置用户任务的审批人与发起人相同时 |
|
712 |
configForm.value.assignStartUserHandlerType = node.assignStartUserHandlerType |
|
713 |
// 3. 操作按钮设置 |
|
714 |
buttonsSetting.value = cloneDeep(node.buttonsSetting) || DEFAULT_BUTTON_SETTING |
|
715 |
// 4. 表单字段权限配置 |
|
716 |
getNodeConfigFormFields(node.fieldsPermission) |
|
717 |
} |
|
718 |
|
|
719 |
defineExpose({ openDrawer, showUserTaskNodeConfig }) // 暴露方法给父组件 |
|
720 |
|
|
721 |
/** |
|
722 |
* @description 操作按钮设置 |
|
723 |
*/ |
|
724 |
function useButtonsSetting() { |
|
725 |
const buttonsSetting = ref<ButtonSetting[]>() |
|
726 |
// 操作按钮显示名称可编辑 |
|
727 |
const btnDisplayNameEdit = ref<boolean[]>([]) |
|
728 |
const changeBtnDisplayName = (index: number) => { |
|
729 |
btnDisplayNameEdit.value[index] = true |
|
730 |
} |
|
731 |
const btnDisplayNameBlurEvent = (index: number) => { |
|
732 |
btnDisplayNameEdit.value[index] = false |
|
733 |
const buttonItem = buttonsSetting.value![index] |
|
734 |
buttonItem.displayName = buttonItem.displayName || OPERATION_BUTTON_NAME.get(buttonItem.id)! |
|
735 |
} |
|
736 |
return { |
|
737 |
buttonsSetting, |
|
738 |
btnDisplayNameEdit, |
|
739 |
changeBtnDisplayName, |
|
740 |
btnDisplayNameBlurEvent |
|
741 |
} |
|
742 |
} |
|
743 |
|
|
744 |
/** |
|
745 |
* @description 审批人超时未处理配置 |
|
746 |
*/ |
|
747 |
function useTimeoutHandler() { |
|
748 |
// 时间单位 |
|
749 |
const timeUnit = ref(TimeUnitType.HOUR) |
|
750 |
|
|
751 |
// 超时开关改变 |
|
752 |
const timeoutHandlerChange = () => { |
|
753 |
if (configForm.value.timeoutHandlerEnable) { |
|
754 |
timeUnit.value = 2 |
|
755 |
configForm.value.timeDuration = 6 |
|
756 |
configForm.value.timeoutHandlerType = 1 |
|
757 |
configForm.value.maxRemindCount = 1 |
|
758 |
} |
|
759 |
} |
|
760 |
// 超时执行的动作 |
|
761 |
const cTimeoutType = computed(() => { |
|
762 |
if (!configForm.value.timeoutHandlerEnable) { |
|
763 |
return undefined |
|
764 |
} |
|
765 |
return configForm.value.timeoutHandlerType |
|
766 |
}) |
|
767 |
|
|
768 |
// 超时处理动作改变 |
|
769 |
const timeoutHandlerTypeChanged = () => { |
|
770 |
if (configForm.value.timeoutHandlerType === TimeoutHandlerType.REMINDER) { |
|
771 |
configForm.value.maxRemindCount = 1 // 超时提醒次数,默认为1 |
|
772 |
} |
|
773 |
} |
|
774 |
|
|
775 |
// 时间单位改变 |
|
776 |
const timeUnitChange = () => { |
|
777 |
// 分钟,默认是 60 分钟 |
|
778 |
if (timeUnit.value === TimeUnitType.MINUTE) { |
|
779 |
configForm.value.timeDuration = 60 |
|
780 |
} |
|
781 |
// 小时,默认是 6 个小时 |
|
782 |
if (timeUnit.value === TimeUnitType.HOUR) { |
|
783 |
configForm.value.timeDuration = 6 |
|
784 |
} |
|
785 |
// 天, 默认 1天 |
|
786 |
if (timeUnit.value === TimeUnitType.DAY) { |
|
787 |
configForm.value.timeDuration = 1 |
|
788 |
} |
|
789 |
} |
|
790 |
// 超时时间的 ISO 表示 |
|
791 |
const isoTimeDuration = computed(() => { |
|
792 |
if (!configForm.value.timeoutHandlerEnable) { |
|
793 |
return undefined |
|
794 |
} |
|
795 |
let strTimeDuration = 'PT' |
|
796 |
if (timeUnit.value === TimeUnitType.MINUTE) { |
|
797 |
strTimeDuration += configForm.value.timeDuration + 'M' |
|
798 |
} |
|
799 |
if (timeUnit.value === TimeUnitType.HOUR) { |
|
800 |
strTimeDuration += configForm.value.timeDuration + 'H' |
|
801 |
} |
|
802 |
if (timeUnit.value === TimeUnitType.DAY) { |
|
803 |
strTimeDuration += configForm.value.timeDuration + 'D' |
|
804 |
} |
|
805 |
return strTimeDuration |
|
806 |
}) |
|
807 |
|
|
808 |
// 超时最大提醒次数 |
|
809 |
const cTimeoutMaxRemindCount = computed(() => { |
|
810 |
if (!configForm.value.timeoutHandlerEnable) { |
|
811 |
return undefined |
|
812 |
} |
|
813 |
if (configForm.value.timeoutHandlerType !== TimeoutHandlerType.REMINDER) { |
|
814 |
return undefined |
|
815 |
} |
|
816 |
return configForm.value.maxRemindCount |
|
817 |
}) |
|
818 |
|
|
819 |
return { |
|
820 |
timeoutHandlerChange, |
|
821 |
cTimeoutType, |
|
822 |
timeoutHandlerTypeChanged, |
|
823 |
timeUnit, |
|
824 |
timeUnitChange, |
|
825 |
isoTimeDuration, |
|
826 |
cTimeoutMaxRemindCount |
|
827 |
} |
|
828 |
} |
|
829 |
</script> |
|
830 |
|
|
831 |
<style lang="scss" scoped> |
|
832 |
.button-setting-pane { |
|
833 |
display: flex; |
|
834 |
flex-direction: column; |
|
835 |
font-size: 14px; |
|
836 |
|
|
837 |
.button-setting-desc { |
|
838 |
padding-right: 8px; |
|
839 |
margin-bottom: 16px; |
|
840 |
font-size: 16px; |
|
841 |
font-weight: 700; |
|
842 |
} |
|
843 |
|
|
844 |
.button-setting-title { |
|
845 |
display: flex; |
|
846 |
justify-content: space-between; |
|
847 |
align-items: center; |
|
848 |
height: 45px; |
|
849 |
padding-left: 12px; |
|
850 |
background-color: #f8fafc0a; |
|
851 |
border: 1px solid #1f38581a; |
|
852 |
|
|
853 |
& > :first-child { |
|
854 |
width: 100px !important; |
|
855 |
text-align: left !important; |
|
856 |
} |
|
857 |
|
|
858 |
& > :last-child { |
|
859 |
text-align: center !important; |
|
860 |
} |
|
861 |
|
|
862 |
.button-title-label { |
|
863 |
width: 150px; |
|
864 |
font-size: 13px; |
|
865 |
font-weight: 700; |
|
866 |
color: #000; |
|
867 |
text-align: left; |
|
868 |
} |
|
869 |
} |
|
870 |
|
|
871 |
.button-setting-item { |
|
872 |
align-items: center; |
|
873 |
display: flex; |
|
874 |
justify-content: space-between; |
|
875 |
height: 38px; |
|
876 |
padding-left: 12px; |
|
877 |
border: 1px solid #1f38581a; |
|
878 |
border-top: 0; |
|
879 |
|
|
880 |
& > :first-child { |
|
881 |
width: 100px !important; |
|
882 |
} |
|
883 |
|
|
884 |
& > :last-child { |
|
885 |
text-align: center !important; |
|
886 |
} |
|
887 |
|
|
888 |
.button-setting-item-label { |
|
889 |
width: 150px; |
|
890 |
overflow: hidden; |
|
891 |
text-align: left; |
|
892 |
text-overflow: ellipsis; |
|
893 |
white-space: nowrap; |
|
894 |
} |
|
895 |
|
|
896 |
.editable-title-input { |
|
897 |
height: 24px; |
|
898 |
max-width: 130px; |
|
899 |
margin-left: 4px; |
|
900 |
line-height: 24px; |
|
901 |
border: 1px solid #d9d9d9; |
|
902 |
border-radius: 4px; |
|
903 |
transition: all 0.3s; |
|
904 |
|
|
905 |
&:focus { |
|
906 |
border-color: #40a9ff; |
|
907 |
outline: 0; |
|
908 |
box-shadow: 0 0 0 2px rgb(24 144 255 / 20%); |
|
909 |
} |
|
910 |
} |
|
911 |
} |
|
912 |
} |
|
913 |
</style> |