<template>
|
<el-drawer
|
:append-to-body="true"
|
v-model="settingVisible"
|
:show-close="false"
|
:size="550"
|
:before-close="saveConfig"
|
class="justify-start"
|
>
|
<template #header>
|
<div class="config-header">
|
<input
|
v-if="showInput"
|
type="text"
|
class="config-editable-input"
|
@blur="blurEvent()"
|
v-mountedFocus
|
v-model="nodeName"
|
:placeholder="nodeName"
|
/>
|
<div v-else class="node-name">
|
{{ nodeName }} <Icon class="ml-1" icon="ep:edit-pen" :size="16" @click="clickIcon()" />
|
</div>
|
<div class="divide-line"></div>
|
</div>
|
</template>
|
<div class="flex flex-items-center mb-3">
|
<span class="font-size-16px mr-3">审批类型 :</span>
|
<el-radio-group v-model="approveType">
|
<el-radio
|
v-for="(item, index) in APPROVE_TYPE"
|
:key="index"
|
:value="item.value"
|
:label="item.value"
|
>
|
{{ item.label }}
|
</el-radio>
|
</el-radio-group>
|
</div>
|
<el-tabs type="border-card" v-model="activeTabName" v-if="approveType === ApproveType.USER">
|
<el-tab-pane label="审批人" name="user">
|
<div>
|
<el-form ref="formRef" :model="configForm" label-position="top" :rules="formRules">
|
<el-form-item label="审批人设置" prop="candidateStrategy">
|
<el-radio-group
|
v-model="configForm.candidateStrategy"
|
@change="changeCandidateStrategy"
|
>
|
<el-radio
|
v-for="(dict, index) in CANDIDATE_STRATEGY"
|
:key="index"
|
:value="dict.value"
|
:label="dict.value"
|
>
|
{{ dict.label }}
|
</el-radio>
|
</el-radio-group>
|
</el-form-item>
|
<el-form-item
|
v-if="configForm.candidateStrategy == CandidateStrategy.ROLE"
|
label="指定角色"
|
prop="roleIds"
|
>
|
<el-select v-model="configForm.roleIds" clearable multiple style="width: 100%">
|
<el-option
|
v-for="item in roleOptions"
|
:key="item.id"
|
:label="item.name"
|
:value="item.id"
|
/>
|
</el-select>
|
</el-form-item>
|
<el-form-item
|
v-if="
|
configForm.candidateStrategy == CandidateStrategy.DEPT_MEMBER ||
|
configForm.candidateStrategy == CandidateStrategy.DEPT_LEADER ||
|
configForm.candidateStrategy == CandidateStrategy.MULTI_LEVEL_DEPT_LEADER
|
"
|
label="指定部门"
|
prop="deptIds"
|
span="24"
|
>
|
<el-tree-select
|
ref="treeRef"
|
v-model="configForm.deptIds"
|
:data="deptTreeOptions"
|
:props="defaultProps"
|
empty-text="加载中,请稍后"
|
multiple
|
node-key="id"
|
:check-strictly="true"
|
style="width: 100%"
|
show-checkbox
|
/>
|
</el-form-item>
|
<el-form-item
|
v-if="configForm.candidateStrategy == CandidateStrategy.POST"
|
label="指定岗位"
|
prop="postIds"
|
span="24"
|
>
|
<el-select v-model="configForm.postIds" clearable multiple style="width: 100%">
|
<el-option
|
v-for="item in postOptions"
|
:key="item.id"
|
:label="item.name"
|
:value="item.id!"
|
/>
|
</el-select>
|
</el-form-item>
|
<el-form-item
|
v-if="configForm.candidateStrategy == CandidateStrategy.USER"
|
label="指定用户"
|
prop="userIds"
|
span="24"
|
>
|
<el-select v-model="configForm.userIds" clearable multiple style="width: 100%">
|
<el-option
|
v-for="item in userOptions"
|
:key="item.id"
|
:label="item.nickname"
|
:value="item.id"
|
/>
|
</el-select>
|
</el-form-item>
|
<el-form-item
|
v-if="configForm.candidateStrategy === CandidateStrategy.USER_GROUP"
|
label="指定用户组"
|
prop="userGroups"
|
>
|
<el-select v-model="configForm.userGroups" clearable multiple style="width: 100%">
|
<el-option
|
v-for="item in userGroupOptions"
|
:key="item.id"
|
:label="item.name"
|
:value="item.id"
|
/>
|
</el-select>
|
</el-form-item>
|
<el-form-item
|
v-if="configForm.candidateStrategy === CandidateStrategy.FORM_USER"
|
label="表单内用户字段"
|
prop="formUser"
|
>
|
<el-select v-model="configForm.formUser" clearable style="width: 100%">
|
<el-option
|
v-for="(item, idx) in userFieldOnFormOptions"
|
:key="idx"
|
:label="item.title"
|
:value="item.field"
|
:disabled ="!item.required"
|
/>
|
</el-select>
|
</el-form-item>
|
<el-form-item
|
v-if="configForm.candidateStrategy === CandidateStrategy.FORM_DEPT_LEADER"
|
label="表单内部门字段"
|
prop="formDept"
|
>
|
<el-select v-model="configForm.formDept" clearable style="width: 100%">
|
<el-option
|
v-for="(item, idx) in deptFieldOnFormOptions"
|
:key="idx"
|
:label="item.title"
|
:value="item.field"
|
:disabled ="!item.required"
|
/>
|
</el-select>
|
</el-form-item>
|
<el-form-item
|
v-if="
|
configForm.candidateStrategy == CandidateStrategy.MULTI_LEVEL_DEPT_LEADER ||
|
configForm.candidateStrategy == CandidateStrategy.START_USER_DEPT_LEADER ||
|
configForm.candidateStrategy ==
|
CandidateStrategy.START_USER_MULTI_LEVEL_DEPT_LEADER ||
|
configForm.candidateStrategy == CandidateStrategy.FORM_DEPT_LEADER
|
"
|
:label="deptLevelLabel!"
|
prop="deptLevel"
|
span="24"
|
>
|
<el-select v-model="configForm.deptLevel" clearable>
|
<el-option
|
v-for="(item, index) in MULTI_LEVEL_DEPT"
|
:key="index"
|
:label="item.label"
|
:value="item.value"
|
/>
|
</el-select>
|
</el-form-item>
|
<!-- TODO @jason:后续要支持选择已经存好的表达式 -->
|
<el-form-item
|
v-if="configForm.candidateStrategy === CandidateStrategy.EXPRESSION"
|
label="流程表达式"
|
prop="expression"
|
>
|
<el-input
|
type="textarea"
|
v-model="configForm.expression"
|
clearable
|
style="width: 100%"
|
/>
|
</el-form-item>
|
<el-form-item label="多人审批方式" prop="approveMethod">
|
<el-radio-group v-model="configForm.approveMethod" @change="approveMethodChanged">
|
<div class="flex-col">
|
<div
|
v-for="(item, index) in APPROVE_METHODS"
|
:key="index"
|
class="flex items-center"
|
>
|
<el-radio :value="item.value" :label="item.value">
|
{{ item.label }}
|
</el-radio>
|
<el-form-item prop="approveRatio">
|
<el-input-number
|
v-model="configForm.approveRatio"
|
:min="10"
|
:max="100"
|
:step="10"
|
size="small"
|
v-if="
|
item.value === ApproveMethodType.APPROVE_BY_RATIO &&
|
configForm.approveMethod === ApproveMethodType.APPROVE_BY_RATIO
|
"
|
/>
|
</el-form-item>
|
</div>
|
</div>
|
</el-radio-group>
|
</el-form-item>
|
|
<el-divider content-position="left">审批人拒绝时</el-divider>
|
<el-form-item prop="rejectHandlerType">
|
<el-radio-group v-model="configForm.rejectHandlerType">
|
<div class="flex-col">
|
<div v-for="(item, index) in REJECT_HANDLER_TYPES" :key="index">
|
<el-radio :key="item.value" :value="item.value" :label="item.label" />
|
</div>
|
</div>
|
</el-radio-group>
|
</el-form-item>
|
<el-form-item
|
v-if="configForm.rejectHandlerType == RejectHandlerType.RETURN_USER_TASK"
|
label="驳回节点"
|
prop="returnNodeId"
|
>
|
<el-select v-model="configForm.returnNodeId" clearable style="width: 100%">
|
<el-option
|
v-for="item in returnTaskList"
|
:key="item.id"
|
:label="item.name"
|
:value="item.id"
|
/>
|
</el-select>
|
</el-form-item>
|
|
<el-divider content-position="left">审批人超时未处理时</el-divider>
|
<el-form-item label="启用开关" prop="timeoutHandlerEnable">
|
<el-switch
|
v-model="configForm.timeoutHandlerEnable"
|
active-text="开启"
|
inactive-text="关闭"
|
@change="timeoutHandlerChange"
|
/>
|
</el-form-item>
|
<el-form-item
|
label="执行动作"
|
prop="timeoutHandlerType"
|
v-if="configForm.timeoutHandlerEnable"
|
>
|
<el-radio-group
|
v-model="configForm.timeoutHandlerType"
|
@change="timeoutHandlerTypeChanged"
|
>
|
<el-radio-button
|
v-for="item in TIMEOUT_HANDLER_TYPES"
|
:key="item.value"
|
:value="item.value"
|
:label="item.label"
|
/>
|
</el-radio-group>
|
</el-form-item>
|
<el-form-item label="超时时间设置" v-if="configForm.timeoutHandlerEnable">
|
<span class="mr-2">当超过</span>
|
<el-form-item prop="timeDuration">
|
<el-input-number
|
class="mr-2"
|
:style="{ width: '100px' }"
|
v-model="configForm.timeDuration"
|
:min="1"
|
controls-position="right"
|
/>
|
</el-form-item>
|
<el-select
|
v-model="timeUnit"
|
class="mr-2"
|
:style="{ width: '100px' }"
|
@change="timeUnitChange"
|
>
|
<el-option
|
v-for="item in TIME_UNIT_TYPES"
|
:key="item.value"
|
:label="item.label"
|
:value="item.value"
|
/>
|
</el-select>
|
未处理
|
</el-form-item>
|
<el-form-item
|
label="最大提醒次数"
|
prop="maxRemindCount"
|
v-if="configForm.timeoutHandlerEnable && configForm.timeoutHandlerType === 1"
|
>
|
<el-input-number v-model="configForm.maxRemindCount" :min="1" :max="10" />
|
</el-form-item>
|
|
<el-divider content-position="left">审批人为空时</el-divider>
|
<el-form-item prop="assignEmptyHandlerType">
|
<el-radio-group v-model="configForm.assignEmptyHandlerType">
|
<div class="flex-col">
|
<div v-for="(item, index) in ASSIGN_EMPTY_HANDLER_TYPES" :key="index">
|
<el-radio :key="item.value" :value="item.value" :label="item.label" />
|
</div>
|
</div>
|
</el-radio-group>
|
</el-form-item>
|
<el-form-item
|
v-if="configForm.assignEmptyHandlerType == AssignEmptyHandlerType.ASSIGN_USER"
|
label="指定用户"
|
prop="assignEmptyHandlerUserIds"
|
span="24"
|
>
|
<el-select
|
v-model="configForm.assignEmptyHandlerUserIds"
|
clearable
|
multiple
|
style="width: 100%"
|
>
|
<el-option
|
v-for="item in userOptions"
|
:key="item.id"
|
:label="item.nickname"
|
:value="item.id"
|
/>
|
</el-select>
|
</el-form-item>
|
|
<el-divider content-position="left">审批人与提交人为同一人时</el-divider>
|
<el-form-item prop="assignStartUserHandlerType">
|
<el-radio-group v-model="configForm.assignStartUserHandlerType">
|
<div class="flex-col">
|
<div v-for="(item, index) in ASSIGN_START_USER_HANDLER_TYPES" :key="index">
|
<el-radio :key="item.value" :value="item.value" :label="item.label" />
|
</div>
|
</div>
|
</el-radio-group>
|
</el-form-item>
|
</el-form>
|
</div>
|
</el-tab-pane>
|
<el-tab-pane label="操作按钮设置" name="buttons">
|
<div class="button-setting-pane">
|
<div class="button-setting-desc">操作按钮</div>
|
<div class="button-setting-title">
|
<div class="button-title-label">操作按钮</div>
|
<div class="pl-4 button-title-label">显示名称</div>
|
<div class="button-title-label">启用</div>
|
</div>
|
<div class="button-setting-item" v-for="(item, index) in buttonsSetting" :key="index">
|
<div class="button-setting-item-label"> {{ OPERATION_BUTTON_NAME.get(item.id) }} </div>
|
<div class="button-setting-item-label">
|
<input
|
type="text"
|
class="editable-title-input"
|
@blur="btnDisplayNameBlurEvent(index)"
|
v-mountedFocus
|
v-model="item.displayName"
|
:placeholder="item.displayName"
|
v-if="btnDisplayNameEdit[index]"
|
/>
|
<el-button v-else text @click="changeBtnDisplayName(index)"
|
>{{ item.displayName }} <Icon icon="ep:edit"
|
/></el-button>
|
</div>
|
<div class="button-setting-item-label">
|
<el-switch v-model="item.enable" />
|
</div>
|
</div>
|
</div>
|
</el-tab-pane>
|
<el-tab-pane label="表单字段权限" name="fields" v-if="formType === 10">
|
<div class="field-setting-pane">
|
<div class="field-setting-desc">字段权限</div>
|
<div class="field-permit-title">
|
<div class="setting-title-label first-title"> 字段名称 </div>
|
<div class="other-titles">
|
<span class="setting-title-label">只读</span>
|
<span class="setting-title-label">可编辑</span>
|
<span class="setting-title-label">隐藏</span>
|
</div>
|
</div>
|
<div
|
class="field-setting-item"
|
v-for="(item, index) in fieldsPermissionConfig"
|
:key="index"
|
>
|
<div class="field-setting-item-label"> {{ item.title }} </div>
|
<el-radio-group class="field-setting-item-group" v-model="item.permission">
|
<div class="item-radio-wrap">
|
<el-radio
|
:value="FieldPermissionType.READ"
|
size="large"
|
:label="FieldPermissionType.READ"
|
><span></span
|
></el-radio>
|
</div>
|
<div class="item-radio-wrap">
|
<el-radio
|
:value="FieldPermissionType.WRITE"
|
size="large"
|
:label="FieldPermissionType.WRITE"
|
><span></span
|
></el-radio>
|
</div>
|
<div class="item-radio-wrap">
|
<el-radio
|
:value="FieldPermissionType.NONE"
|
size="large"
|
:label="FieldPermissionType.NONE"
|
><span></span
|
></el-radio>
|
</div>
|
</el-radio-group>
|
</div>
|
</div>
|
</el-tab-pane>
|
</el-tabs>
|
<template #footer>
|
<el-divider />
|
<div>
|
<el-button type="primary" @click="saveConfig">确 定</el-button>
|
<el-button @click="closeDrawer">取 消</el-button>
|
</div>
|
</template>
|
</el-drawer>
|
</template>
|
|
<script setup lang="ts">
|
import {
|
SimpleFlowNode,
|
APPROVE_TYPE,
|
ApproveType,
|
APPROVE_METHODS,
|
CandidateStrategy,
|
NodeType,
|
ApproveMethodType,
|
TimeUnitType,
|
RejectHandlerType,
|
TIMEOUT_HANDLER_TYPES,
|
TIME_UNIT_TYPES,
|
REJECT_HANDLER_TYPES,
|
DEFAULT_BUTTON_SETTING,
|
OPERATION_BUTTON_NAME,
|
ButtonSetting,
|
MULTI_LEVEL_DEPT,
|
CANDIDATE_STRATEGY,
|
ASSIGN_START_USER_HANDLER_TYPES,
|
TimeoutHandlerType,
|
ASSIGN_EMPTY_HANDLER_TYPES,
|
AssignEmptyHandlerType,
|
FieldPermissionType,
|
ProcessVariableEnum
|
} from '../consts'
|
|
import {
|
useWatchNode,
|
useNodeName,
|
useFormFieldsPermission,
|
useNodeForm,
|
UserTaskFormType,
|
useDrawer
|
} from '../node'
|
import { defaultProps } from '@/utils/tree'
|
import { cloneDeep } from 'lodash-es'
|
import { convertTimeUnit, getApproveTypeText } from '../utils'
|
defineOptions({
|
name: 'UserTaskNodeConfig'
|
})
|
const props = defineProps({
|
flowNode: {
|
type: Object as () => SimpleFlowNode,
|
required: true
|
}
|
})
|
const emits = defineEmits<{
|
'find:returnTaskNodes': [nodeList: SimpleFlowNode[]]
|
}>()
|
const deptLevelLabel = computed(() => {
|
let label = '部门负责人来源'
|
if (configForm.value.candidateStrategy == CandidateStrategy.MULTI_LEVEL_DEPT_LEADER) {
|
label = label + '(指定部门向上)'
|
} else if (configForm.value.candidateStrategy == CandidateStrategy.FORM_DEPT_LEADER) {
|
label = label + '(表单内部门向上)'
|
} else {
|
label = label + '(发起人部门向上)'
|
}
|
return label
|
})
|
// 监控节点的变化
|
const currentNode = useWatchNode(props)
|
// 抽屉配置
|
const { settingVisible, closeDrawer, openDrawer } = useDrawer()
|
// 节点名称配置
|
const { nodeName, showInput, clickIcon, blurEvent } = useNodeName(NodeType.USER_TASK_NODE)
|
// 激活的 Tab 标签页
|
const activeTabName = ref('user')
|
// 表单字段权限设置
|
const { formType, fieldsPermissionConfig, formFieldOptions, getNodeConfigFormFields } =
|
useFormFieldsPermission(FieldPermissionType.READ)
|
// 表单内用户字段选项, 必须是必填和用户选择器
|
const userFieldOnFormOptions = computed(() => {
|
// 固定添加发起人 ID 字段
|
formFieldOptions.unshift({
|
field: ProcessVariableEnum.START_USER_ID,
|
title: '发起人',
|
type: 'UserSelect',
|
required: true
|
})
|
return formFieldOptions.filter((item) => item.type === 'UserSelect')
|
})
|
// 表单内部门字段选项, 必须是必填和部门选择器
|
const deptFieldOnFormOptions = computed(() => {
|
return formFieldOptions.filter((item) => item.type === 'DeptSelect')
|
})
|
// 操作按钮设置
|
const { buttonsSetting, btnDisplayNameEdit, changeBtnDisplayName, btnDisplayNameBlurEvent } =
|
useButtonsSetting()
|
const approveType = ref(ApproveType.USER)
|
// 审批人表单设置
|
const formRef = ref() // 表单 Ref
|
// 表单校验规则
|
const formRules = reactive({
|
candidateStrategy: [{ required: true, message: '审批人设置不能为空', trigger: 'change' }],
|
userIds: [{ required: true, message: '用户不能为空', trigger: 'change' }],
|
roleIds: [{ required: true, message: '角色不能为空', trigger: 'change' }],
|
deptIds: [{ required: true, message: '部门不能为空', trigger: 'change' }],
|
userGroups: [{ required: true, message: '用户组不能为空', trigger: 'change' }],
|
formUser: [{ required: true, message: '表单内用户字段不能为空', trigger: 'change' }],
|
formDept: [{ required: true, message: '表单内部门字段不能为空', trigger: 'change' }],
|
postIds: [{ required: true, message: '岗位不能为空', trigger: 'change' }],
|
expression: [{ required: true, message: '流程表达式不能为空', trigger: 'blur' }],
|
approveMethod: [{ required: true, message: '多人审批方式不能为空', trigger: 'change' }],
|
approveRatio: [{ required: true, message: '通过比例不能为空', trigger: 'blur' }],
|
returnNodeId: [{ required: true, message: '驳回节点不能为空', trigger: 'change' }],
|
timeoutHandlerEnable: [{ required: true }],
|
timeoutHandlerType: [{ required: true }],
|
timeDuration: [{ required: true, message: '超时时间不能为空', trigger: 'blur' }],
|
maxRemindCount: [{ required: true, message: '提醒次数不能为空', trigger: 'blur' }],
|
assignEmptyHandlerType: [{ required: true }],
|
assignEmptyHandlerUserIds: [{ required: true, message: '用户不能为空', trigger: 'change' }],
|
assignStartUserHandlerType: [{ required: true }]
|
})
|
|
const {
|
configForm: tempConfigForm,
|
roleOptions,
|
postOptions,
|
userOptions,
|
userGroupOptions,
|
deptTreeOptions,
|
handleCandidateParam,
|
parseCandidateParam,
|
getShowText
|
} = useNodeForm(NodeType.USER_TASK_NODE)
|
const configForm = tempConfigForm as Ref<UserTaskFormType>
|
|
// 改变审批人设置策略
|
const changeCandidateStrategy = () => {
|
configForm.value.userIds = []
|
configForm.value.deptIds = []
|
configForm.value.roleIds = []
|
configForm.value.postIds = []
|
configForm.value.userGroups = []
|
configForm.value.deptLevel = 1
|
configForm.value.formUser = ''
|
configForm.value.formDept = ''
|
configForm.value.approveMethod = ApproveMethodType.SEQUENTIAL_APPROVE
|
}
|
|
// 审批方式改变
|
const approveMethodChanged = () => {
|
configForm.value.rejectHandlerType = RejectHandlerType.FINISH_PROCESS
|
if (configForm.value.approveMethod === ApproveMethodType.APPROVE_BY_RATIO) {
|
configForm.value.approveRatio = 100
|
}
|
formRef.value.clearValidate('approveRatio')
|
}
|
// 审批拒绝 可退回的节点
|
const returnTaskList = ref<SimpleFlowNode[]>([])
|
// 审批人超时未处理设置
|
const {
|
timeoutHandlerChange,
|
cTimeoutType,
|
timeoutHandlerTypeChanged,
|
timeUnit,
|
timeUnitChange,
|
isoTimeDuration,
|
cTimeoutMaxRemindCount
|
} = useTimeoutHandler()
|
|
// 保存配置
|
const saveConfig = async () => {
|
activeTabName.value = 'user'
|
// 设置审批节点名称
|
currentNode.value.name = nodeName.value!
|
// 设置审批类型
|
currentNode.value.approveType = approveType.value
|
// 如果不是人工审批。返回
|
if (approveType.value !== ApproveType.USER) {
|
currentNode.value.showText = getApproveTypeText(approveType.value)
|
settingVisible.value = false
|
return true
|
}
|
|
if (!formRef) return false
|
const valid = await formRef.value.validate()
|
if (!valid) return false
|
const showText = getShowText()
|
if (!showText) return false
|
|
currentNode.value.candidateStrategy = configForm.value.candidateStrategy
|
// 处理 candidateParam 参数
|
currentNode.value.candidateParam = handleCandidateParam()
|
// 设置审批方式
|
currentNode.value.approveMethod = configForm.value.approveMethod
|
if (configForm.value.approveMethod === ApproveMethodType.APPROVE_BY_RATIO) {
|
currentNode.value.approveRatio = configForm.value.approveRatio
|
}
|
// 设置拒绝处理
|
currentNode.value.rejectHandler = {
|
type: configForm.value.rejectHandlerType!,
|
returnNodeId: configForm.value.returnNodeId
|
}
|
// 设置超时处理
|
currentNode.value.timeoutHandler = {
|
enable: configForm.value.timeoutHandlerEnable!,
|
type: cTimeoutType.value,
|
timeDuration: isoTimeDuration.value,
|
maxRemindCount: cTimeoutMaxRemindCount.value
|
}
|
// 设置审批人为空时
|
currentNode.value.assignEmptyHandler = {
|
type: configForm.value.assignEmptyHandlerType!,
|
userIds:
|
configForm.value.assignEmptyHandlerType === AssignEmptyHandlerType.ASSIGN_USER
|
? configForm.value.assignEmptyHandlerUserIds
|
: undefined
|
}
|
// 设置审批人与发起人相同时
|
currentNode.value.assignStartUserHandlerType = configForm.value.assignStartUserHandlerType
|
// 设置表单权限
|
currentNode.value.fieldsPermission = fieldsPermissionConfig.value
|
// 设置按钮权限
|
currentNode.value.buttonsSetting = buttonsSetting.value
|
|
currentNode.value.showText = showText
|
settingVisible.value = false
|
return true
|
}
|
|
// 显示审批节点配置, 由父组件传过来
|
const showUserTaskNodeConfig = (node: SimpleFlowNode) => {
|
nodeName.value = node.name
|
// 1 审批类型
|
approveType.value = node.approveType ? node.approveType : ApproveType.USER
|
// 如果审批类型不是人工审批返回
|
if (approveType.value !== ApproveType.USER) {
|
return
|
}
|
|
//2.1 审批人设置
|
configForm.value.candidateStrategy = node.candidateStrategy!
|
// 解析候选人参数
|
parseCandidateParam(node.candidateStrategy!, node?.candidateParam)
|
// 2.2 设置审批方式
|
configForm.value.approveMethod = node.approveMethod!
|
if (node.approveMethod == ApproveMethodType.APPROVE_BY_RATIO) {
|
configForm.value.approveRatio = node.approveRatio!
|
}
|
// 2.3 设置审批拒绝处理
|
configForm.value.rejectHandlerType = node.rejectHandler!.type
|
configForm.value.returnNodeId = node.rejectHandler?.returnNodeId
|
const matchNodeList = []
|
emits('find:returnTaskNodes', matchNodeList)
|
returnTaskList.value = matchNodeList
|
// 2.4 设置审批超时处理
|
configForm.value.timeoutHandlerEnable = node.timeoutHandler!.enable
|
if (node.timeoutHandler?.enable && node.timeoutHandler?.timeDuration) {
|
const strTimeDuration = node.timeoutHandler.timeDuration
|
let parseTime = strTimeDuration.slice(2, strTimeDuration.length - 1)
|
let parseTimeUnit = strTimeDuration.slice(strTimeDuration.length - 1)
|
configForm.value.timeDuration = parseInt(parseTime)
|
timeUnit.value = convertTimeUnit(parseTimeUnit)
|
}
|
configForm.value.timeoutHandlerType = node.timeoutHandler?.type
|
configForm.value.maxRemindCount = node.timeoutHandler?.maxRemindCount
|
// 2.5 设置审批人为空时
|
configForm.value.assignEmptyHandlerType = node.assignEmptyHandler?.type
|
configForm.value.assignEmptyHandlerUserIds = node.assignEmptyHandler?.userIds
|
// 2.6 设置用户任务的审批人与发起人相同时
|
configForm.value.assignStartUserHandlerType = node.assignStartUserHandlerType
|
// 3. 操作按钮设置
|
buttonsSetting.value = cloneDeep(node.buttonsSetting) || DEFAULT_BUTTON_SETTING
|
// 4. 表单字段权限配置
|
getNodeConfigFormFields(node.fieldsPermission)
|
}
|
|
defineExpose({ openDrawer, showUserTaskNodeConfig }) // 暴露方法给父组件
|
|
/**
|
* @description 操作按钮设置
|
*/
|
function useButtonsSetting() {
|
const buttonsSetting = ref<ButtonSetting[]>()
|
// 操作按钮显示名称可编辑
|
const btnDisplayNameEdit = ref<boolean[]>([])
|
const changeBtnDisplayName = (index: number) => {
|
btnDisplayNameEdit.value[index] = true
|
}
|
const btnDisplayNameBlurEvent = (index: number) => {
|
btnDisplayNameEdit.value[index] = false
|
const buttonItem = buttonsSetting.value![index]
|
buttonItem.displayName = buttonItem.displayName || OPERATION_BUTTON_NAME.get(buttonItem.id)!
|
}
|
return {
|
buttonsSetting,
|
btnDisplayNameEdit,
|
changeBtnDisplayName,
|
btnDisplayNameBlurEvent
|
}
|
}
|
|
/**
|
* @description 审批人超时未处理配置
|
*/
|
function useTimeoutHandler() {
|
// 时间单位
|
const timeUnit = ref(TimeUnitType.HOUR)
|
|
// 超时开关改变
|
const timeoutHandlerChange = () => {
|
if (configForm.value.timeoutHandlerEnable) {
|
timeUnit.value = 2
|
configForm.value.timeDuration = 6
|
configForm.value.timeoutHandlerType = 1
|
configForm.value.maxRemindCount = 1
|
}
|
}
|
// 超时执行的动作
|
const cTimeoutType = computed(() => {
|
if (!configForm.value.timeoutHandlerEnable) {
|
return undefined
|
}
|
return configForm.value.timeoutHandlerType
|
})
|
|
// 超时处理动作改变
|
const timeoutHandlerTypeChanged = () => {
|
if (configForm.value.timeoutHandlerType === TimeoutHandlerType.REMINDER) {
|
configForm.value.maxRemindCount = 1 // 超时提醒次数,默认为1
|
}
|
}
|
|
// 时间单位改变
|
const timeUnitChange = () => {
|
// 分钟,默认是 60 分钟
|
if (timeUnit.value === TimeUnitType.MINUTE) {
|
configForm.value.timeDuration = 60
|
}
|
// 小时,默认是 6 个小时
|
if (timeUnit.value === TimeUnitType.HOUR) {
|
configForm.value.timeDuration = 6
|
}
|
// 天, 默认 1天
|
if (timeUnit.value === TimeUnitType.DAY) {
|
configForm.value.timeDuration = 1
|
}
|
}
|
// 超时时间的 ISO 表示
|
const isoTimeDuration = computed(() => {
|
if (!configForm.value.timeoutHandlerEnable) {
|
return undefined
|
}
|
let strTimeDuration = 'PT'
|
if (timeUnit.value === TimeUnitType.MINUTE) {
|
strTimeDuration += configForm.value.timeDuration + 'M'
|
}
|
if (timeUnit.value === TimeUnitType.HOUR) {
|
strTimeDuration += configForm.value.timeDuration + 'H'
|
}
|
if (timeUnit.value === TimeUnitType.DAY) {
|
strTimeDuration += configForm.value.timeDuration + 'D'
|
}
|
return strTimeDuration
|
})
|
|
// 超时最大提醒次数
|
const cTimeoutMaxRemindCount = computed(() => {
|
if (!configForm.value.timeoutHandlerEnable) {
|
return undefined
|
}
|
if (configForm.value.timeoutHandlerType !== TimeoutHandlerType.REMINDER) {
|
return undefined
|
}
|
return configForm.value.maxRemindCount
|
})
|
|
return {
|
timeoutHandlerChange,
|
cTimeoutType,
|
timeoutHandlerTypeChanged,
|
timeUnit,
|
timeUnitChange,
|
isoTimeDuration,
|
cTimeoutMaxRemindCount
|
}
|
}
|
</script>
|
|
<style lang="scss" scoped>
|
.button-setting-pane {
|
display: flex;
|
flex-direction: column;
|
font-size: 14px;
|
|
.button-setting-desc {
|
padding-right: 8px;
|
margin-bottom: 16px;
|
font-size: 16px;
|
font-weight: 700;
|
}
|
|
.button-setting-title {
|
display: flex;
|
justify-content: space-between;
|
align-items: center;
|
height: 45px;
|
padding-left: 12px;
|
background-color: #f8fafc0a;
|
border: 1px solid #1f38581a;
|
|
& > :first-child {
|
width: 100px !important;
|
text-align: left !important;
|
}
|
|
& > :last-child {
|
text-align: center !important;
|
}
|
|
.button-title-label {
|
width: 150px;
|
font-size: 13px;
|
font-weight: 700;
|
color: #000;
|
text-align: left;
|
}
|
}
|
|
.button-setting-item {
|
align-items: center;
|
display: flex;
|
justify-content: space-between;
|
height: 38px;
|
padding-left: 12px;
|
border: 1px solid #1f38581a;
|
border-top: 0;
|
|
& > :first-child {
|
width: 100px !important;
|
}
|
|
& > :last-child {
|
text-align: center !important;
|
}
|
|
.button-setting-item-label {
|
width: 150px;
|
overflow: hidden;
|
text-align: left;
|
text-overflow: ellipsis;
|
white-space: nowrap;
|
}
|
|
.editable-title-input {
|
height: 24px;
|
max-width: 130px;
|
margin-left: 4px;
|
line-height: 24px;
|
border: 1px solid #d9d9d9;
|
border-radius: 4px;
|
transition: all 0.3s;
|
|
&:focus {
|
border-color: #40a9ff;
|
outline: 0;
|
box-shadow: 0 0 0 2px rgb(24 144 255 / 20%);
|
}
|
}
|
}
|
}
|
</style>
|