Merge branch 'develop' of https://e.coding.net/ctcdevteam/hberp/hberp into develop

This commit is contained in:
caoqianming 2021-10-18 11:00:48 +08:00
commit 96b40f101d
9 changed files with 932 additions and 413 deletions

View File

@ -163,13 +163,45 @@ export function ticketAccpet(id,data) {
}) })
} }
//撤回工单,允许创建人在指定状态撤回工单至初始状态 //撤回工单,允许创建人在指定状态撤回工单至初始状态
export function getTicketRetreat(id,data) { export function ticketRetreat(id,data) {
return request({ return request({
url: `/wf/ticket/${id}/retreat/`, url: `/wf/ticket/${id}/retreat/`,
method: 'post', method: 'post',
data data
}) })
} }
//关闭工单,仅允许创建人在初始状态关闭工单
export function ticketAddNode(id,data) {
return request({
url: `/wf/ticket/${id}/add_node/`,
method: 'post',
data
})
}
//加签
export function ticketClose(id,data) {
return request({
url: `/wf/ticket/${id}/close/`,
method: 'post',
data
})
}
//加签
export function ticketAddNodeEnd(id,data) {
return request({
url: `/wf/ticket/${id}/add_node_end/`,
method: 'post',
data
})
}
//工单删除
export function ticketDestory(data) {
return request({
url: `/wf/ticket/destory/`,
method: 'post',
data
})
}
//工单详情 //工单详情
export function getTicketDetail(id) { export function getTicketDetail(id) {
return request({ return request({

View File

@ -86,13 +86,13 @@ export const asyncRoutes = [
component: Layout, component: Layout,
redirect: '/mtm/material/', redirect: '/mtm/material/',
name: 'mtm', name: 'mtm',
meta: { title: '制造管理', icon: 'example', perms: ['procurement_set'] }, meta: { title: '制造管理', icon: 'example', perms: ['mtm_manage'] },
children: [ children: [
{ {
path: 'material', path: 'material',
name: 'material', name: 'material',
component: () => import('@/views/mtm/material'), component: () => import('@/views/mtm/material'),
meta: { title: '物料清单', icon: 'example', perms: ['vendor_manage'] } meta: { title: '物料清单', icon: 'example', perms: ['mtm_material'] }
} }
, ,
{ {
@ -106,7 +106,7 @@ export const asyncRoutes = [
path: 'process', path: 'process',
name: 'process', name: 'process',
component: () => import('@/views/mtm/process'), component: () => import('@/views/mtm/process'),
meta: { title: '工序管理', icon: 'example', perms: ['vendor_manage'] } meta: { title: '工序管理', icon: 'example', perms: ['mtm_process'] }
}, },
{ {
path: 'step/:id', path: 'step/:id',
@ -127,7 +127,7 @@ export const asyncRoutes = [
path: '/mtm/productprocess/', path: '/mtm/productprocess/',
name: 'productprocess', name: 'productprocess',
component: () => import('@/views/mtm/productprocess'), component: () => import('@/views/mtm/productprocess'),
meta: { title: '产品管理', icon: 'example', perms: ['vendor_manage'] } meta: { title: '产品管理', icon: 'example', perms: ['mtm_productprocess'] }
}, },
] ]
} }
@ -137,7 +137,7 @@ export const asyncRoutes = [
component: Layout, component: Layout,
redirect: '/pm/plan', redirect: '/pm/plan',
name: 'pm', name: 'pm',
meta: { title: '生产管理', icon: 'example', perms: ['equipment_set'] }, meta: { title: '生产管理', icon: 'example', perms: ['pm_manage'] },
children: [ children: [
{ {
path: 'plan', path: 'plan',
@ -157,13 +157,13 @@ export const asyncRoutes = [
path: 'resources', path: 'resources',
name: 'resources', name: 'resources',
component: () => import('@/views/pm/resources'), component: () => import('@/views/pm/resources'),
meta: { title: '生产资源配置', icon: 'example', perms: ['index_manage'] } meta: { title: '生产资源配置', icon: 'example', perms: ['pm_resources'] }
}, },
{ {
path: 'testitem', path: 'testitem',
name: 'testitem', name: 'testitem',
component: () => import('@/views/pm/plan'), component: () => import('@/views/pm/plan'),
meta: { title: '生产作业管理', icon: 'example', perms: ['index_manage'] } meta: { title: '生产作业管理', icon: 'example', perms: ['pm_testitem'] }
} }
] ]
}, },
@ -172,31 +172,31 @@ export const asyncRoutes = [
component: Layout, component: Layout,
redirect: '/em/equipment', redirect: '/em/equipment',
name: 'em', name: 'em',
meta: { title: '设备管理', icon: 'example', perms: ['equipment_set'] }, meta: { title: '设备管理', icon: 'example', perms: ['em_manage'] },
children: [ children: [
{ {
path: 'equipment', path: 'equipment',
name: 'equipment', name: 'equipment',
component: () => import('@/views/em/equipment'), component: () => import('@/views/em/equipment'),
meta: { title: '生产设备', icon: 'example', perms: ['index_manage'] } meta: { title: '生产设备', icon: 'example', perms: ['em_equipment'] }
}, },
{ {
path: 'detection ', path: 'detection ',
name: 'detection ', name: 'detection ',
component: () => import('@/views/em/detection'), component: () => import('@/views/em/detection'),
meta: { title: '监视和测量设备', icon: 'example', perms: ['index_manage'] } meta: { title: '监视和测量设备', icon: 'example', perms: ['em_detection'] }
}, },
{ {
path: 'record', path: 'record',
name: 'record', name: 'record',
component: () => import('@/views/em/record'), component: () => import('@/views/em/record'),
meta: { title: '校准检定记录', icon: 'example', perms: ['index_manage'] } meta: { title: '校准检定记录', icon: 'example', perms: ['em_record'] }
}, },
{ {
path: 'detection ', path: 'detection ',
name: 'detection ', name: 'detection ',
component: () => import('@/views/em/detection'), component: () => import('@/views/em/detection'),
meta: { title: '运维记录', icon: 'example', perms: ['index_manage'] } meta: { title: '运维记录', icon: 'example', perms: ['em_detection'] }
} }
] ]
}, },
@ -327,32 +327,39 @@ export const asyncRoutes = [
component: Layout, component: Layout,
redirect: '/workflow/index', redirect: '/workflow/index',
name: 'workflow', name: 'workflow',
meta: { title: '工作流', icon: 'example', perms: ['workflow_set'] }, meta: { title: '工作流', icon: 'example', perms: ['workflow_manage'] },
children: [ children: [
{ {
path: 'index', path: 'index',
name: 'index', name: 'index',
component: () => import('@/views/workflow/index'), component: () => import('@/views/workflow/index'),
meta: { title: '工作流配置', icon: 'example', perms: ['workflow_manage'] } meta: { title: '工作流配置', icon: 'example', perms: ['workflow_index'] }
}, },
{ {
path: 'ticket', path: 'ticket',
name: 'ticket', name: 'ticket',
component: () => import('@/views/workflow/ticket'), component: () => import('@/views/workflow/ticket'),
meta: { title: '工单管理', icon: 'example', perms: ['workflow_manage'] }, meta: { title: '工单管理', icon: 'example' ,noCache: true, perms: ['workflow_ticket'] },
},
{
path: 'workFlowTickets',
name: 'workFlowTickets',
component: () => import('@/views/workflow/workFlowTickets'),
meta: { title: '工单管理', icon: 'example' ,noCache: true,},
hidden: true
}, },
{ {
path: 'configuration', path: 'configuration',
name: 'configuration', name: 'configuration',
component: () => import('@/views/workflow/configuration'), component: () => import('@/views/workflow/configuration'),
meta: { title: '人员信息详情', icon: 'example', perms: ['workflow_manage'] }, meta: { title: '人员信息详情', icon: 'example' },
hidden: true hidden: true
}, },
{ {
path: 'ticketHandle', path: 'ticketHandle',
name: 'ticketHandle', name: 'ticketHandle',
component: () => import('@/views/workflow/ticketHandle'), component: () => import('@/views/workflow/ticketHandle'),
meta: { title: '工单处理', icon: 'example', perms: ['workflow_manage'] }, meta: { title: '工单处理', icon: 'example',noCache: true,},
hidden: true hidden: true
}, },
] ]

View File

@ -86,21 +86,21 @@ div:focus {
.el-dialog__header { .el-dialog__header {
padding: 10px 10px 6px; padding: 10px 10px 6px;
} }
// .el-dialog{ .el-dialog{
// display: flex; display: flex;
// flex-direction: column; flex-direction: column;
// margin:0 !important; margin:0 !important;
// position:absolute; position:absolute;
// top:50%; top:50%;
// left:50%; left:50%;
// transform:translate(-50%,-50%); transform:translate(-50%,-50%);
// /*height:600px;*/ /*height:600px;*/
// max-height:calc(100% - 30px); max-height:calc(100% - 30px);
// max-width:calc(100% - 30px); max-width:calc(100% - 30px);
// } }
.el-dialog .el-dialog__body{ .el-dialog .el-dialog__body{
// flex:1; flex:1;
// overflow: auto; overflow: auto;
padding: 8px 12px; padding: 8px 12px;
} }

View File

@ -37,6 +37,7 @@
name="password" name="password"
tabindex="2" tabindex="2"
auto-complete="on" auto-complete="on"
id="passwordInput"
@keyup.enter.native="handleLogin" @keyup.enter.native="handleLogin"
><svg-icon ><svg-icon
slot="prefix" slot="prefix"
@ -200,7 +201,9 @@
margin-left: 2px; margin-left: 2px;
} }
} }
#passwordInput{
padding-right: 35px;
}
.show-pwd { .show-pwd {
height: 39px; height: 39px;
margin-right: 2px; margin-right: 2px;

View File

@ -102,7 +102,7 @@
<p>创建时间 {{watchedCreateTime}}</p> <p>创建时间 {{watchedCreateTime}}</p>
</el-col> </el-col>
</el-row> </el-row>
<svg height=800 id="mySvg" style="width: max-content!important"> <svg height=800 id="mySvg" style="width: 100%;">
</svg> </svg>
</div> </div>
</div> </div>
@ -278,7 +278,7 @@ export default {
}); });
}, },
handleTicket(scope){ handleTicket(scope){
this.$router.push({name:"ticket",params:{workflow:scope.row.id}}) this.$router.push({name:"workFlowTickets",params:{workflow:scope.row.id}})
}, },
async confirm(form) { async confirm(form) {
debugger; debugger;
@ -321,12 +321,12 @@ export default {
that.limitedWatch = true; that.limitedWatch = true;
that.$nextTick(()=>{ that.$nextTick(()=>{
var g = new dagreD3.graphlib.Graph().setGraph({ var g = new dagreD3.graphlib.Graph().setGraph({
align: 'DL', rankdir: 'DL',
nodesep: 100, nodesep: 100,
edgesep: 100, edgesep: 10,//两条线之间的距离
ranksep: 50, ranksep: 50,//节点之间的距离
marginx: 0, marginx: 160,
marginy: 50, marginy: 20,
}); });
//获取state得到节点 //获取state得到节点
getWfStateList(workFlow).then((response) => { getWfStateList(workFlow).then((response) => {
@ -343,8 +343,6 @@ export default {
//节点样式 //节点样式
style: "fill:#fff;stroke:#000", style: "fill:#fff;stroke:#000",
labelStyle: "fill:#000;", labelStyle: "fill:#000;",
// width: 83,
// height: 40,
rx :5,//矩形节点圆角度 rx :5,//矩形节点圆角度
ry :5 ry :5
}); });
@ -356,9 +354,8 @@ export default {
getWfTransitionList(workFlow).then((res)=>{ getWfTransitionList(workFlow).then((res)=>{
if(res.data){ if(res.data){
let transitionList = res.data; let transitionList = res.data;
transitionList.forEach((transitions)=>{ transitionList.forEach((transition0)=>{
let transition0 = transitions; if (transition0.condition_expression.length>0){
if (transition0.condition_expression.length>3){
debugger; debugger;
g.setNode(transition0.source_state_.id+100000, {label: "条件表达式",style: "stroke: #000;fill: #afa", shape: "diamond"}); g.setNode(transition0.source_state_.id+100000, {label: "条件表达式",style: "stroke: #000;fill: #afa", shape: "diamond"});
g.setEdge(transition0.source_state_.id, transition0.source_state_.id+100000, { g.setEdge(transition0.source_state_.id, transition0.source_state_.id+100000, {
@ -366,7 +363,7 @@ export default {
label: transition0.name, label: transition0.name,
style: "fill:#ffffff;stroke:#c0c1c3;stroke-width:1.5px" style: "fill:#ffffff;stroke:#c0c1c3;stroke-width:1.5px"
}); });
let condition_expression = JSON.parse(transition0.condition_expression); let condition_expression = transition0.condition_expression;
condition_expression.forEach(condition_expression0=>{ condition_expression.forEach(condition_expression0=>{
g.setEdge(transition0.source_state_.id+100000, condition_expression0.target_state, { g.setEdge(transition0.source_state_.id+100000, condition_expression0.target_state, {
label: condition_expression0.label, label: condition_expression0.label,
@ -386,7 +383,6 @@ export default {
g.nodes().forEach(function (v) { g.nodes().forEach(function (v) {
console.log("Node " + v + ": " + JSON.stringify(g.node(v))); console.log("Node " + v + ": " + JSON.stringify(g.node(v)));
}); });
// 创建渲染器 // 创建渲染器
let render = new dagreD3.render(); let render = new dagreD3.render();
// 选择 svg 并添加一个g元素作为绘图容器. // 选择 svg 并添加一个g元素作为绘图容器.
@ -399,6 +395,7 @@ export default {
} }
}); });
}) })
}, },
closeMark(){ closeMark(){
@ -415,33 +412,25 @@ export default {
this.hasJsonFlag = true this.hasJsonFlag = true
}, },
onError(value) { onError(value) {
// console.log("json错误了value:", value);
this.hasJsonFlag = false this.hasJsonFlag = false
}, },
onJsonChange1 (value) { onJsonChange1 (value) {
// console.log('更改value:', value);
// 实时保存 // 实时保存
this.onJsonSave1(value) this.onJsonSave1(value)
}, },
onJsonSave1 (value) { onJsonSave1 (value) {
// console.log('保存value:', value);
this.display_form_str = value this.display_form_str = value
this.hasJsonFlag1 = true this.hasJsonFlag1 = true
}, },
onError1(value) { onError1(value) {
// console.log("json错误了value:", value);
this.hasJsonFlag1 = false this.hasJsonFlag1 = false
}, },
// 检查json // 检查json
checkJson(){ checkJson(){
if (this.hasJsonFlag == false){ if (this.hasJsonFlag == false){
// console.log("json验证失败")
// this.$message.error("json验证失败")
alert("限制表达式json验证失败") alert("限制表达式json验证失败")
return false return false
} else { } else {
// console.log("json验证成功")
// this.$message.success("json验证成功")
alert("限制表达式json验证成功") alert("限制表达式json验证成功")
return true return true
} }
@ -449,13 +438,9 @@ export default {
// 检查json // 检查json
checkJson2(){ checkJson2(){
if (this.hasJsonFlag1 == false){ if (this.hasJsonFlag1 == false){
// console.log("json验证失败")
// this.$message.error("json验证失败")
alert("展现表单字段json验证失败") alert("展现表单字段json验证失败")
return false return false
} else { } else {
// console.log("json验证成功")
// this.$message.success("json验证成功")
alert("展现表单字段json1验证成功") alert("展现表单字段json1验证成功")
return true return true
} }
@ -464,6 +449,19 @@ export default {
}; };
</script> </script>
<style scoped> <style scoped>
::-webkit-scrollbar {
width: 15px;
}
::-webkit-scrollbar-track{
-webkit-box-shadow: inset 0 0 6px rgba(0,0,0,.2);
background-color: #fefefe;
border-radius: 7px;
}
::-webkit-scrollbar-thumb{
border-radius: 7px;
-webkit-box-shadow: inset 0 0 6px rgba(0,0,0,.5);
background-color: #f5f5f5;
}
.svgMark{ .svgMark{
width: 100%; width: 100%;
height: 100%; height: 100%;
@ -483,6 +481,8 @@ export default {
margin: 10vh auto 0; margin: 10vh auto 0;
text-align: center; text-align: center;
border-radius: 2px; border-radius: 2px;
max-height: 75vh;
overflow-y: scroll;
} }
.svgItem{ .svgItem{
padding: 20px 40px 0 ; padding: 20px 40px 0 ;

View File

@ -1,11 +1,9 @@
<template> <template>
<div class="app-container"> <div class="app-container">
<el-card> <el-card>
<div style="margin-top: 10px"> <div style="margin-top: 10px">
<el-button type="primary" icon="el-icon-plus" @click="handleCreate" <el-button type="primary" icon="el-icon-plus" @click="handleCreate">新增
>新增</el-button </el-button>
>
</div> </div>
</el-card> </el-card>
<el-card style="margin-top: 10px"> <el-card style="margin-top: 10px">
@ -13,7 +11,7 @@
:data="wfstateList" :data="wfstateList"
style="width: 100%" style="width: 100%"
> >
<el-table-column type="index" width="50" /> <el-table-column type="index" width="50"/>
<el-table-column width="180" label="名称"> <el-table-column width="180" label="名称">
<template slot-scope="scope">{{ scope.row.name }}</template> <template slot-scope="scope">{{ scope.row.name }}</template>
</el-table-column> </el-table-column>
@ -53,21 +51,19 @@
width="220px" width="220px"
> >
<template slot-scope="scope"> <template slot-scope="scope">
<!-- v-if="checkPermission(['workflow_update'])" -->
<el-link <el-link
v-if="checkPermission(['wfstate_update'])"
@click="handleEdit(scope)" @click="handleEdit(scope)"
>编辑</el-link >编辑
> </el-link>
<el-link <el-link
v-if="checkPermission(['wfstate_delete'])"
type="danger" type="danger"
@click="handleDelete(scope)" @click="handleDelete(scope)"
>删除</el-link >删除
> </el-link>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
</el-card> </el-card>
<el-dialog <el-dialog
:visible.sync="dialogVisible" :visible.sync="dialogVisible"
@ -81,7 +77,7 @@
:rules="rule1" :rules="rule1"
> >
<el-form-item label="名称" prop="name"> <el-form-item label="名称" prop="name">
<el-input v-model="wfstate.name" placeholder="名称" /> <el-input v-model="wfstate.name" placeholder="名称"/>
</el-form-item> </el-form-item>
<el-form-item <el-form-item
label="是否隐藏" label="是否隐藏"
@ -92,7 +88,7 @@
</el-form-item> </el-form-item>
<el-form-item label="状态顺序" prop="sort"> <el-form-item label="状态顺序" prop="sort">
<el-input v-model="wfstate.sort" type="number" placeholder="状态顺序" /> <el-input v-model="wfstate.sort" type="number" placeholder="状态顺序"/>
</el-form-item> </el-form-item>
<el-form-item label="状态类型" prop="type"> <el-form-item label="状态类型" prop="type">
@ -153,7 +149,8 @@
<el-row v-for="(item,$index) in statusChange" :key="item+$index" style="margin-top: 10px"> <el-row v-for="(item,$index) in statusChange" :key="item+$index" style="margin-top: 10px">
<el-col :span="11"> <el-col :span="11">
<el-select style="width: 100%" v-model="item.name" placeholder="请选择字段"> <el-select style="width: 100%" v-model="item.name" placeholder="请选择字段">
<el-option v-for="item in customfieldList" :key="item.id" :label="item.field_name" :value="item.field_key"> <el-option v-for="item in customfieldList" :key="item.id" :label="item.field_name"
:value="item.field_key">
</el-option> </el-option>
</el-select> </el-select>
</el-col> </el-col>
@ -166,7 +163,8 @@
</el-select> </el-select>
</el-col> </el-col>
<el-col :span="2" style="text-align: center" v-if="$index!==0"> <el-col :span="2" style="text-align: center" v-if="$index!==0">
<i class="el-icon-remove-outline" @click.prevent="removeStatusChange($index)" style="color: red;font-size: 16px;"></i> <i class="el-icon-remove-outline" @click.prevent="removeStatusChange($index)"
style="color: red;font-size: 16px;"></i>
</el-col> </el-col>
</el-row> </el-row>
</el-form-item> </el-form-item>
@ -179,51 +177,51 @@
</div> </div>
</template> </template>
<script> <script>
import { getWfStateList, createWfState,updateWfState,deleteWfState,getWfCustomfieldList } from "@/api/workflow"; import {getWfStateList, createWfState, updateWfState, deleteWfState, getWfCustomfieldList} from "@/api/workflow";
import { getOrganizationList,getUserList } from "@/api/user"; import {getOrganizationList, getUserList} from "@/api/user";
import checkPermission from "@/utils/permission"; import checkPermission from "@/utils/permission";
const defaultwfstate = { const defaultwfstate = {
name: "", name: "",
}; };
export default { export default {
components: { }, components: {},
name: "State", name: "State",
props: ["ID"], props: ["ID"],
data() { data() {
return { return {
wfstate: { wfstate: {
name:'', name: '',
is_hidden:'', is_hidden: '',
sort:'', sort: '',
type:'', type: '',
enable_retreat:'', enable_retreat: '',
participant_type:'', participant_type: '',
participant:'', participant: '',
distribute_type:'',//分发类型 distribute_type: '',//分发类型
state_fields:{}//字段状态是否可写 state_fields: {}//字段状态是否可写
}, },
statusChange:[], statusChange: [],
participant:'', participant: '',
participants:[], participants: [],
is_hidden:false, is_hidden: false,
enable_retreat:false, enable_retreat: false,
staffs:[], staffs: [],
departments:[], departments: [],
customfieldList:[], customfieldList: [],
/*wfstateList: { /*wfstateList: {
count:0 count:0
},*/ },*/
wfstateList:[], wfstateList: [],
options_:{ options_: {
"0":'无处理', "0": '无处理',
"1":'个人', "1": '个人',
"2":'多人', "2": '多人',
"3":'部门', "3": '部门',
"4":'角色', "4": '角色',
"5":'变量', "5": '变量',
"6":'普通类型', "6": '普通类型',
"7":'工单字段', "7": '工单字段',
"8":'父工单字段', "8": '父工单字段',
}, },
options: [{ options: [{
value: 0, value: 0,
@ -265,14 +263,14 @@ export default {
value: 8, value: 8,
label: '父工单字段' label: '父工单字段'
}], }],
display_form_str:[], display_form_str: [],
limit_expression:[], limit_expression: [],
dialogVisible: false, dialogVisible: false,
dialogType: "new", dialogType: "new",
rule1: { rule1: {
name: [{ required: true, message: "请输入", trigger: "blur" }], name: [{required: true, message: "请输入", trigger: "blur"}],
sort: [{ required: true, message: "请输入", trigger: "blur" }], sort: [{required: true, message: "请输入", trigger: "blur"}],
type:[{ required: true, message: "选择", trigger: "blur" }], type: [{required: true, message: "选择", trigger: "blur"}],
}, },
}; };
@ -285,7 +283,12 @@ export default {
this.getDepartment(); this.getDepartment();
}, },
methods: { methods: {
checkPermission,
checkPermission(value){
debugger;
console.log(checkPermission(value))
checkPermission(value);
},
getList() { getList() {
@ -303,22 +306,22 @@ export default {
}); });
}, },
getUser(){ getUser() {
getUserList({}).then(res=>{ getUserList({}).then(res => {
if(res.data){ if (res.data) {
this.staffs = res.data.results; this.staffs = res.data.results;
} }
}) })
}, },
getDepartment(){ getDepartment() {
getOrganizationList().then(res=>{ getOrganizationList().then(res => {
if(res.data){ if (res.data) {
this.departments = res.data; this.departments = res.data;
} }
}) })
}, },
//参与者类型变化 //参与者类型变化
typeChange(){ typeChange() {
this.participant = ''; this.participant = '';
this.participants = []; this.participants = [];
}, },
@ -354,10 +357,10 @@ export default {
this.$refs["Form"].clearValidate(); this.$refs["Form"].clearValidate();
}); });
}, },
addWordStatusChange(){ addWordStatusChange() {
this.statusChange.push({name:'',value:''}) this.statusChange.push({name: '', value: ''})
}, },
removeStatusChange(index){ removeStatusChange(index) {
this.statusChange.splice(index, 1) this.statusChange.splice(index, 1)
}, },
//编辑新建 //编辑新建
@ -367,15 +370,15 @@ export default {
const isEdit = this.dialogType === "edit"; const isEdit = this.dialogType === "edit";
// this.wfstate.participant = 1; // this.wfstate.participant = 1;
let state_fields = {}; let state_fields = {};
if(this.statusChange.length>0){ if (this.statusChange.length > 0) {
for(let i=0;i<this.statusChange.length;i++){ for (let i = 0; i < this.statusChange.length; i++) {
state_fields[this.statusChange[i].name] = this.statusChange[i].value; state_fields[this.statusChange[i].name] = this.statusChange[i].value;
} }
} }
debugger; debugger;
console.log(state_fields); console.log(state_fields);
this.wfstate.state_fields = state_fields; this.wfstate.state_fields = state_fields;
this.wfstate.participant = this.participant!==''?this.participant:this.participants; this.wfstate.participant = this.wfstate.participant_type === 1 ? this.participant : this.participants;
if (isEdit) { if (isEdit) {
updateWfState(this.wfstate.id, this.wfstate).then((res) => { updateWfState(this.wfstate.id, this.wfstate).then((res) => {
if (res.code >= 200) { if (res.code >= 200) {
@ -386,7 +389,7 @@ export default {
} }
}); });
} else { } else {
this.wfstate.workflow=this.ID; this.wfstate.workflow = this.ID;
createWfState(this.wfstate).then((res) => { createWfState(this.wfstate).then((res) => {
if (res.code >= 200) { if (res.code >= 200) {
this.getList(); this.getList();
@ -420,7 +423,6 @@ export default {
}, },
}, },
}; };
</script> </script>

View File

@ -61,10 +61,13 @@
<el-table-column align="center" label="操作"> <el-table-column align="center" label="操作">
<template slot-scope="scope"> <template slot-scope="scope">
<el-link v-if="scope.row.state_.distribute_type==1&&scope.row.participant_type==2" type="danger" @click="handleGetTicket(scope)">接单</el-link> <el-link v-if="scope.row.state_.distribute_type==1&&scope.row.participant_type==2" type="danger" @click="handleGetTicket(scope)">接单</el-link>
<el-link v-else-if="(scope.row.act_state==1||scope.row.act_state==3)&&scope.row.participant_type!==2" type="primary" @click="handleDetail(scope)">处理</el-link> <el-link v-else-if="(scope.row.act_state==1||scope.row.act_state==3)&&scope.row.participant_type!==2&&scope.row.state_.type===0" type="primary" @click="handleDetail(scope)">处理</el-link>
<!--如果state_.retreat为可退回则显示撤回按钮--> <el-link v-if="scope.row.state_.type==1&&userId==1" type="danger" @click="handleClose(scope,'2')">关闭</el-link>
<el-link v-if="scope.row.state_.enable_retreat&&userId==scope.row.create_by" type="danger" @click="handleRetreat(scope)">撤回</el-link> <!--如果state_.retreat为可退回则显示撤回按钮 state_.type==1处于草稿状态 -->
<el-link v-if="scope.row.state_.enable_retreat&&userId==scope.row.create_by&&scope.row.state_.type!==1" type="danger" @click="handleClose(scope,'1')">撤回</el-link>
<el-link type="success" @click="handleDelete(scope)">删除</el-link>
<el-link type="success" @click="handlePicture(scope)">查看流程图</el-link> <el-link type="success" @click="handlePicture(scope)">查看流程图</el-link>
<el-link type="success" @click="handleLogs(scope)">工单日志</el-link>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
@ -107,7 +110,12 @@
</el-table-column> </el-table-column>
<el-table-column align="center" label="操作"> <el-table-column align="center" label="操作">
<template slot-scope="scope"> <template slot-scope="scope">
<el-link type="primary" v-if="scope.row.act_state==4" @click="handlePicture(scope)">查看流程图</el-link> <!-- <el-link v-if="scope.row.state_.distribute_type==1&&scope.row.participant_type==2" type="danger" @click="handleGetTicket(scope)">接单</el-link>
<el-link v-else-if="(scope.row.act_state==1||scope.row.act_state==3)&&scope.row.participant_type!==2&&scope.row.state_.type===0" type="primary" @click="handleDetail(scope)">处理</el-link>
-->
<el-link v-if="scope.row.state_.type==1&&userId==1" type="danger" @click="handleClose(scope,'2')">关闭</el-link>
<el-link type="success" @click="handlePicture(scope)">查看流程图</el-link>
<el-link type="success" @click="handleLogs(scope)">工单日志</el-link>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
@ -150,10 +158,11 @@
</el-table-column> </el-table-column>
<el-table-column align="center" label="操作"> <el-table-column align="center" label="操作">
<template slot-scope="scope" v-if="scope.row.state_"> <template slot-scope="scope" v-if="scope.row.state_">
<el-link v-if="scope.row.state_.distribute_type==1&&scope.row.participant_type==2" type="danger" @click="handleGetTicket(scope)">接单</el-link> <el-link v-if="scope.row.state_.enable_retreat&&scope.row.state_.type!==1" type="danger" @click="handleClose(scope,'1')">撤回</el-link>
<el-link v-else-if="(scope.row.act_state==1||scope.row.act_state==3)&&scope.row.participant_type!==2" type="primary" @click="handleDetail(scope)">处理</el-link> <el-link v-if="scope.row.state_.type==1" type="danger" @click="handleClose(scope,'2')">关闭</el-link>
<el-link v-if="scope.row.state_.enable_retreat" type="danger" @click="handleRetreat(scope)">撤回</el-link> <el-link type="success" @click="handleDelete(scope)">删除</el-link>
<el-link type="success" @click="handlePicture(scope)">查看流程图</el-link> <el-link type="success" @click="handlePicture(scope)">查看流程图</el-link>
<el-link type="success" @click="handleLogs(scope)">工单日志</el-link>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
@ -197,7 +206,7 @@
<el-table-column align="center" label="操作"> <el-table-column align="center" label="操作">
<template slot-scope="scope"> <template slot-scope="scope">
<el-link type="danger" @click="handlePicture(scope)">查看流程图</el-link> <el-link type="danger" @click="handlePicture(scope)">查看流程图</el-link>
<el-link v-if="scope.row.act_state==1" type="danger" @click="handleDetail(scope)">处理</el-link> <el-link type="success" @click="handleLogs(scope)">工单日志</el-link>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
@ -224,25 +233,27 @@
<el-step :title="item.name" v-for="item in flowSteps " :key="item.id"> <el-step :title="item.name" v-for="item in flowSteps " :key="item.id">
</el-step> </el-step>
</el-steps> </el-steps>
<div style="width: 90%;margin: auto;">
<svg height=800 id="mySvg" style="width:100%!important;"> <svg height=800 id="mySvg" style="width:100%!important;">
</svg> </svg>
</div> </div>
</div>
<el-dialog :visible.sync="limitedRetreat" title="工单撤回">
<el-row>
<el-form ref="Form" :model="handleForm" label-width="100px" label-position="right" :rules="handleRule">
<el-col :span="2" style="height: 1px;"></el-col> </div>
<el-col :span="20" style="margin:6vh 0"> </div>
<el-form-item label="撤回原因"> <el-dialog :visible.sync="limitedRetreat" :title="handleTitle">
<el-row>
<el-form ref="Form" :model="handleForm" label-width="100px" label-position="right">
<el-col :span="1" style="height: 1px;"></el-col>
<el-col :span="22" style="margin:3vh 0">
<el-form-item :label="handleLabel">
<el-input type="textarea" :rows="3" v-model="handleForm.suggestion" placeholder="撤回原因"/> <el-input type="textarea" :rows="3" v-model="handleForm.suggestion" placeholder="撤回原因"/>
</el-form-item> </el-form-item>
</el-col> </el-col>
</el-form> </el-form>
</el-row> </el-row>
<div style="text-align: center"> <div style="text-align: center">
<el-button class="filter-item" type="" @click="retreatCancel">取消</el-button> <el-button class="filter-item" type="" @click="handleCancel">取消</el-button>
<el-button class="filter-item" type="primary" @click="retreatSubmit">确定</el-button> <el-button class="filter-item" type="primary" @click="handleSubmit">确定</el-button>
</div> </div>
</el-dialog> </el-dialog>
<el-dialog :visible.sync="limitedAdd" title="新增工单"> <el-dialog :visible.sync="limitedAdd" title="新增工单">
@ -337,13 +348,49 @@
<el-button type="primary" @click="confirm('Form')">确认</el-button> <el-button type="primary" @click="confirm('Form')">确认</el-button>
</div> </div>
</el-dialog> </el-dialog>
<el-dialog :visible.sync="limitedFlowLogs" title="工单日志">
<el-table :data="floeLogs" fit stripe
style="width: 100%;border-top:1px solid #EBEEF5;"
height="100"
highlight-current-row
v-el-height-adaptive-table="{bottomOffset: 60}">
<el-table-column label="工单标题" min-width="100">
<template slot-scope="scope" v-if="scope.row.ticket_data">
<span>{{scope.row.ticket_data.title}}</span>
</template>
</el-table-column>
<!-- <el-table-column label="当前状态" min-width="100">
<template slot-scope="scope">
<el-tag v-if="scope.row.act_state==0" label="草稿中" value="scope.row.act_state">草稿中</el-tag>
<el-tag v-else-if="scope.row.act_state==1" label="进行中" value="scope.row.act_state">进行中</el-tag>
<el-tag v-else-if="scope.row.act_state==2" label="被退回" value="scope.row.act_state">被退回</el-tag>
<el-tag v-else-if="scope.row.act_state==3" label="被撤回" value="scope.row.act_state">被撤回</el-tag>
<el-tag v-else-if="scope.row.act_state==4" label="已完成" value="scope.row.act_state">已完成</el-tag>
<el-tag v-else-if="scope.row.act_state==5" label="已关闭" value="scope.row.act_state">已关闭</el-tag>
</template>
</el-table-column>-->
<el-table-column label="进行状态" min-width="100">
<template slot-scope="scope" v-if="scope.row.state_">
<span v-if="scope.row.state_.type==0">{{scope.row.state_.name}}</span>
<span v-else>{{scope.row.state_.name}}</span>
</template>
</el-table-column>
<el-table-column label="操作人" min-width="100">
<template slot-scope="scope" v-if="scope.row.participant_">{{ scope.row.participant_.name }}</template>
</el-table-column>
<el-table-column label="操作意见" min-width="100" prop="suggestion">
</el-table-column>
<el-table-column label="更新时间" min-width="100" prop="update_time">
</el-table-column>
</el-table>
</el-dialog>
</div> </div>
</template> </template>
<script src="https://d3js.org/d3.v4.min.js"></script> <script src="https://d3js.org/d3.v4.min.js"></script>
<script> <script>
import { upUrl, upHeaders } from "@/api/file"; import { upUrl, upHeaders } from "@/api/file";
import {getWorkflowList,getWfCustomfieldList,createTicket,getWfStateList,getTickets,ticketAccpet,getWfTransitionList, import {getWorkflowList,getWfCustomfieldList,createTicket,getWfStateList,getTickets,ticketAccpet,getWfTransitionList,
ticketHandle,getWfFlowSteps,getTicketDetail,getTicketTransitions,getTicketFlowlog,getTicketRetreat} from "@/api/workflow"; ticketHandle,getWfFlowSteps,getTicketDetail,getTicketTransitions,getTicketFlowlog,ticketRetreat,ticketClose,ticketDestory} from "@/api/workflow";
import Pagination from "@/components/Pagination"; import Pagination from "@/components/Pagination";
import dagreD3 from 'dagre-d3' import dagreD3 from 'dagre-d3'
import * as d3 from 'd3' import * as d3 from 'd3'
@ -389,6 +436,8 @@
limitedStep:false, limitedStep:false,
limitedAdd:false, limitedAdd:false,
limitedRetreat:false, limitedRetreat:false,
limitedFlowLogs:false,
floeLogs:[],
nodes: [], nodes: [],
tooltip:null, tooltip:null,
edges: [], edges: [],
@ -397,11 +446,12 @@
operationBtn:[], operationBtn:[],
customfields:[], customfields:[],
transitions:[], transitions:[],
handleLabel:'撤回原因',
handleTitle:'工单撤回',
rule1:{ rule1:{
title: [{ required: true, message: "请输入", trigger: "blur" }], title: [{ required: true, message: "请输入", trigger: "blur" }],
workflow: [{ required: true, message: "选择", trigger: "blur" }] workflow: [{ required: true, message: "选择", trigger: "blur" }]
} }
, ,
} }
}, },
@ -416,15 +466,6 @@
this.getStates(); this.getStates();
this.getWorkFlow(); this.getWorkFlow();
}, },
activated(){
this.workflow = this.$route.params.workflow;
if(this.workflow){//有传参
this.pageForm.workflow = parseInt(this.workflow);
}
this.getList();
this.getStates();
this.getWorkFlow();
},
methods:{ methods:{
getList(){ getList(){
this.listLoading = true; this.listLoading = true;
@ -439,16 +480,12 @@
getTicketFlowlogs(id){ getTicketFlowlogs(id){
getTicketFlowlog(id).then(res=>{ getTicketFlowlog(id).then(res=>{
if(res.data){ if(res.data){
debugger;
console.log(res)
this.logs = res.data; this.logs = res.data;
} }
}) })
}, },
handleClick(tab, event) { handleClick(tab, event) {
console.log(tab, event); console.log(tab, event);
console.log(this.pageForm.category);
debugger;
this.getList(); this.getList();
}, },
//获取工作流所有状态 //获取工作流所有状态
@ -563,17 +600,19 @@
} }
}); });
}, },
retreatCancel(){ handleCancel(){
this.limitedRetreat = false; this.limitedRetreat = false;
}, },
retreatSubmit(){ handleSubmit(){
this.$confirm("确认撤回吗?", "温馨提示", { debugger;
if(this.handleTitle === '撤回工单'){
this.$confirm("确认撤回工单吗?", "温馨提示", {
confirmButtonText: "确认", confirmButtonText: "确认",
cancelButtonText: "取消", cancelButtonText: "取消",
type: "warning", type: "warning",
}) })
.then(async () => { .then(async () => {
getTicketRetreat(this.retreatId,this.handleForm).then(res=>{ ticketRetreat(this.retreatId,this.handleForm).then(res=>{
this.limitedRetreat = false; this.limitedRetreat = false;
this.getList(); this.getList();
this.$message.success("成功"); this.$message.success("成功");
@ -582,11 +621,37 @@
.catch((err) => { .catch((err) => {
console.error(err); console.error(err);
}); });
}else{
this.$confirm("确认关闭工单吗?", "温馨提示", {
confirmButtonText: "确认",
cancelButtonText: "取消",
type: "warning",
})
.then(async () => {
ticketClose(this.retreatId,this.handleForm).then(res=>{
this.limitedRetreat = false;
this.getList();
this.$message.success("成功");
})
})
.catch((err) => {
console.error(err);
});
}
}, },
handleRetreat(scope){ handleClose(scope,index){
if(index==='1'){
this.handleTitle = '撤回工单';
this.handleLabel='撤回原因';
}else{
this.handleTitle = '关闭工单';
this.handleLabel='关闭原因';
}
this.limitedRetreat = true; this.limitedRetreat = true;
this.retreatId = scope.row.id; this.retreatId = scope.row.id;
}, },
handleLogs(){},
handlePicture(scope){ handlePicture(scope){
let that = this; let that = this;
that.dialogVisible = true; that.dialogVisible = true;
@ -594,8 +659,6 @@
that.watchedCreateTime = scope.row.create_time; that.watchedCreateTime = scope.row.create_time;
that.pageForm.workflow = scope.row.workflow; that.pageForm.workflow = scope.row.workflow;
let ticketId = scope.row.id; let ticketId = scope.row.id;
// that.getStates();
// that.getTicketFlowlogs(ticketId);
getWfFlowSteps( ticketId).then((res)=>{ getWfFlowSteps( ticketId).then((res)=>{
if(res.data){ if(res.data){
debugger; debugger;
@ -616,12 +679,12 @@
that.actives =that.flowSteps.length; that.actives =that.flowSteps.length;
} }
var g = new dagreD3.graphlib.Graph().setGraph({ var g = new dagreD3.graphlib.Graph().setGraph({
align: 'DL', rankdir: 'DL',
nodesep: 100, nodesep: 100,
edgesep: 100, edgesep: 10,//两条线之间的距离
ranksep: 50, ranksep: 50,//节点之间的距离
marginx: 300, marginx: 60,
marginy: 50, marginy: 20,
}); });
//获取state得到节点 //获取state得到节点
getWfStateList(that.pageForm.workflow).then((response) => { getWfStateList(that.pageForm.workflow).then((response) => {
@ -629,9 +692,6 @@
let nodes = response.data; let nodes = response.data;
// 添加节点 // 添加节点
nodes.forEach((item) => { nodes.forEach((item) => {
/*debugger;
console.log(item.sort)
console.log(that.state)*/
if(item.sort==that.sort){ if(item.sort==that.sort){
g.setNode(item.id, { g.setNode(item.id, {
// 节点标签 // 节点标签
@ -645,7 +705,6 @@
rx :5,//矩形节点圆角度 rx :5,//矩形节点圆角度
ry :5 ry :5
}); });
}else{ }else{
g.setNode(item.id, { g.setNode(item.id, {
// 节点标签 // 节点标签
@ -677,12 +736,14 @@
g.setEdge(transition0.source_state_.id, transition0.source_state_.id+100000, { g.setEdge(transition0.source_state_.id, transition0.source_state_.id+100000, {
// 边标签 // 边标签
label: transition0.name, label: transition0.name,
// curve: d3.curveBasis,
style: "fill:#ffffff;stroke:#c0c1c3;stroke-width:1.5px" style: "fill:#ffffff;stroke:#c0c1c3;stroke-width:1.5px"
}); });
let condition_expression = transition0.condition_expression; let condition_expression = transition0.condition_expression;
condition_expression.forEach(condition_expression0=>{ condition_expression.forEach(condition_expression0=>{
g.setEdge(transition0.source_state_.id+100000, condition_expression0.target_state, { g.setEdge(transition0.source_state_.id+100000, condition_expression0.target_state, {
label: condition_expression0.label, label: condition_expression0.label,
// curve: d3.curveBasis,
style: "fill:#ffffff;stroke:#c0c1c3;stroke-width:1.5px" style: "fill:#ffffff;stroke:#c0c1c3;stroke-width:1.5px"
}) })
}) })
@ -690,6 +751,7 @@
g.setEdge(transition0.source_state_.id, transition0.destination_state_.id, { g.setEdge(transition0.source_state_.id, transition0.destination_state_.id, {
// 边标签 // 边标签
label: transition0.name, label: transition0.name,
// curve: d3.curveBasis,
// 边样式 // 边样式
style: "fill:#ffffff;stroke:#c0c1c3;stroke-width:1.5px" // 根据后台数据来改变连线的颜色 style: "fill:#ffffff;stroke:#c0c1c3;stroke-width:1.5px" // 根据后台数据来改变连线的颜色
}); });
@ -699,6 +761,10 @@
g.nodes().forEach(function (v) { g.nodes().forEach(function (v) {
console.log("Node " + v + ": " + JSON.stringify(g.node(v))); console.log("Node " + v + ": " + JSON.stringify(g.node(v)));
}); });
console.log("1111111111111");
g.edges().forEach(function (e) {
console.log("Edge " + e.v + " -> " + e.w + ": " + JSON.stringify(g.edge(e)));
});
// 创建渲染器 // 创建渲染器
let render = new dagreD3.render(); let render = new dagreD3.render();
// 选择 svg 并添加一个g元素作为绘图容器. // 选择 svg 并添加一个g元素作为绘图容器.
@ -800,6 +866,31 @@
} }
}) })
}, },
handleLogs(scope){
let id = scope.row.id;
this.limitedFlowLogs = true;
getTicketFlowlog(id).then(res=>{
if(res.data){
this.floeLogs = res.data;
}
})
},
handleDelete(scope){
let data = new Object(),ids=[];
ids.push(scope.row.id);
data.ids = ids;
this.$confirm("确认删除该工单吗?", "温馨提示", {
confirmButtonText: "确认",
cancelButtonText: "取消",
type: "warning",
}).then(async () => {
ticketDestory(data).then(res=>{
if(res.code==200){
this.getList();
}
})
})
},
stepclick(){}, stepclick(){},
closeMark(){ closeMark(){
this.dialogVisible = false; this.dialogVisible = false;
@ -809,6 +900,19 @@
</script> </script>
<style scoped> <style scoped>
::-webkit-scrollbar {
width: 15px;
}
::-webkit-scrollbar-track{
-webkit-box-shadow: inset 0 0 6px rgba(0,0,0,.2);
background-color: #fefefe;
border-radius: 7px;
}
::-webkit-scrollbar-thumb{
border-radius: 7px;
-webkit-box-shadow: inset 0 0 6px rgba(0,0,0,.5);
background-color: #f5f5f5;
}
.svgMark{ .svgMark{
width: 100%; width: 100%;
height: 100%; height: 100%;
@ -828,6 +932,8 @@
margin: 10vh auto 0; margin: 10vh auto 0;
text-align: center; text-align: center;
border-radius: 2px; border-radius: 2px;
max-height: 80vh;
overflow-y: scroll;
} }
.svgItem{ .svgItem{
padding: 20px 40px 0 ; padding: 20px 40px 0 ;

View File

@ -13,7 +13,7 @@
</el-col> </el-col>
<el-col :span="16"> <el-col :span="16">
<el-card style="margin-left: 10px"> <el-card style="margin-left: 10px">
<el-form ref="Form" :model="handleForm" label-width="100px" label-position="right" :rules="handleRule"> <el-form ref="Form" :model="ticketForm" label-width="100px" label-position="right" :rules="ticketFormRule">
<el-col :span="12"> <el-col :span="12">
<el-form-item label="标题" style="margin-bottom: 0"> <el-form-item label="标题" style="margin-bottom: 0">
<span>{{ticketDetail.title}}</span> <span>{{ticketDetail.title}}</span>
@ -114,24 +114,102 @@
<el-row> <el-row>
<el-col :span="12"> <el-col :span="12">
<el-form-item label="审批意见"> <el-form-item label="审批意见">
<el-input type="textarea" :rows="3" v-model="handleForm.suggestion" placeholder="审批意见"/> <el-input type="textarea" :rows="3" v-model="ticketForm.suggestion" placeholder="审批意见"/>
</el-form-item> </el-form-item>
</el-col> </el-col>
</el-row> </el-row>
</el-form> </el-form>
<div style="text-align: center"> <div style="text-align: center;display: flex;justify-content: space-between;">
<div>
<template v-if="ticketDetail.state_">
<!--state_.type===0普通状态&&act_state==1进行中participant_type!==2处理人不是多人!ticketDetail.in_add_node非加签状态-->
<el-button v-if="ticketDetail.act_state==1&&ticketDetail.state_.type===0&&!ticketDetail.in_add_node" type="primary" @click="handleAdd">加签</el-button>
<el-button v-if="ticketDetail.state_.type===1" type="primary" @click="handleClick('2')">关闭</el-button>
<el-button v-if="ticketDetail.state_.enable_retreat&&(userId==ticketDetail.create_by)&&ticketDetail.state_.type!==1" type="danger" @click="handleClick('1')">撤回</el-button>
</template>
</div>
<div>
<el-button v-if="ticketDetail.in_add_node" class="filter-item" type="primary" @click="handleClick('3')">加签处理</el-button>
<el-button v-for="item in operationBtn" :key="item.id" class="filter-item" type="primary" @click="operationSubmit(item.id)">{{item.name}}</el-button> <el-button v-for="item in operationBtn" :key="item.id" class="filter-item" type="primary" @click="operationSubmit(item.id)">{{item.name}}</el-button>
</div> </div>
</div>
</el-card>
<el-card style="margin-left: 10px">
<el-table :data="logs" fit stripe
style="width: 100%;border-top:1px solid #EBEEF5;"
height="100"
highlight-current-row
v-el-height-adaptive-table="{bottomOffset: 60}">
<el-table-column label="工单标题" min-width="100">
<template slot-scope="scope" v-if="scope.row.ticket_data">
<span>{{scope.row.ticket_data.title}}</span>
</template>
</el-table-column>
<el-table-column label="进行状态" min-width="100">
<template slot-scope="scope" v-if="scope.row.state_">
<span v-if="scope.row.state_.type==0">{{scope.row.state_.name}}</span>
<span v-else>{{scope.row.state_.name}}</span>
</template>
</el-table-column>
<el-table-column label="操作人" min-width="100">
<template slot-scope="scope" v-if="scope.row.participant_">{{ scope.row.participant_.name }}</template>
</el-table-column>
<el-table-column label="操作意见" min-width="100" prop="suggestion">
</el-table-column>
<el-table-column label="更新时间" min-width="100" prop="update_time">
</el-table-column>
</el-table>
</el-card> </el-card>
</el-col> </el-col>
</el-row> </el-row>
<!--撤回1关闭2加签结束3-->
<el-dialog :visible.sync="limitedHandle" :title="handleTitle">
<el-row>
<el-form ref="Form" :model="handleForm" label-width="100px" label-position="right">
<el-col :span="1" style="height: 1px;"></el-col>
<el-col :span="22" style="margin:3vh 0">
<el-form-item :label="handleLabel">
<el-input type="textarea" :rows="3" v-model="handleForm.suggestion" placeholder="撤回原因"/>
</el-form-item>
</el-col>
</el-form>
</el-row>
<div style="text-align: center">
<el-button class="filter-item" type="" @click="retreatCancel">取消</el-button>
<el-button class="filter-item" type="primary" @click="handleSubmit">确定</el-button>
</div>
</el-dialog>
<el-dialog :visible.sync="limitedAdd" title="加签">
<el-row>
<el-form ref="Form" :model="addForm" label-width="100px" label-position="right">
<el-col :span="22">
<el-form-item label="加签操作人">
<el-select style="width: 100%" v-model="addForm.toadd_user" placeholder="加签操作人">
<el-option v-for="item in staffs" :key="item.id" :label="item.name" :value="item.id">
</el-option>
</el-select>
</el-form-item>
<el-form-item label="加签原因">
<el-input type="textarea" :rows="3" v-model="addForm.suggestion" placeholder="加签原因"/>
</el-form-item>
</el-col>
</el-form>
</el-row>
<div style="text-align: center">
<el-button class="filter-item" type="" @click="addCancel">取消</el-button>
<el-button class="filter-item" type="primary" @click="addSubmit">确定</el-button>
</div>
</el-dialog>
</div> </div>
</template> </template>
<script src="https://d3js.org/d3.v4.min.js"></script> <script src="https://d3js.org/d3.v4.min.js"></script>
<script> <script>
import { upUrl, upHeaders } from "@/api/file"; import { upUrl, upHeaders } from "@/api/file";
import {getWorkflowList,getWfCustomfieldList,getWfStateList,getWfTransitionList, import { getUserList } from "@/api/user";
ticketHandle,getWfFlowSteps,getTicketDetail,getTicketTransitions,getTicketFlowlog } from "@/api/workflow"; import {getWorkflowList,getWfCustomfieldList,getWfStateList,getWfTransitionList,ticketHandle
,getWfFlowSteps,getTicketDetail,getTicketTransitions,getTicketFlowlog,ticketRetreat,
ticketAddNode,ticketAddNodeEnd,ticketClose } from "@/api/workflow";
import Pagination from "@/components/Pagination"; import Pagination from "@/components/Pagination";
import dagreD3 from 'dagre-d3' import dagreD3 from 'dagre-d3'
import * as d3 from 'd3' import * as d3 from 'd3'
@ -145,19 +223,31 @@
actives:4, actives:4,
ticketId:0, ticketId:0,
stateSteps:0, stateSteps:0,
handleForm:{ ticketForm:{
transition:'', transition:'',
ticket_data:{}, ticket_data:{},
suggestion:'', suggestion:'',
}, },
handleType:'1',
handleLabel:'撤回原因',
handleTitle:'工单撤回',
handleForm:{
suggestion:'',
},
addForm:{
toadd_user:'',
suggestion:'',
},
tooltip:null, tooltip:null,
handleRule:{}, handleRule:{},
upUrl: upUrl(), upUrl: upUrl(),
upHeaders: upHeaders(), upHeaders: upHeaders(),
userId:'',
workflow:'', workflow:'',
watchedName:'', watchedName:'',
watchedCreateTime:'', watchedCreateTime:'',
logs:[], logs:[],
staffs:[],
edges: [], edges: [],
nodes: [], nodes: [],
tooltip: [], tooltip: [],
@ -167,11 +257,15 @@
ticketDetail:{}, ticketDetail:{},
operationBtn:[], operationBtn:[],
customfields:[], customfields:[],
transitions:[] transitions:[],
limitedAdd:false,
limitedHandle:false,
ticketFormRule:{},
} }
}, },
mounted(){ mounted(){
let that = this; let that = this;
that.userId = sessionStorage.getItem('userId')
let ticketId = that.ticketId = this.$route.params.ticketId; let ticketId = that.ticketId = this.$route.params.ticketId;
let workflow = that.workflow = this.$route.params.workflow; let workflow = that.workflow = this.$route.params.workflow;
let handleTicketId = sessionStorage.getItem('handleTicketId'); let handleTicketId = sessionStorage.getItem('handleTicketId');
@ -190,6 +284,7 @@
ticketId = that.ticketId = handleTicketId; ticketId = that.ticketId = handleTicketId;
workflow = that.workflow = handleTicketWorkflow; workflow = that.workflow = handleTicketWorkflow;
} }
this.getTicketFlowlogs(ticketId);
getTicketTransitions(ticketId).then(res=>{ getTicketTransitions(ticketId).then(res=>{
this.operationBtn = res.data; this.operationBtn = res.data;
}) })
@ -201,6 +296,9 @@
if(res.data){ if(res.data){
that.tooltip = that.createTooltip(); that.tooltip = that.createTooltip();
that.ticketDetail = res.data; that.ticketDetail = res.data;
console.log(that.ticketDetail.state_.enable_retreat);
debugger;
let state = res.data.state; let state = res.data.state;
let dat = that.flowSteps.filter((item)=>{ let dat = that.flowSteps.filter((item)=>{
return item.id==state; return item.id==state;
@ -211,12 +309,12 @@
that.actives =that.flowSteps.length; that.actives =that.flowSteps.length;
} }
var g = new dagreD3.graphlib.Graph().setGraph({ var g = new dagreD3.graphlib.Graph().setGraph({
align: 'DL', rankdir: 'DL',
nodesep: 100, nodesep: 100,
edgesep: 100, edgesep: 10,//两条线之间的距离
ranksep: 50, ranksep: 50,//节点之间的距离
marginx: 50, marginx: 60,
marginy: 50, marginy: 20,
}); });
//获取state得到节点 //获取state得到节点
getWfStateList(workflow).then((response) => { getWfStateList(workflow).then((response) => {
@ -345,6 +443,13 @@
activated(){ activated(){
}, },
methods:{ methods:{
getUser(){
getUserList({}).then(res=>{
if(res.data){
this.staffs = res.data.results;
}
})
},
//工单流转记录 //工单流转记录
getTicketFlowlogs(id){ getTicketFlowlogs(id){
getTicketFlowlog(id).then(res=>{ getTicketFlowlog(id).then(res=>{
@ -403,19 +508,86 @@
.style('display', 'none') .style('display', 'none')
}, },
operationSubmit(id){ operationSubmit(id){
this.handleForm.transition = id; this.ticketForm.transition = id;
this.handleForm.ticket_data = this.ticketDetail.ticket_data; this.ticketForm.ticket_data = this.ticketDetail.ticket_data;
let obj = new Object(); let obj = new Object();
obj.transition = id; obj.transition = id;
obj.ticket_data = this.ticketDetail.ticket_data; obj.ticket_data = this.ticketDetail.ticket_data;
obj.suggestion = this.handleForm.suggestion; obj.suggestion = this.ticketForm.suggestion;
console.log(this.handleForm); console.log(this.ticketForm);
ticketHandle(this.ticketId,obj).then(res=>{ ticketHandle(this.ticketId,obj).then(res=>{
if (res.data){ if (res.data){
this.$router.replace({name:"ticket"}) this.$router.replace({name:"ticket"})
} }
}) })
}, },
retreatCancel(){
this.limitedHandle = false;
},
handleSubmit(){
let text = '',url='';
if(this.handleType==='1'){
text = "确认撤回工单吗?";
url = ticketRetreat;
}else if(this.handleType==='2'){
text = "确认关闭工单吗?";
url = ticketClose;
}if(this.handleType==='3'){
text = "确认处理工单吗?";
url = ticketAddNodeEnd;
}
this.$confirm(text, "温馨提示", {
confirmButtonText: "确认",
cancelButtonText: "取消",
type: "warning",
})
.then(async () => {
url(this.ticketId,this.handleForm).then(res=>{
this.limitedHandle = false;
this.$router.replace({name:"ticket"})
this.$message.success("成功");
})
})
.catch((err) => {
console.error(err);
});
},
handleClick(index){
this.limitedHandle = true;
this.handleType = index;
if(index==='1'){
this.handleTitle = '工单撤回';
this.handleLabel = '撤回原因';
}else if(index==='2'){
this.handleTitle = '工单关闭';
this.handleLabel = '关闭原因';
}else if(index==='3'){
this.handleTitle = '加签处理';
this.handleLabel = '加签意见';
}
},
handleAdd(){
this.getUser();
this.limitedAdd = true;
},
addCancel(){
this.limitedAdd = false;
},
addSubmit(){
ticketAddNode(this.ticketId,this.addForm).then(res=>{
if(res.data){
this.limitedAdd = false;
this.$nextTick(function () {
this.limitedAdd = true;
this.$router.replace({name:"ticket"})
})
}
})
},
addNode(){
},
} }
} }
</script> </script>

View File

@ -0,0 +1,197 @@
<template>
<div class="app-container">
<el-card style="margin-bottom: 10px">
<div>
<el-select v-model="filter.workflow" placeholder="工作流" clearable style="width: 200px" class="filter-item" @change="handleFilter">
<el-option
v-for="item in workFlows"
:key="item.id"
:label="item.name"
:value="item.id"
/>
</el-select>
<el-select v-model="filter.act_state" placeholder="工单进行状态" clearable style="width: 200px" class="filter-item" @change="handleFilter">
<el-option label="草稿中" value="0" />
<el-option label="进行中" value="1" />
<el-option label="被退回" value="2" />
<el-option label="被撤回" value="3" />
<el-option label="已完成" value="4" />
<el-option label="已关闭" value="5" />
</el-select>
<el-button
class="filter-item"
type="primary"
icon="el-icon-refresh-left"
@click="resetFilter"
>重置</el-button>
<el-button
class="filter-item"
type="primary"
icon="el-icon-search"
@click="handleFilter"
>搜索</el-button>
</div>
</el-card>
<el-card>
<el-table :data="tickets"
border fit stripe
style="width: 100%"
height="100"
highlight-current-row
v-el-height-adaptive-table="{bottomOffset: 60}">
<el-table-column label="工单标题" min-width="100" prop="title">
</el-table-column>
<el-table-column label="当前状态" min-width="100">
<template slot-scope="scope">
<el-tag v-if="scope.row.act_state==0" label="草稿中" value="scope.row.act_state">草稿中</el-tag>
<el-tag v-else-if="scope.row.act_state==1" label="进行中" value="scope.row.act_state">进行中</el-tag>
<el-tag v-else-if="scope.row.act_state==2" label="被退回" value="scope.row.act_state">被退回</el-tag>
<el-tag v-else-if="scope.row.act_state==3" label="被撤回" value="scope.row.act_state">被撤回</el-tag>
<el-tag v-else-if="scope.row.act_state==4" label="已完成" value="scope.row.act_state">已完成</el-tag>
<el-tag v-else-if="scope.row.act_state==5" label="已关闭" value="scope.row.act_state">已关闭</el-tag>
</template>
</el-table-column>
<el-table-column label="进行状态" min-width="100">
<template slot-scope="scope">
<span v-if="scope.row.state_.type==0">{{scope.row.state_.name}}</span>
<span v-else>{{scope.row.state_.name}}</span>
</template>
</el-table-column>
<el-table-column label="类型" min-width="100">
<template slot-scope="scope">{{ scope.row.workflow_.name }}</template>
</el-table-column>
<el-table-column label="创建时间" min-width="100" prop="create_time">
</el-table-column>
<el-table-column align="center" label="操作">
<template slot-scope="scope">
<el-link type="success" @click="handleLogs(scope)">工单日志</el-link>
</template>
</el-table-column>
</el-table>
<pagination
:total="total"
:page.sync="filter.page"
:limit.sync="filter.page_size"
@pagination="getList"
/>
</el-card>
<el-dialog :visible.sync="limitedFlowLogs" title="工单日志">
<el-table :data="floeLogs" fit stripe
style="width: 100%;border-top:1px solid #EBEEF5;"
height="100"
highlight-current-row
v-el-height-adaptive-table="{bottomOffset: 60}">
<el-table-column label="工单标题" min-width="100">
<template slot-scope="scope" v-if="scope.row.ticket_data">
<span>{{scope.row.ticket_data.title}}</span>
</template>
</el-table-column>
<!-- <el-table-column label="当前状态" min-width="100">
<template slot-scope="scope">
<el-tag v-if="scope.row.act_state==0" label="草稿中" value="scope.row.act_state">草稿中</el-tag>
<el-tag v-else-if="scope.row.act_state==1" label="进行中" value="scope.row.act_state">进行中</el-tag>
<el-tag v-else-if="scope.row.act_state==2" label="被退回" value="scope.row.act_state">被退回</el-tag>
<el-tag v-else-if="scope.row.act_state==3" label="被撤回" value="scope.row.act_state">被撤回</el-tag>
<el-tag v-else-if="scope.row.act_state==4" label="已完成" value="scope.row.act_state">已完成</el-tag>
<el-tag v-else-if="scope.row.act_state==5" label="已关闭" value="scope.row.act_state">已关闭</el-tag>
</template>
</el-table-column>-->
<el-table-column label="进行状态" min-width="100">
<template slot-scope="scope" v-if="scope.row.state_">
<span v-if="scope.row.state_.type==0">{{scope.row.state_.name}}</span>
<span v-else>{{scope.row.state_.name}}</span>
</template>
</el-table-column>
<el-table-column label="操作人" min-width="100">
<template slot-scope="scope" v-if="scope.row.participant_">{{ scope.row.participant_.name }}</template>
</el-table-column>
<el-table-column label="操作意见" min-width="100" prop="suggestion">
</el-table-column>
<el-table-column label="更新时间" min-width="100" prop="update_time">
</el-table-column>
</el-table>
</el-dialog>
</div>
</template>
<script>
import {getWorkflowList,getTickets,getTicketFlowlog} from "@/api/workflow";
import Pagination from "@/components/Pagination";
export default {
name: "workFlowTickets",
components: { Pagination },
inject:['reload'],
data(){
return{
sort:0,
total:0,
filter:{
page:1,
page_size:20,
state:'',
act_state:'',
workflow:'',
category:'all',
search:'',
},
listLoading:false,
limitedFlowLogs:false,
floeLogs:[],
tickets:[],
workFlows:[],
}
},
mounted(){
this.workflow = this.$route.params.workflow;
if(this.workflow){//有传参
this.filter.workflow = parseInt(this.workflow);
}
this.getList();
this.getWorkFlow();
},
methods:{
getList(){
this.listLoading = true;
getTickets( this.filter).then((res)=>{
if(res.data.results){
this.total = res.data.count;
this.tickets = res.data.results;
this.listLoading = false;
}
})
},
//获取工作流
getWorkFlow(){
let listForm = {page:0};
getWorkflowList(listForm).then((response) => {
if (response.data) {
this.workFlows = response.data;
}
});
},
//查询搜索
handleFilter(){
this.filter.page=1;
this.getList();
},
//筛选重置
resetFilter(){
this.filter.workflow = '';
this.filter.act_state = '';
this.getList();
},
handleLogs(scope){
let id = scope.row.id;
this.limitedFlowLogs = true;
getTicketFlowlog(id).then(res=>{
if(res.data){
this.floeLogs = res.data;
}
})
},
}
}
</script>
<style scoped>
</style>