Compare commits
34 Commits
82f7d4be5e
...
dev-zcx
| Author | SHA1 | Date | |
|---|---|---|---|
| 85f5b87f84 | |||
|
|
0253dfa47f | ||
|
|
81a3847992 | ||
|
|
49b5d302a9 | ||
|
|
8a2d976552 | ||
|
|
f1d2449599 | ||
|
|
e97de4342c | ||
|
|
bfdbef4bd7 | ||
|
|
c519172f5e | ||
|
|
41d52f4ce7 | ||
|
|
79530d221f | ||
|
|
d6bfbcf96a | ||
|
|
b155d3ce9e | ||
|
|
5097cf38f6 | ||
|
|
ae11ac4dc7 | ||
|
|
b0573fdf17 | ||
|
|
b91fe3b05a | ||
|
|
75572f6104 | ||
|
|
a712e3f85f | ||
|
|
a5a64b7131 | ||
|
|
b196135ddb | ||
|
|
34de068ffa | ||
|
|
b1feb33c00 | ||
|
|
853a18df9b | ||
|
|
288770adce | ||
|
|
7bb3a74095 | ||
|
|
34a1fc56d3 | ||
|
|
926cefaee9 | ||
|
|
e5e141fb84 | ||
|
|
ca10366398 | ||
|
|
d39ea3270e | ||
|
|
4726770e3f | ||
|
|
514b602599 | ||
|
|
17208922ad |
@@ -4,11 +4,13 @@ VITE_NODE_ENV=development
|
|||||||
VITE_DEV=true
|
VITE_DEV=true
|
||||||
|
|
||||||
# 请求路径
|
# 请求路径
|
||||||
VITE_BASE_URL='http://118.31.23.45:48080'
|
VITE_BASE_URL='http://localhost:48080'
|
||||||
|
|
||||||
|
# VITE_BASE_URL='http://47.98.161.246:48080'
|
||||||
# VITE_BASE_URL='http://114.55.169.15:48080'
|
# VITE_BASE_URL='http://114.55.169.15:48080'
|
||||||
|
|
||||||
# 上传路径
|
# 上传路径
|
||||||
VITE_UPLOAD_URL='http://118.31.23.45:48080/admin-api/system/file/upload'
|
VITE_UPLOAD_URL='http://47.98.161.246:48080/admin-api/system/file/upload'
|
||||||
# VITE_UPLOAD_URL='http://114.55.169.15:48080/admin-api/system/file/upload'
|
# VITE_UPLOAD_URL='http://114.55.169.15:48080/admin-api/system/file/upload'
|
||||||
|
|
||||||
# 接口前缀
|
# 接口前缀
|
||||||
|
|||||||
2
.env.dev
@@ -7,7 +7,7 @@ VITE_DEV=false
|
|||||||
VITE_BASE_URL='http://localhost:48080'
|
VITE_BASE_URL='http://localhost:48080'
|
||||||
|
|
||||||
# 上传路径
|
# 上传路径
|
||||||
VITE_UPLOAD_URL='http://118.31.23.45:48080/admin-api/system/file/upload'
|
VITE_UPLOAD_URL='http://47.98.161.246:48080/admin-api/system/file/upload'
|
||||||
|
|
||||||
# 接口前缀
|
# 接口前缀
|
||||||
VITE_API_BASEPATH=/oa-api
|
VITE_API_BASEPATH=/oa-api
|
||||||
|
|||||||
@@ -4,10 +4,10 @@ VITE_NODE_ENV=development
|
|||||||
VITE_DEV=true
|
VITE_DEV=true
|
||||||
|
|
||||||
# 请求路径
|
# 请求路径
|
||||||
VITE_BASE_URL='http://118.31.23.45:48080'
|
VITE_BASE_URL='http://47.98.161.246:48080'
|
||||||
|
|
||||||
# 上传路径
|
# 上传路径
|
||||||
VITE_UPLOAD_URL='http://118.31.23.45:48080/admin-api/system/file/upload'
|
VITE_UPLOAD_URL='http://47.98.161.246:48080/admin-api/system/file/upload'
|
||||||
|
|
||||||
# 接口前缀
|
# 接口前缀
|
||||||
VITE_API_BASEPATH=/oa-api
|
VITE_API_BASEPATH=/oa-api
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ VITE_DEV=false
|
|||||||
VITE_BASE_URL='http://api-dashboard.yudao.iocoder.cn'
|
VITE_BASE_URL='http://api-dashboard.yudao.iocoder.cn'
|
||||||
|
|
||||||
# 上传路径
|
# 上传路径
|
||||||
VITE_UPLOAD_URL='http://118.31.23.45:48080/admin-api/system/file/upload'
|
VITE_UPLOAD_URL='http://47.98.161.246:48080/admin-api/system/file/upload'
|
||||||
|
|
||||||
# 接口前缀
|
# 接口前缀
|
||||||
VITE_API_BASEPATH=/oa-api
|
VITE_API_BASEPATH=/oa-api
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ VITE_DEV=false
|
|||||||
VITE_BASE_URL='http://localhost:48080'
|
VITE_BASE_URL='http://localhost:48080'
|
||||||
|
|
||||||
# 上传路径
|
# 上传路径
|
||||||
VITE_UPLOAD_URL='http://118.31.23.45:48080/admin-api/system/file/upload'
|
VITE_UPLOAD_URL='http://47.98.161.246:48080/admin-api/system/file/upload'
|
||||||
|
|
||||||
# 接口前缀
|
# 接口前缀
|
||||||
VITE_API_BASEPATH=/oa-api
|
VITE_API_BASEPATH=/oa-api
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ VITE_DEV=false
|
|||||||
VITE_BASE_URL='http://localhost:48080'
|
VITE_BASE_URL='http://localhost:48080'
|
||||||
|
|
||||||
# 上传路径
|
# 上传路径
|
||||||
VITE_UPLOAD_URL='http://118.31.23.45:48080/admin-api/system/file/upload'
|
VITE_UPLOAD_URL='http://47.98.161.246:48080/admin-api/system/file/upload'
|
||||||
|
|
||||||
# 接口前缀
|
# 接口前缀
|
||||||
VITE_API_BASEPATH=/oa-api
|
VITE_API_BASEPATH=/oa-api
|
||||||
|
|||||||
5
.vscode/settings.json
vendored
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
{
|
||||||
|
"i18n-ally.localesPaths": [
|
||||||
|
"src/locales"
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "ss-oa-manage",
|
"name": "ss-oa-manage",
|
||||||
"version": "1.0.0",
|
"version": "1.0.1",
|
||||||
"description": "莳松oa管理系统",
|
"description": "莳松oa管理系统",
|
||||||
"author": "ss",
|
"author": "ss",
|
||||||
"private": false,
|
"private": false,
|
||||||
@@ -51,7 +51,7 @@
|
|||||||
"diagram-js": "^11.6.0",
|
"diagram-js": "^11.6.0",
|
||||||
"echarts": "^5.4.2",
|
"echarts": "^5.4.2",
|
||||||
"echarts-wordcloud": "^2.1.0",
|
"echarts-wordcloud": "^2.1.0",
|
||||||
"element-plus": "2.3.4",
|
"element-plus": "2.9.4",
|
||||||
"fast-xml-parser": "^4.2.2",
|
"fast-xml-parser": "^4.2.2",
|
||||||
"highlight.js": "^11.8.0",
|
"highlight.js": "^11.8.0",
|
||||||
"intro.js": "^7.0.1",
|
"intro.js": "^7.0.1",
|
||||||
@@ -72,6 +72,7 @@
|
|||||||
"vue-i18n": "9.2.2",
|
"vue-i18n": "9.2.2",
|
||||||
"vue-router": "^4.2.1",
|
"vue-router": "^4.2.1",
|
||||||
"vue-types": "^5.0.3",
|
"vue-types": "^5.0.3",
|
||||||
|
"vue3-tree-org": "^4.2.2",
|
||||||
"vuedraggable": "^4.1.0",
|
"vuedraggable": "^4.1.0",
|
||||||
"web-storage-cache": "^1.1.1",
|
"web-storage-cache": "^1.1.1",
|
||||||
"xe-utils": "^3.5.7",
|
"xe-utils": "^3.5.7",
|
||||||
|
|||||||
42
src/api/kpi/index.js
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
import request from '@/config/axios'
|
||||||
|
|
||||||
|
// 创建
|
||||||
|
export const createKpi = (data) => {
|
||||||
|
return request.post({
|
||||||
|
url: '/admin-api/oa/examine-configuration/create',
|
||||||
|
data,
|
||||||
|
isSubmitForm: true
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 修改
|
||||||
|
export const updateKpi = (data) => {
|
||||||
|
return request.put({ url: '/admin-api/oa/examine-configuration/update', data })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 修改状态
|
||||||
|
export const updateKpiStatus = (data) => {
|
||||||
|
return request.put({ url: '/admin-api/oa/examine-configuration/status/update', data })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 分页
|
||||||
|
export const getKpiPage = (params) => {
|
||||||
|
return request.get({ url: '/admin-api/oa/examine-configuration/page', params })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 详情
|
||||||
|
export const getKpiDetail = (id) => {
|
||||||
|
return request.get({ url: '/admin-api/oa/examine-configuration/get', params: { id } })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 删除
|
||||||
|
export const deleteKpi = (id) => {
|
||||||
|
return request.delete({ url: '/admin-api/oa/examine-configuration/delete', params: { id } })
|
||||||
|
}
|
||||||
|
|
||||||
|
//
|
||||||
|
export const getKpiEmployees = () => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/oa/employee/examinedEmployees'
|
||||||
|
})
|
||||||
|
}
|
||||||
41
src/api/kpi/score.js
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
import request from '@/config/axios'
|
||||||
|
|
||||||
|
// 打分数据
|
||||||
|
export const getKpiEmployees = (params) => {
|
||||||
|
return request.get({ url: '/admin-api/oa/examine-score/getExamineUserExamineScore', params })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 保存考勤打分数据
|
||||||
|
export const saveKpiScore = (data) => {
|
||||||
|
return request.put({ url: '/admin-api/oa/examine-score/update', data })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 删除考核项
|
||||||
|
export const deleteKpiItem = (params) => {
|
||||||
|
return request.delete({ url: '/admin-api/oa/examine-score/delete', params })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 考评分页数据
|
||||||
|
export const getScorePage = (params) => {
|
||||||
|
return request.get({ url: '/admin-api/oa/examine-period/page', params })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 封存
|
||||||
|
export const saveScoreRecord = (params) => {
|
||||||
|
return request.get({ url: '/admin-api/oa/examine-period/confirm', params })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 查询周期是否已封存
|
||||||
|
export const getKpiPeriodStatus = (params) => {
|
||||||
|
return request.get({ url: '/admin-api/oa/examine-period/getExaminePeriodByPeriod', params })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 查询考核记录
|
||||||
|
export const getScoreRecord = (data) => {
|
||||||
|
return request.post({ url: '/admin-api/oa/examine-result/periodPage', data })
|
||||||
|
}
|
||||||
|
|
||||||
|
// 查详情
|
||||||
|
export const getScoreDetail = (params) => {
|
||||||
|
return request.get({ url: '/admin-api/oa/examine-result/detail', params })
|
||||||
|
}
|
||||||
38
src/api/okr/comment.js
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
import request from '@/config/axios'
|
||||||
|
|
||||||
|
// 创建
|
||||||
|
export const createComment = (data) => {
|
||||||
|
return request.post({
|
||||||
|
url: '/admin-api/okr/comments/create',
|
||||||
|
data,
|
||||||
|
isSubmitForm: true
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 分页
|
||||||
|
export const getCommentPage = (params) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/comments/page',
|
||||||
|
params
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取评论类型
|
||||||
|
export const getCommentTypeOptions = () => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/dict-data/get-by-type',
|
||||||
|
params: { dictType: 'comment_type' }
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 点赞评论
|
||||||
|
export const likeComment = (commentId) => {
|
||||||
|
return request.put({
|
||||||
|
url: '/admin-api/okr/comments-likes/update',
|
||||||
|
data: { commentId }
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
110
src/api/okr/okr.js
Normal file
@@ -0,0 +1,110 @@
|
|||||||
|
import request from '@/config/axios'
|
||||||
|
|
||||||
|
// 创建
|
||||||
|
export const createOkrNode = (data) => {
|
||||||
|
return request.post({
|
||||||
|
url: '/admin-api/okr/node/add',
|
||||||
|
data,
|
||||||
|
isSubmitForm: true
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 修改
|
||||||
|
export const updateOkrNode = (data) => {
|
||||||
|
return request.put({
|
||||||
|
url: '/admin-api/okr/node/update',
|
||||||
|
data
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 查询详情
|
||||||
|
export const getOkrNodeDetail = (nodeId) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/node/get',
|
||||||
|
params: { nodeId }
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 我负责的 - 节点树
|
||||||
|
export const getMyNodeTree = (params) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/node/my/list',
|
||||||
|
params
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 我负责的 - okr列表
|
||||||
|
export const getMyOkrPage = (params) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/node/my/object/list',
|
||||||
|
params
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 全部目标 - 节点树
|
||||||
|
export const getAllNodeTree = (params) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/node/all/list',
|
||||||
|
params
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 全部目标 - okr列表
|
||||||
|
export const getAllOkrPage = (params) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/node/all/object/list',
|
||||||
|
params
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取节点操作历史
|
||||||
|
export const getOkrNodeHistory = (nodeId) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/record/list',
|
||||||
|
params: { nodeId }
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取系统默认的关键成果内容
|
||||||
|
export const getDefaultOkrOptions = () => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/dict-data/get-by-type',
|
||||||
|
params: { dictType: 'key_result_source' }
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 更新OKR进度
|
||||||
|
export const updateOkrProgress = (data) => {
|
||||||
|
return request.put({
|
||||||
|
url: '/admin-api/okr/node/progress/update',
|
||||||
|
data
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取目标关系树一级节点
|
||||||
|
export const getOkrRelationTree = (params) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/node/first-node',
|
||||||
|
params
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取目标关系树下级节点数据
|
||||||
|
export const getOkrRelationTreeChildren = (params) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/node/node-tree',
|
||||||
|
params
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
83
src/api/okr/wait.js
Normal file
@@ -0,0 +1,83 @@
|
|||||||
|
import request from '@/config/axios'
|
||||||
|
|
||||||
|
// 创建
|
||||||
|
export const createWait = (data) => {
|
||||||
|
return request.post({
|
||||||
|
url: '/admin-api/okr/agent-work/create',
|
||||||
|
data,
|
||||||
|
isSubmitForm: true
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 修改
|
||||||
|
export const updateWait = (data) => {
|
||||||
|
return request.put({
|
||||||
|
url: '/admin-api/okr/agent-work/update',
|
||||||
|
data
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 分页
|
||||||
|
export const getWaitPage = (params) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/agent-work/page',
|
||||||
|
params
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 详情
|
||||||
|
export const getWaitDetail = (id) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/agent-work/get',
|
||||||
|
params: { id }
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 删除
|
||||||
|
export const deleteWait = (id) => {
|
||||||
|
return request.delete({
|
||||||
|
url: '/admin-api/okr/agent-work/delete',
|
||||||
|
params: { id }
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 催办
|
||||||
|
export const urgeWait = (workId) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/agent-work/urge',
|
||||||
|
params: { workId }
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 跟进待办
|
||||||
|
export const followWait = (data) => {
|
||||||
|
return request.post({
|
||||||
|
url: '/admin-api/okr/agent-work-follow/create',
|
||||||
|
data,
|
||||||
|
isSubmitForm: true
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
// 查询跟进记录
|
||||||
|
export const getFollowWaitPage = (params) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/agent-work-follow/list',
|
||||||
|
params
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export const getWaitCount = (params) => {
|
||||||
|
return request.get({
|
||||||
|
url: '/admin-api/okr/agent-work/getAgentWorkNum',
|
||||||
|
params
|
||||||
|
// headers: { 'instance-id': 1016 }
|
||||||
|
})
|
||||||
|
}
|
||||||
@@ -5,6 +5,11 @@ export const createEmployee = (data) => {
|
|||||||
return request.post({ url: '/admin-api/oa/employee/create', data, isSubmitForm: true })
|
return request.post({ url: '/admin-api/oa/employee/create', data, isSubmitForm: true })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 分页
|
||||||
|
export const getEmployeeSimpleList = (params) => {
|
||||||
|
return request.get({ url: '/admin-api/oa/employee/list-all-simple', params })
|
||||||
|
}
|
||||||
|
|
||||||
// 修改
|
// 修改
|
||||||
export const updateEmployee = (data) => {
|
export const updateEmployee = (data) => {
|
||||||
return request.put({ url: '/admin-api/oa/employee/update', data })
|
return request.put({ url: '/admin-api/oa/employee/update', data })
|
||||||
@@ -30,7 +35,7 @@ export const deleteEmployee = (id) => {
|
|||||||
return request.delete({ url: '/admin-api/oa/employee/delete', params: { id } })
|
return request.delete({ url: '/admin-api/oa/employee/delete', params: { id } })
|
||||||
}
|
}
|
||||||
|
|
||||||
// 详情
|
// 获取钉钉id
|
||||||
export const getDingUserId = (params) => {
|
export const getDingUserId = (params) => {
|
||||||
return request.get({ url: '/admin-api/oa/employee/getDingTalkUserIdByMobile', params })
|
return request.get({ url: '/admin-api/oa/employee/getDingTalkUserIdByMobile', params })
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -47,3 +47,8 @@ export const deleteDictData = (id: number) => {
|
|||||||
export const exportDictData = (params) => {
|
export const exportDictData = (params) => {
|
||||||
return request.get({ url: '/admin-api/oa/dict-data/export', params })
|
return request.get({ url: '/admin-api/oa/dict-data/export', params })
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 获取通用字典数据
|
||||||
|
export const getGeneralSysDictData = (dictType: string) => {
|
||||||
|
return request.get({ url: '/admin-api/system/dict-data/get-by-type', params: { dictType } })
|
||||||
|
}
|
||||||
|
|||||||
|
Before Width: | Height: | Size: 341 B |
|
Before Width: | Height: | Size: 357 B |
|
Before Width: | Height: | Size: 353 B |
|
Before Width: | Height: | Size: 350 B |
|
Before Width: | Height: | Size: 358 B |
|
Before Width: | Height: | Size: 355 B |
|
Before Width: | Height: | Size: 903 B |
|
Before Width: | Height: | Size: 958 B |
|
Before Width: | Height: | Size: 926 B |
|
Before Width: | Height: | Size: 926 B |
|
Before Width: | Height: | Size: 950 B |
|
Before Width: | Height: | Size: 970 B |
BIN
src/assets/imgs/login2.gif
Normal file
|
After Width: | Height: | Size: 309 KiB |
BIN
src/assets/imgs/shisong.jpg
Normal file
|
After Width: | Height: | Size: 192 KiB |
@@ -542,10 +542,10 @@ const submit = () => {
|
|||||||
<el-form>
|
<el-form>
|
||||||
<el-form-item label="类型">
|
<el-form-item label="类型">
|
||||||
<el-radio-group v-model="cronValue.second.type">
|
<el-radio-group v-model="cronValue.second.type">
|
||||||
<el-radio-button label="0">任意值</el-radio-button>
|
<el-radio-button value="0">任意值</el-radio-button>
|
||||||
<el-radio-button label="1">范围</el-radio-button>
|
<el-radio-button value="1">范围</el-radio-button>
|
||||||
<el-radio-button label="2">间隔</el-radio-button>
|
<el-radio-button value="2">间隔</el-radio-button>
|
||||||
<el-radio-button label="3">指定</el-radio-button>
|
<el-radio-button value="3">指定</el-radio-button>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item v-if="cronValue.second.type == '1'" label="范围">
|
<el-form-item v-if="cronValue.second.type == '1'" label="范围">
|
||||||
@@ -601,10 +601,10 @@ const submit = () => {
|
|||||||
<el-form>
|
<el-form>
|
||||||
<el-form-item label="类型">
|
<el-form-item label="类型">
|
||||||
<el-radio-group v-model="cronValue.minute.type">
|
<el-radio-group v-model="cronValue.minute.type">
|
||||||
<el-radio-button label="0">任意值</el-radio-button>
|
<el-radio-button value="0">任意值</el-radio-button>
|
||||||
<el-radio-button label="1">范围</el-radio-button>
|
<el-radio-button value="1">范围</el-radio-button>
|
||||||
<el-radio-button label="2">间隔</el-radio-button>
|
<el-radio-button value="2">间隔</el-radio-button>
|
||||||
<el-radio-button label="3">指定</el-radio-button>
|
<el-radio-button value="3">指定</el-radio-button>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item v-if="cronValue.minute.type == '1'" label="范围">
|
<el-form-item v-if="cronValue.minute.type == '1'" label="范围">
|
||||||
@@ -660,10 +660,10 @@ const submit = () => {
|
|||||||
<el-form>
|
<el-form>
|
||||||
<el-form-item label="类型">
|
<el-form-item label="类型">
|
||||||
<el-radio-group v-model="cronValue.hour.type">
|
<el-radio-group v-model="cronValue.hour.type">
|
||||||
<el-radio-button label="0">任意值</el-radio-button>
|
<el-radio-button value="0">任意值</el-radio-button>
|
||||||
<el-radio-button label="1">范围</el-radio-button>
|
<el-radio-button value="1">范围</el-radio-button>
|
||||||
<el-radio-button label="2">间隔</el-radio-button>
|
<el-radio-button value="2">间隔</el-radio-button>
|
||||||
<el-radio-button label="3">指定</el-radio-button>
|
<el-radio-button value="3">指定</el-radio-button>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item v-if="cronValue.hour.type == '1'" label="范围">
|
<el-form-item v-if="cronValue.hour.type == '1'" label="范围">
|
||||||
@@ -719,12 +719,12 @@ const submit = () => {
|
|||||||
<el-form>
|
<el-form>
|
||||||
<el-form-item label="类型">
|
<el-form-item label="类型">
|
||||||
<el-radio-group v-model="cronValue.day.type">
|
<el-radio-group v-model="cronValue.day.type">
|
||||||
<el-radio-button label="0">任意值</el-radio-button>
|
<el-radio-button value="0">任意值</el-radio-button>
|
||||||
<el-radio-button label="1">范围</el-radio-button>
|
<el-radio-button value="1">范围</el-radio-button>
|
||||||
<el-radio-button label="2">间隔</el-radio-button>
|
<el-radio-button value="2">间隔</el-radio-button>
|
||||||
<el-radio-button label="3">指定</el-radio-button>
|
<el-radio-button value="3">指定</el-radio-button>
|
||||||
<el-radio-button label="4">本月最后一天</el-radio-button>
|
<el-radio-button value="4">本月最后一天</el-radio-button>
|
||||||
<el-radio-button label="5">不指定</el-radio-button>
|
<el-radio-button value="5">不指定</el-radio-button>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item v-if="cronValue.day.type == '1'" label="范围">
|
<el-form-item v-if="cronValue.day.type == '1'" label="范围">
|
||||||
@@ -780,10 +780,10 @@ const submit = () => {
|
|||||||
<el-form>
|
<el-form>
|
||||||
<el-form-item label="类型">
|
<el-form-item label="类型">
|
||||||
<el-radio-group v-model="cronValue.month.type">
|
<el-radio-group v-model="cronValue.month.type">
|
||||||
<el-radio-button label="0">任意值</el-radio-button>
|
<el-radio-button value="0">任意值</el-radio-button>
|
||||||
<el-radio-button label="1">范围</el-radio-button>
|
<el-radio-button value="1">范围</el-radio-button>
|
||||||
<el-radio-button label="2">间隔</el-radio-button>
|
<el-radio-button value="2">间隔</el-radio-button>
|
||||||
<el-radio-button label="3">指定</el-radio-button>
|
<el-radio-button value="3">指定</el-radio-button>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item v-if="cronValue.month.type == '1'" label="范围">
|
<el-form-item v-if="cronValue.month.type == '1'" label="范围">
|
||||||
@@ -840,12 +840,12 @@ const submit = () => {
|
|||||||
<el-form>
|
<el-form>
|
||||||
<el-form-item label="类型">
|
<el-form-item label="类型">
|
||||||
<el-radio-group v-model="cronValue.week.type">
|
<el-radio-group v-model="cronValue.week.type">
|
||||||
<el-radio-button label="0">任意值</el-radio-button>
|
<el-radio-button value="0">任意值</el-radio-button>
|
||||||
<el-radio-button label="1">范围</el-radio-button>
|
<el-radio-button value="1">范围</el-radio-button>
|
||||||
<el-radio-button label="2">间隔</el-radio-button>
|
<el-radio-button value="2">间隔</el-radio-button>
|
||||||
<el-radio-button label="3">指定</el-radio-button>
|
<el-radio-button value="3">指定</el-radio-button>
|
||||||
<el-radio-button label="4">本月最后一周</el-radio-button>
|
<el-radio-button value="4">本月最后一周</el-radio-button>
|
||||||
<el-radio-button label="5">不指定</el-radio-button>
|
<el-radio-button value="5">不指定</el-radio-button>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item v-if="cronValue.week.type == '1'" label="范围">
|
<el-form-item v-if="cronValue.week.type == '1'" label="范围">
|
||||||
@@ -919,11 +919,11 @@ const submit = () => {
|
|||||||
<el-form>
|
<el-form>
|
||||||
<el-form-item label="类型">
|
<el-form-item label="类型">
|
||||||
<el-radio-group v-model="cronValue.year.type">
|
<el-radio-group v-model="cronValue.year.type">
|
||||||
<el-radio-button label="-1">忽略</el-radio-button>
|
<el-radio-button value="-1">忽略</el-radio-button>
|
||||||
<el-radio-button label="0">任意值</el-radio-button>
|
<el-radio-button value="0">任意值</el-radio-button>
|
||||||
<el-radio-button label="1">范围</el-radio-button>
|
<el-radio-button value="1">范围</el-radio-button>
|
||||||
<el-radio-button label="2">间隔</el-radio-button>
|
<el-radio-button value="2">间隔</el-radio-button>
|
||||||
<el-radio-button label="3">指定</el-radio-button>
|
<el-radio-button value="3">指定</el-radio-button>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item v-if="cronValue.year.type == '1'" label="范围">
|
<el-form-item v-if="cronValue.year.type == '1'" label="范围">
|
||||||
|
|||||||
@@ -24,7 +24,27 @@ const props = defineProps({
|
|||||||
default: () => undefined
|
default: () => undefined
|
||||||
},
|
},
|
||||||
readonly: propTypes.bool.def(false),
|
readonly: propTypes.bool.def(false),
|
||||||
modelValue: propTypes.string.def('')
|
modelValue: propTypes.string.def(''),
|
||||||
|
toolbarConfig: {
|
||||||
|
type: Object,
|
||||||
|
default: () => ({
|
||||||
|
excludeKeys: [
|
||||||
|
'insertVideo', // 网络视频
|
||||||
|
'insertImage', // 网络图片
|
||||||
|
'insertLink', // 链接
|
||||||
|
'codeBlock', // 代码块
|
||||||
|
'headerSelect', // 标题
|
||||||
|
'blockquote', // 引用
|
||||||
|
'fontFamily', // 字体
|
||||||
|
'todo', // 代办
|
||||||
|
'group-indent', // 缩进
|
||||||
|
'emotion', // 表情
|
||||||
|
'undo', // 撤销
|
||||||
|
'redo', // 重做
|
||||||
|
'fullScreen'
|
||||||
|
]
|
||||||
|
})
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
const emit = defineEmits(['change', 'update:modelValue'])
|
const emit = defineEmits(['change', 'update:modelValue'])
|
||||||
@@ -212,24 +232,6 @@ const editorStyle = computed(() => {
|
|||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
const toolbarConfig = ref({
|
|
||||||
excludeKeys: [
|
|
||||||
'insertVideo', // 网络视频
|
|
||||||
'insertImage', // 网络图片
|
|
||||||
'insertLink', // 链接
|
|
||||||
'codeBlock', // 代码块
|
|
||||||
'headerSelect', // 标题
|
|
||||||
'blockquote', // 引用
|
|
||||||
'fontFamily', // 字体
|
|
||||||
'todo', // 代办
|
|
||||||
'group-indent', // 缩进
|
|
||||||
'emotion', // 表情
|
|
||||||
'undo', // 撤销
|
|
||||||
'redo', // 重做
|
|
||||||
'fullScreen'
|
|
||||||
]
|
|
||||||
})
|
|
||||||
|
|
||||||
// 回调函数
|
// 回调函数
|
||||||
const handleChange = (editor: IDomEditor) => {
|
const handleChange = (editor: IDomEditor) => {
|
||||||
emit('change', editor)
|
emit('change', editor)
|
||||||
|
|||||||
@@ -4,7 +4,6 @@
|
|||||||
v-show="total > 0"
|
v-show="total > 0"
|
||||||
v-model:current-page="currentPage"
|
v-model:current-page="currentPage"
|
||||||
v-model:page-size="pageSize"
|
v-model:page-size="pageSize"
|
||||||
:small="small"
|
|
||||||
:background="true"
|
:background="true"
|
||||||
:page-sizes="[10, 20, 50, 100, 200]"
|
:page-sizes="[10, 20, 50, 100, 200]"
|
||||||
:pager-count="pagerCount"
|
:pager-count="pagerCount"
|
||||||
@@ -43,10 +42,6 @@ const props = defineProps({
|
|||||||
layout: {
|
layout: {
|
||||||
type: String,
|
type: String,
|
||||||
default: 'total, sizes, prev, pager, next, jumper'
|
default: 'total, sizes, prev, pager, next, jumper'
|
||||||
},
|
|
||||||
small: {
|
|
||||||
type: Boolean,
|
|
||||||
default: false
|
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import { config } from './config'
|
|||||||
const { default_headers } = config
|
const { default_headers } = config
|
||||||
|
|
||||||
const request = (option: any) => {
|
const request = (option: any) => {
|
||||||
const { url, method, params, data, headersType, responseType, isSubmitForm } = option
|
const { url, method, params, data, headersType, responseType, isSubmitForm, headers } = option
|
||||||
return service({
|
return service({
|
||||||
url: url,
|
url: url,
|
||||||
method,
|
method,
|
||||||
@@ -14,7 +14,8 @@ const request = (option: any) => {
|
|||||||
data,
|
data,
|
||||||
responseType: responseType,
|
responseType: responseType,
|
||||||
headers: {
|
headers: {
|
||||||
'Content-Type': headersType || default_headers
|
'Content-Type': headersType || default_headers,
|
||||||
|
...headers
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -66,7 +66,8 @@ service.interceptors.request.use(
|
|||||||
|
|
||||||
// 设置实例
|
// 设置实例
|
||||||
const appId = getAppId()
|
const appId = getAppId()
|
||||||
if (appId) (config as Recordable).headers['instance-id'] = appId
|
if (appId && !(config as Recordable).headers['instance-id'])
|
||||||
|
(config as Recordable).headers['instance-id'] = appId
|
||||||
|
|
||||||
const params = config.params || {}
|
const params = config.params || {}
|
||||||
const data = config.data || false
|
const data = config.data || false
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import router from './router'
|
import router from './router'
|
||||||
import { isRelogin } from '@/config/axios/service'
|
import { isRelogin } from '@/config/axios/service'
|
||||||
import { getAccessToken } from '@/utils/auth'
|
import { getAccessToken, removeToken } from '@/utils/auth'
|
||||||
import { useTitle } from '@/hooks/web/useTitle'
|
import { useTitle } from '@/hooks/web/useTitle'
|
||||||
import { useNProgress } from '@/hooks/web/useNProgress'
|
import { useNProgress } from '@/hooks/web/useNProgress'
|
||||||
import { usePageLoading } from '@/hooks/web/usePageLoading'
|
import { usePageLoading } from '@/hooks/web/usePageLoading'
|
||||||
|
|||||||
@@ -106,27 +106,41 @@ const remainingRouter: AppRouteRecordRaw[] = [
|
|||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
// {
|
{
|
||||||
// path: '/Basic',
|
path: '/Basic',
|
||||||
// component: Layout,
|
component: Layout,
|
||||||
// name: 'Basic',
|
name: 'Basic',
|
||||||
// meta: {},
|
meta: {
|
||||||
// redirect: '/Basic/menu',
|
title: '菜单管理'
|
||||||
// children: [
|
},
|
||||||
// {
|
redirect: '/Basic/menu',
|
||||||
// path: 'menu',
|
children: [
|
||||||
// component: () => import('@/views/Basic/Menu/index.vue'),
|
{
|
||||||
// name: 'Menu',
|
path: 'menu',
|
||||||
// meta: {
|
component: () => import('@/views/Basic/Menu/index.vue'),
|
||||||
// canTo: true,
|
name: 'Menu',
|
||||||
// hidden: true,
|
meta: {
|
||||||
// noTagsView: false,
|
canTo: true,
|
||||||
// icon: 'ep:user',
|
// hidden: true,
|
||||||
// title: '菜单管理'
|
noTagsView: false,
|
||||||
// }
|
icon: 'ep:user',
|
||||||
// }
|
title: '菜单管理'
|
||||||
// ]
|
}
|
||||||
// },
|
},
|
||||||
|
{
|
||||||
|
path: 'okr',
|
||||||
|
component: () => import('@/views/OKR/Management/index.vue'),
|
||||||
|
name: 'OkrManagement',
|
||||||
|
meta: {
|
||||||
|
canTo: true,
|
||||||
|
// hidden: true,
|
||||||
|
noTagsView: false,
|
||||||
|
icon: 'ep:user',
|
||||||
|
title: 'Okr管理'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
{
|
{
|
||||||
path: '/login',
|
path: '/login',
|
||||||
component: () => import('@/views/Login/Login.vue'),
|
component: () => import('@/views/Login/Login.vue'),
|
||||||
|
|||||||
@@ -69,3 +69,32 @@
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.dialog-okr {
|
||||||
|
width: 94vw;
|
||||||
|
height: 94vh;
|
||||||
|
max-width: 1800px;
|
||||||
|
max-height: 1000px;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
.dialog-okr .el-dialog__header {
|
||||||
|
padding: 0 !important;
|
||||||
|
}
|
||||||
|
.okr-info-dialog {
|
||||||
|
.el-dialog__body {
|
||||||
|
padding-top: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 去除 Chrome、Safari、Edge、Opera 中的指示器 */
|
||||||
|
.el-input__inner::-webkit-outer-spin-button,
|
||||||
|
.el-input__inner::-webkit-inner-spin-button {
|
||||||
|
-webkit-appearance: none;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 去除 Firefox 中的指示器 */
|
||||||
|
.el-input__inner[type='number'] {
|
||||||
|
-moz-appearance: textfield;
|
||||||
|
}
|
||||||
@@ -32,9 +32,10 @@
|
|||||||
<el-form-item label="负责人" prop="leaderUserId">
|
<el-form-item label="负责人" prop="leaderUserId">
|
||||||
<el-select v-model="formData.leaderUserId" clearable placeholder="请输入负责人">
|
<el-select v-model="formData.leaderUserId" clearable placeholder="请输入负责人">
|
||||||
<el-option
|
<el-option
|
||||||
v-for="item in userList"
|
v-for="item in employeeOptions"
|
||||||
:key="item.id"
|
:key="item.id"
|
||||||
:label="item.nickname"
|
:disabled="item.status == 1"
|
||||||
|
:label="item.name"
|
||||||
:value="item.id"
|
:value="item.id"
|
||||||
/>
|
/>
|
||||||
</el-select>
|
</el-select>
|
||||||
@@ -43,8 +44,8 @@
|
|||||||
<el-col :span="12" :offset="0">
|
<el-col :span="12" :offset="0">
|
||||||
<el-form-item label="状态" prop="status">
|
<el-form-item label="状态" prop="status">
|
||||||
<el-radio-group v-model="formData.status">
|
<el-radio-group v-model="formData.status">
|
||||||
<el-radio :label="0"> 启用 </el-radio>
|
<el-radio :value="0"> 启用 </el-radio>
|
||||||
<el-radio :label="1"> 禁用 </el-radio>
|
<el-radio :value="1"> 禁用 </el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
@@ -55,6 +56,26 @@
|
|||||||
<el-input-number v-model="formData.sort" :min="0" />
|
<el-input-number v-model="formData.sort" :min="0" />
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
|
<el-col :span="12" :offset="0">
|
||||||
|
<el-form-item label="开通系统" prop="instanceIds">
|
||||||
|
<el-select
|
||||||
|
v-model="formData.instanceIds"
|
||||||
|
placeholder="选择开通系统"
|
||||||
|
multiple
|
||||||
|
collapse-tags
|
||||||
|
clearable
|
||||||
|
filterable
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="item in instanceIdsOptions"
|
||||||
|
:key="item.instanceId"
|
||||||
|
:label="item.instanceName"
|
||||||
|
:disabled="item.status == 1"
|
||||||
|
:value="item.instanceId"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
</el-row>
|
</el-row>
|
||||||
<el-row :gutter="20">
|
<el-row :gutter="20">
|
||||||
<el-col :span="24" :offset="0">
|
<el-col :span="24" :offset="0">
|
||||||
@@ -73,8 +94,9 @@
|
|||||||
<script lang="ts" name="SystemDeptForm" setup>
|
<script lang="ts" name="SystemDeptForm" setup>
|
||||||
import { defaultProps, handleTree } from '@/utils/tree'
|
import { defaultProps, handleTree } from '@/utils/tree'
|
||||||
import * as DeptApi from '@/api/system/dept'
|
import * as DeptApi from '@/api/system/dept'
|
||||||
import * as UserApi from '@/api/system/user'
|
import { getEmployeeSimpleList } from '@/api/pers/employee'
|
||||||
import { CommonStatusEnum } from '@/utils/constants'
|
import { CommonStatusEnum } from '@/utils/constants'
|
||||||
|
import { getSimpleAppList } from '@/api/system/app'
|
||||||
|
|
||||||
const { t } = useI18n() // 国际化
|
const { t } = useI18n() // 国际化
|
||||||
const message = useMessage() // 消息弹窗
|
const message = useMessage() // 消息弹窗
|
||||||
@@ -90,7 +112,8 @@ const formData = ref({
|
|||||||
sort: 1,
|
sort: 1,
|
||||||
leaderUserId: undefined,
|
leaderUserId: undefined,
|
||||||
status: CommonStatusEnum.ENABLE,
|
status: CommonStatusEnum.ENABLE,
|
||||||
remark: undefined
|
remark: undefined,
|
||||||
|
instanceIds: undefined
|
||||||
})
|
})
|
||||||
const formRules = reactive<any>({
|
const formRules = reactive<any>({
|
||||||
parentId: [{ required: true, message: '上级部门不能为空', trigger: 'blur' }],
|
parentId: [{ required: true, message: '上级部门不能为空', trigger: 'blur' }],
|
||||||
@@ -104,7 +127,8 @@ const formRules = reactive<any>({
|
|||||||
})
|
})
|
||||||
const formRef = ref() // 表单 Ref
|
const formRef = ref() // 表单 Ref
|
||||||
const deptTree = ref() // 树形结构
|
const deptTree = ref() // 树形结构
|
||||||
const userList = ref<UserApi.UserVO[]>([]) // 用户列表
|
const employeeOptions = ref<any>([]) // 用户列表
|
||||||
|
const instanceIdsOptions = ref<any>([]) // 实例ids
|
||||||
|
|
||||||
/** 打开弹窗 */
|
/** 打开弹窗 */
|
||||||
const open = async (type: string, id?: number) => {
|
const open = async (type: string, id?: number) => {
|
||||||
@@ -122,7 +146,13 @@ const open = async (type: string, id?: number) => {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// 获得用户列表
|
// 获得用户列表
|
||||||
userList.value = await UserApi.getSimpleUserList()
|
getEmployeeSimpleList().then((data) => {
|
||||||
|
employeeOptions.value = data
|
||||||
|
})
|
||||||
|
//实例
|
||||||
|
getSimpleAppList().then((data) => {
|
||||||
|
instanceIdsOptions.value = data
|
||||||
|
})
|
||||||
// 获得部门树
|
// 获得部门树
|
||||||
await getTree()
|
await getTree()
|
||||||
}
|
}
|
||||||
@@ -163,7 +193,8 @@ const resetForm = () => {
|
|||||||
sort: 1,
|
sort: 1,
|
||||||
leaderUserId: undefined,
|
leaderUserId: undefined,
|
||||||
status: CommonStatusEnum.ENABLE,
|
status: CommonStatusEnum.ENABLE,
|
||||||
remark: undefined
|
remark: undefined,
|
||||||
|
instanceIds: undefined
|
||||||
}
|
}
|
||||||
formRef.value?.resetFields()
|
formRef.value?.resetFields()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -22,8 +22,15 @@
|
|||||||
<el-table v-loading="loading" :data="list" row-key="id" default-expand-all border>
|
<el-table v-loading="loading" :data="list" row-key="id" default-expand-all border>
|
||||||
<el-table-column prop="name" label="部门名称" />
|
<el-table-column prop="name" label="部门名称" />
|
||||||
<el-table-column prop="leaderUserName" label="负责人" width="120" />
|
<el-table-column prop="leaderUserName" label="负责人" width="120" />
|
||||||
<el-table-column prop="sort" label="排序" width="200" />
|
<el-table-column prop="instanceNames" label="业务系统" width="200" />
|
||||||
<el-table-column prop="status" label="状态" width="100" />
|
<el-table-column prop="sort" label="排序" width="80" />
|
||||||
|
<el-table-column prop="status" label="状态" width="100">
|
||||||
|
<template #default="scope">
|
||||||
|
<el-tag :type="scope.row.status == 0 ? 'success' : 'danger'">
|
||||||
|
{{ scope.row.status == 0 ? '正常' : '停用' }}
|
||||||
|
</el-tag>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
<el-table-column label="创建时间" prop="createTime" width="180" :formatter="dateFormatter" />
|
<el-table-column label="创建时间" prop="createTime" width="180" :formatter="dateFormatter" />
|
||||||
<el-table-column label="操作" class-name="fixed-width" width="160">
|
<el-table-column label="操作" class-name="fixed-width" width="160">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
|
|||||||
@@ -32,9 +32,9 @@
|
|||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item label="菜单类型" prop="type">
|
<el-form-item label="菜单类型" prop="type">
|
||||||
<el-radio-group v-model="formData.type">
|
<el-radio-group v-model="formData.type">
|
||||||
<el-radio-button :label="1"> 目录 </el-radio-button>
|
<el-radio-button :value="1"> 目录 </el-radio-button>
|
||||||
<el-radio-button :label="2"> 菜单 </el-radio-button>
|
<el-radio-button :value="2"> 菜单 </el-radio-button>
|
||||||
<el-radio-button :label="3"> 按钮 </el-radio-button>
|
<el-radio-button :value="3"> 按钮 </el-radio-button>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item v-if="formData.type !== 3" label="菜单图标">
|
<el-form-item v-if="formData.type !== 3" label="菜单图标">
|
||||||
@@ -69,8 +69,8 @@
|
|||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item label="菜单状态" prop="status">
|
<el-form-item label="菜单状态" prop="status">
|
||||||
<el-radio-group v-model="formData.status">
|
<el-radio-group v-model="formData.status">
|
||||||
<el-radio :label="0"> 启用 </el-radio>
|
<el-radio :value="0"> 启用 </el-radio>
|
||||||
<el-radio :label="1"> 禁用 </el-radio>
|
<el-radio :value="1"> 禁用 </el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item v-if="formData.type !== 3" label="显示状态" prop="visible">
|
<el-form-item v-if="formData.type !== 3" label="显示状态" prop="visible">
|
||||||
|
|||||||
@@ -16,8 +16,8 @@
|
|||||||
<el-col :span="12">
|
<el-col :span="12">
|
||||||
<el-form-item label="用户性别">
|
<el-form-item label="用户性别">
|
||||||
<el-radio-group v-model="formData.sex">
|
<el-radio-group v-model="formData.sex">
|
||||||
<el-radio :label="1"> 男 </el-radio>
|
<el-radio :value="1"> 男 </el-radio>
|
||||||
<el-radio :label="2"> 女 </el-radio>
|
<el-radio :value="2"> 女 </el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
|
|||||||
@@ -17,10 +17,10 @@
|
|||||||
<el-col :span="12" :offset="0">
|
<el-col :span="12" :offset="0">
|
||||||
<el-form-item label="结算方式">
|
<el-form-item label="结算方式">
|
||||||
<el-radio-group v-model="formData.percentageType">
|
<el-radio-group v-model="formData.percentageType">
|
||||||
<el-radio :label="1">
|
<el-radio :value="1">
|
||||||
<Tooltip message="可配置多级,命中某档位后,按照档位分级结算" />阶梯结算
|
<Tooltip message="可配置多级,命中某档位后,按照档位分级结算" />阶梯结算
|
||||||
</el-radio>
|
</el-radio>
|
||||||
<el-radio :label="2">
|
<el-radio :value="2">
|
||||||
<Tooltip message="可配置多级,命中某档位后所有金额全部按照该档位结算" />常规结算
|
<Tooltip message="可配置多级,命中某档位后所有金额全部按照该档位结算" />常规结算
|
||||||
</el-radio>
|
</el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
@@ -164,8 +164,8 @@
|
|||||||
<el-col :span="8" :offset="0">
|
<el-col :span="8" :offset="0">
|
||||||
<el-form-item label="包含区域提成" labelWidth="150px">
|
<el-form-item label="包含区域提成" labelWidth="150px">
|
||||||
<el-radio-group v-model="formData.isAreaPercentage">
|
<el-radio-group v-model="formData.isAreaPercentage">
|
||||||
<el-radio :label="true"> 是 </el-radio>
|
<el-radio :value="true"> 是 </el-radio>
|
||||||
<el-radio :label="false"> 否 </el-radio>
|
<el-radio :value="false"> 否 </el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
@@ -174,24 +174,24 @@
|
|||||||
<el-col :span="8" :offset="0">
|
<el-col :span="8" :offset="0">
|
||||||
<el-form-item label="是否关联成交率" labelWidth="150px">
|
<el-form-item label="是否关联成交率" labelWidth="150px">
|
||||||
<el-radio-group v-model="formData.isRelateSignRate">
|
<el-radio-group v-model="formData.isRelateSignRate">
|
||||||
<el-radio :label="true"> 是 </el-radio>
|
<el-radio :value="true"> 是 </el-radio>
|
||||||
<el-radio :label="false"> 否 </el-radio>
|
<el-radio :value="false"> 否 </el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
<el-col :span="8" :offset="0">
|
<el-col :span="8" :offset="0">
|
||||||
<el-form-item label="是否扣除其他支出" labelWidth="150px">
|
<el-form-item label="是否扣除其他支出" labelWidth="150px">
|
||||||
<el-radio-group v-model="formData.isDeductExtraPay">
|
<el-radio-group v-model="formData.isDeductExtraPay">
|
||||||
<el-radio :label="true"> 是 </el-radio>
|
<el-radio :value="true"> 是 </el-radio>
|
||||||
<el-radio :label="false"> 否 </el-radio>
|
<el-radio :value="false"> 否 </el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
<el-col :span="8" :offset="0">
|
<el-col :span="8" :offset="0">
|
||||||
<el-form-item label="是否扣除售后" labelWidth="150px">
|
<el-form-item label="是否扣除售后" labelWidth="150px">
|
||||||
<el-radio-group v-model="formData.isDeductAfterSale">
|
<el-radio-group v-model="formData.isDeductAfterSale">
|
||||||
<el-radio :label="true"> 是 </el-radio>
|
<el-radio :value="true"> 是 </el-radio>
|
||||||
<el-radio :label="false"> 否 </el-radio>
|
<el-radio :value="false"> 否 </el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
|
|||||||
@@ -18,8 +18,8 @@
|
|||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
<el-col :span="12" :offset="0">
|
<el-col :span="12" :offset="0">
|
||||||
<el-form-item label="奖金" prop="rewardSalary">
|
<el-form-item label="绩效(%)">
|
||||||
<el-input-number v-model="formData.rewardSalary" :min="0" :controls="false" />
|
<el-input-number v-model="formData.meritsRate" :min="0" :max="100" :controls="false" />
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
</el-row>
|
</el-row>
|
||||||
@@ -49,6 +49,11 @@
|
|||||||
<el-input-number v-model="formData.fullWorkSalary" :min="0" :controls="false" />
|
<el-input-number v-model="formData.fullWorkSalary" :min="0" :controls="false" />
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
|
<el-col :span="12" :offset="0">
|
||||||
|
<el-form-item label="奖金" prop="rewardSalary">
|
||||||
|
<el-input-number v-model="formData.rewardSalary" :min="0" :controls="false" />
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
<el-col :span="12" :offset="0">
|
<el-col :span="12" :offset="0">
|
||||||
<el-form-item label="其他应发" prop="extraSalary">
|
<el-form-item label="其他应发" prop="extraSalary">
|
||||||
<el-input-number v-model="formData.extraSalary" :min="0" :controls="false" />
|
<el-input-number v-model="formData.extraSalary" :min="0" :controls="false" />
|
||||||
@@ -57,16 +62,11 @@
|
|||||||
</el-row>
|
</el-row>
|
||||||
<el-divider direction="horizontal" content-position="left">应扣</el-divider>
|
<el-divider direction="horizontal" content-position="left">应扣</el-divider>
|
||||||
<el-row :gutter="20">
|
<el-row :gutter="20">
|
||||||
<el-col :span="12" :offset="0">
|
<!-- <el-col :span="12" :offset="0">
|
||||||
<el-form-item label="绩效">
|
<el-form-item label="绩效">
|
||||||
<el-input placeholder="生成后录入" disabled />
|
<el-input placeholder="生成后录入" disabled />
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col> -->
|
||||||
<el-col :span="12" :offset="0">
|
|
||||||
<el-form-item label="考勤">
|
|
||||||
<el-input placeholder="自动计算" disabled />
|
|
||||||
</el-form-item>
|
|
||||||
</el-col>
|
|
||||||
</el-row>
|
</el-row>
|
||||||
<el-row :gutter="20">
|
<el-row :gutter="20">
|
||||||
<el-col :span="12" :offset="0">
|
<el-col :span="12" :offset="0">
|
||||||
@@ -86,6 +86,11 @@
|
|||||||
<el-input placeholder="自动计算" disabled />
|
<el-input placeholder="自动计算" disabled />
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
|
<el-col :span="12" :offset="0">
|
||||||
|
<el-form-item label="考勤">
|
||||||
|
<el-input placeholder="自动计算" disabled />
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
<el-col :span="12" :offset="0">
|
<el-col :span="12" :offset="0">
|
||||||
<el-form-item label="其他应扣" prop="extraDeduct">
|
<el-form-item label="其他应扣" prop="extraDeduct">
|
||||||
<el-input-number v-model="formData.extraDeduct" :min="0" :controls="false" />
|
<el-input-number v-model="formData.extraDeduct" :min="0" :controls="false" />
|
||||||
@@ -127,9 +132,11 @@ const open = async (row) => {
|
|||||||
getOptions()
|
getOptions()
|
||||||
formLoading.value = true
|
formLoading.value = true
|
||||||
try {
|
try {
|
||||||
formData.value = await SalaryApi.getSalaryDetail(row.id)
|
const data = await SalaryApi.getSalaryDetail(row.id)
|
||||||
if (!formData.value) {
|
if (!data) {
|
||||||
resetForm(row.id)
|
resetForm(row.id)
|
||||||
|
} else {
|
||||||
|
formData.value = { ...data }
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
formLoading.value = false
|
formLoading.value = false
|
||||||
@@ -157,7 +164,8 @@ const resetForm = (id) => {
|
|||||||
socialDeduct: 0,
|
socialDeduct: 0,
|
||||||
accumulationFundDeduct: 0,
|
accumulationFundDeduct: 0,
|
||||||
meritsSalary: 0,
|
meritsSalary: 0,
|
||||||
extraDeduct: 0
|
extraDeduct: 0,
|
||||||
|
meritsRate: 0
|
||||||
}
|
}
|
||||||
formRef.value?.resetFields()
|
formRef.value?.resetFields()
|
||||||
}
|
}
|
||||||
@@ -172,7 +180,7 @@ const submitForm = async () => {
|
|||||||
// 提交请求
|
// 提交请求
|
||||||
formLoading.value = true
|
formLoading.value = true
|
||||||
try {
|
try {
|
||||||
const data = formData.value
|
const data = { ...formData.value }
|
||||||
await SalaryApi.updateSalary(data)
|
await SalaryApi.updateSalary(data)
|
||||||
message.success('配置成功')
|
message.success('配置成功')
|
||||||
dialogVisible.value = false
|
dialogVisible.value = false
|
||||||
|
|||||||
@@ -11,8 +11,8 @@
|
|||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
<el-radio-group v-model="searchForm.status" @change="handleQuery">
|
<el-radio-group v-model="searchForm.status" @change="handleQuery">
|
||||||
<el-radio :label="0"> 在职 </el-radio>
|
<el-radio :value="0"> 在职 </el-radio>
|
||||||
<el-radio :label="1"> 离职 </el-radio>
|
<el-radio :value="1"> 离职 </el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
|
|||||||
@@ -48,7 +48,7 @@
|
|||||||
:tree-props="{ children: 'userDingAttendanceRespVOList', hasChildren: 'hasChildren' }"
|
:tree-props="{ children: 'userDingAttendanceRespVOList', hasChildren: 'hasChildren' }"
|
||||||
:span-method="spanMethod"
|
:span-method="spanMethod"
|
||||||
>
|
>
|
||||||
<el-table-column label="姓名" min-width="90px">
|
<el-table-column label="姓名" min-width="90px" fixed="left">
|
||||||
<template #default="{ row }">
|
<template #default="{ row }">
|
||||||
<span v-if="row.userDingAttendanceRespVOList && row.userDingAttendanceRespVOList.length">
|
<span v-if="row.userDingAttendanceRespVOList && row.userDingAttendanceRespVOList.length">
|
||||||
<span>{{ row.period }}</span>
|
<span>{{ row.period }}</span>
|
||||||
@@ -67,7 +67,8 @@
|
|||||||
{{ row.needAttendance ? '考勤' : '不考勤' }}
|
{{ row.needAttendance ? '考勤' : '不考勤' }}
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column prop="shouldWorkDay" label="计薪天数" min-width="90px" />
|
<el-table-column prop="shouldWorkDay" label="应出勤天数" min-width="90px" />
|
||||||
|
<el-table-column prop="paidLeaveDay" label="带薪假天数" min-width="90px" />
|
||||||
<el-table-column label="出勤天数" align="center">
|
<el-table-column label="出勤天数" align="center">
|
||||||
<template #default="{ row }">
|
<template #default="{ row }">
|
||||||
<el-input-number
|
<el-input-number
|
||||||
@@ -268,6 +269,9 @@ async function getList() {
|
|||||||
} else {
|
} else {
|
||||||
tableList.value = data.list.map((it, index) => ({
|
tableList.value = data.list.map((it, index) => ({
|
||||||
...it,
|
...it,
|
||||||
|
userDingAttendanceRespVOList: it.userDingAttendanceRespVOList.sort((pre, cur) =>
|
||||||
|
pre.employeeName.localeCompare(cur.employeeName)
|
||||||
|
),
|
||||||
id: index + 1,
|
id: index + 1,
|
||||||
edit: it.status == 1 ? '2' : '0'
|
edit: it.status == 1 ? '2' : '0'
|
||||||
}))
|
}))
|
||||||
@@ -283,8 +287,8 @@ async function getList() {
|
|||||||
function spanMethod({ row, columnIndex }) {
|
function spanMethod({ row, columnIndex }) {
|
||||||
if (row.userDingAttendanceRespVOList && row.userDingAttendanceRespVOList.length > 0) {
|
if (row.userDingAttendanceRespVOList && row.userDingAttendanceRespVOList.length > 0) {
|
||||||
if (columnIndex === 0) {
|
if (columnIndex === 0) {
|
||||||
return [1, 13]
|
return [1, 14]
|
||||||
} else if (columnIndex == 13 && row.id) {
|
} else if (columnIndex == 14 && row.id) {
|
||||||
return [1, 1]
|
return [1, 1]
|
||||||
} else {
|
} else {
|
||||||
return [0, 0]
|
return [0, 0]
|
||||||
|
|||||||
@@ -41,7 +41,7 @@
|
|||||||
:tree-props="{ children: 'userSalaryGrantRespVOList', hasChildren: 'hasChildren' }"
|
:tree-props="{ children: 'userSalaryGrantRespVOList', hasChildren: 'hasChildren' }"
|
||||||
:span-method="spanMethod"
|
:span-method="spanMethod"
|
||||||
>
|
>
|
||||||
<el-table-column label="姓名" min-width="90px">
|
<el-table-column label="姓名" min-width="90px" fixed="left">
|
||||||
<template #default="{ row }">
|
<template #default="{ row }">
|
||||||
<span v-if="row.userSalaryGrantRespVOList && row.userSalaryGrantRespVOList.length">
|
<span v-if="row.userSalaryGrantRespVOList && row.userSalaryGrantRespVOList.length">
|
||||||
<span>{{ row.period }}</span>
|
<span>{{ row.period }}</span>
|
||||||
@@ -57,7 +57,7 @@
|
|||||||
<el-table-column prop="dept" label="部门" min-width="90px" />
|
<el-table-column prop="dept" label="部门" min-width="90px" />
|
||||||
<el-table-column prop="post" label="职位" min-width="90px" />
|
<el-table-column prop="post" label="职位" min-width="90px" />
|
||||||
<el-table-column label="应发工资" align="center">
|
<el-table-column label="应发工资" align="center">
|
||||||
<el-table-column label="基本工资" min-width="90px">
|
<el-table-column label="无责底薪" min-width="90px">
|
||||||
<template #default="{ row }">
|
<template #default="{ row }">
|
||||||
<el-input-number
|
<el-input-number
|
||||||
v-if="row.edit"
|
v-if="row.edit"
|
||||||
@@ -70,7 +70,7 @@
|
|||||||
<span v-else> {{ row.baseSalary }}</span>
|
<span v-else> {{ row.baseSalary }}</span>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
<el-table-column label="绩效" min-width="90px">
|
<el-table-column label="标准绩效" min-width="90px">
|
||||||
<template #default="{ row }">
|
<template #default="{ row }">
|
||||||
<el-input-number
|
<el-input-number
|
||||||
v-if="row.edit"
|
v-if="row.edit"
|
||||||
@@ -83,6 +83,19 @@
|
|||||||
<span v-else> {{ row.meritsSalary }}</span>
|
<span v-else> {{ row.meritsSalary }}</span>
|
||||||
</template>
|
</template>
|
||||||
</el-table-column>
|
</el-table-column>
|
||||||
|
<el-table-column label="绩效实发" min-width="90px">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-input-number
|
||||||
|
v-if="row.edit"
|
||||||
|
v-model="row.reallyMeritsSalary"
|
||||||
|
:min="0"
|
||||||
|
:controls="false"
|
||||||
|
size="small"
|
||||||
|
style="width: 65px"
|
||||||
|
/>
|
||||||
|
<span v-else> {{ row.reallyMeritsSalary }}</span>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
<el-table-column label="补贴" min-width="90px">
|
<el-table-column label="补贴" min-width="90px">
|
||||||
<template #default="{ row }">
|
<template #default="{ row }">
|
||||||
<el-input-number
|
<el-input-number
|
||||||
@@ -364,6 +377,9 @@ async function getList() {
|
|||||||
} else {
|
} else {
|
||||||
tableList.value = data.list.map((it, index) => ({
|
tableList.value = data.list.map((it, index) => ({
|
||||||
...it,
|
...it,
|
||||||
|
userSalaryGrantRespVOList: it.userSalaryGrantRespVOList.sort((pre, cur) =>
|
||||||
|
pre.name.localeCompare(cur.name)
|
||||||
|
),
|
||||||
id: index + 1,
|
id: index + 1,
|
||||||
edit: it.status == 1 ? '2' : '0'
|
edit: it.status == 1 ? '2' : '0'
|
||||||
}))
|
}))
|
||||||
@@ -389,8 +405,8 @@ function importHistorySalary() {
|
|||||||
function spanMethod({ row, columnIndex }) {
|
function spanMethod({ row, columnIndex }) {
|
||||||
if (row.userSalaryGrantRespVOList && row.userSalaryGrantRespVOList.length > 0) {
|
if (row.userSalaryGrantRespVOList && row.userSalaryGrantRespVOList.length > 0) {
|
||||||
if (columnIndex === 0) {
|
if (columnIndex === 0) {
|
||||||
return [1, 22]
|
return [1, 23]
|
||||||
} else if (columnIndex == 22 && row.id) {
|
} else if (columnIndex == 23 && row.id) {
|
||||||
return [1, 1]
|
return [1, 1]
|
||||||
} else {
|
} else {
|
||||||
return [0, 0]
|
return [0, 0]
|
||||||
|
|||||||
225
src/views/Kpi/Appraise/Components/DialogAppraise.vue
Normal file
@@ -0,0 +1,225 @@
|
|||||||
|
<template>
|
||||||
|
<Dialog :title="title" v-model="show" width="800px">
|
||||||
|
<el-form v-loading="formLoading" :model="form" ref="formRef" :rules="rules" label-width="80px">
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="12" :offset="0">
|
||||||
|
<el-form-item label="考核指标" prop="examineTarget">
|
||||||
|
<el-input v-model="form.examineTarget" placeholder="请输入" />
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="12" :offset="0">
|
||||||
|
<el-form-item label="生效日期" prop="effectiveDate">
|
||||||
|
<el-date-picker
|
||||||
|
v-model="form.effectiveDate"
|
||||||
|
type="date"
|
||||||
|
:disabled="form.id"
|
||||||
|
format="YYYY-MM-DD"
|
||||||
|
value-format="YYYY-MM-DD"
|
||||||
|
placeholder="选择日期时间"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="24" :offset="0">
|
||||||
|
<el-form-item label="模式" prop="type">
|
||||||
|
<el-radio-group v-model="form.type">
|
||||||
|
<el-radio
|
||||||
|
v-for="(item, index) in prop.kpiModeOoptions"
|
||||||
|
:key="index"
|
||||||
|
:label="item.value"
|
||||||
|
>
|
||||||
|
{{ item.label }}
|
||||||
|
</el-radio>
|
||||||
|
</el-radio-group>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="12" :offset="0">
|
||||||
|
<el-form-item label="权重" prop="weight">
|
||||||
|
<el-input-number v-model="form.weight" :min="0" :step="1" :controls="false" />
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="12" :offset="0">
|
||||||
|
<el-form-item label="评分上限" prop="examineScore">
|
||||||
|
<el-input-number v-model="form.examineScore" :min="0" :step="1" :controls="false" />
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="24" :offset="0">
|
||||||
|
<el-form-item label="考核内容" prop="examineContent">
|
||||||
|
<Editor
|
||||||
|
v-model:modelValue="form.examineContent"
|
||||||
|
height="150px"
|
||||||
|
:toolbarConfig="{
|
||||||
|
toolbarKeys: []
|
||||||
|
}"
|
||||||
|
style="width: 100%"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="24" :offset="0">
|
||||||
|
<el-form-item label="考核规则" prop="examineRule">
|
||||||
|
<Editor
|
||||||
|
v-model:modelValue="form.examineRule"
|
||||||
|
height="150px"
|
||||||
|
:toolbarConfig="{ toolbarKeys: [] }"
|
||||||
|
style="width: 100%"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="24" :offset="0">
|
||||||
|
<el-form-item label="考核员工" prop="examinedUserIdList">
|
||||||
|
<div>
|
||||||
|
<el-checkbox
|
||||||
|
v-model="checkAll"
|
||||||
|
:indeterminate="isIndeterminate"
|
||||||
|
@change="handleCheckAllChange"
|
||||||
|
>
|
||||||
|
全选
|
||||||
|
</el-checkbox>
|
||||||
|
<el-checkbox-group v-model="form.examinedUserIdList" @change="handleCheckedChange">
|
||||||
|
<el-checkbox
|
||||||
|
v-for="item in employeeOptions"
|
||||||
|
:key="item.id"
|
||||||
|
:disabled="item.status == 1"
|
||||||
|
:label="item.id"
|
||||||
|
:value="item.id"
|
||||||
|
>
|
||||||
|
{{ item.name }}
|
||||||
|
</el-checkbox>
|
||||||
|
</el-checkbox-group>
|
||||||
|
</div>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
</el-form>
|
||||||
|
<template #footer>
|
||||||
|
<span>
|
||||||
|
<el-button @click="show = false">取 消</el-button>
|
||||||
|
<el-button type="primary" :disabled="formLoading" @click="handleSave">保 存</el-button>
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
</Dialog>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="DialogAppraise">
|
||||||
|
import * as KpiApi from '@/api/kpi/index.js'
|
||||||
|
|
||||||
|
const prop = defineProps({
|
||||||
|
kpiModeOoptions: {
|
||||||
|
type: Array,
|
||||||
|
default: () => []
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const { t } = useI18n() // 国际化
|
||||||
|
const message = useMessage() // 消息弹窗
|
||||||
|
|
||||||
|
const show = ref(false)
|
||||||
|
const title = ref('')
|
||||||
|
const formType = ref('create')
|
||||||
|
|
||||||
|
const form = ref({})
|
||||||
|
const formLoading = ref(false)
|
||||||
|
const rules = ref({
|
||||||
|
examineTarget: { required: true, message: '标题不可为空', trigger: 'blur' },
|
||||||
|
effectiveDate: { required: true, message: '生效日期不可为空', trigger: 'blur,change' },
|
||||||
|
examineContent: { required: true, message: '标题不可为空', trigger: 'blur' },
|
||||||
|
examineRule: { required: true, message: '标题不可为空', trigger: 'blur' }
|
||||||
|
})
|
||||||
|
|
||||||
|
async function open(type, val) {
|
||||||
|
show.value = true
|
||||||
|
title.value = type == 'update' ? '修改考核项' : '新增考核项'
|
||||||
|
formType.value = type
|
||||||
|
resetForm()
|
||||||
|
if (val) {
|
||||||
|
formLoading.value = true
|
||||||
|
try {
|
||||||
|
form.value = await KpiApi.getKpiDetail(val)
|
||||||
|
} finally {
|
||||||
|
formLoading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
getOptions()
|
||||||
|
}
|
||||||
|
|
||||||
|
function getOptions() {
|
||||||
|
KpiApi.getKpiEmployees().then((data) => {
|
||||||
|
employeeOptions.value = data
|
||||||
|
if (formType.value == 'update') {
|
||||||
|
handleCheckedChange(form.value.examinedUserIdList)
|
||||||
|
} else {
|
||||||
|
handleCheckAllChange(true)
|
||||||
|
checkAll.value = true
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function resetForm() {
|
||||||
|
form.value = {
|
||||||
|
examineTarget: '',
|
||||||
|
type: '3',
|
||||||
|
weight: 0,
|
||||||
|
examineContent: ``,
|
||||||
|
examineRule: ``,
|
||||||
|
examineScore: 5,
|
||||||
|
examinedUserIdList: []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
defineExpose({ open }) // 提供 open 方法,用于打开弹窗
|
||||||
|
|
||||||
|
const formRef = ref()
|
||||||
|
|
||||||
|
/** 提交表单 */
|
||||||
|
const emit = defineEmits(['success']) // 定义 success 事件,用于操作成功后的回调
|
||||||
|
async function handleSave() {
|
||||||
|
// 校验表单
|
||||||
|
if (!formRef.value) return
|
||||||
|
const valid = await formRef.value.validate()
|
||||||
|
if (!valid) return
|
||||||
|
// 提交请求
|
||||||
|
formLoading.value = true
|
||||||
|
try {
|
||||||
|
if (formType.value === 'create') {
|
||||||
|
await KpiApi.createKpi(form.value)
|
||||||
|
message.success(t('common.createSuccess'))
|
||||||
|
} else {
|
||||||
|
await KpiApi.updateKpi(form.value)
|
||||||
|
message.success(t('common.updateSuccess'))
|
||||||
|
}
|
||||||
|
show.value = false
|
||||||
|
// 发送操作成功的事件
|
||||||
|
emit('success')
|
||||||
|
} finally {
|
||||||
|
formLoading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const checkAll = ref(false)
|
||||||
|
const isIndeterminate = ref(false)
|
||||||
|
const employeeOptions = ref([])
|
||||||
|
|
||||||
|
function handleCheckAllChange(val) {
|
||||||
|
form.value.examinedUserIdList = val ? employeeOptions.value.map((it) => it.id) : []
|
||||||
|
isIndeterminate.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleCheckedChange(value) {
|
||||||
|
const checkedCount = value.length
|
||||||
|
checkAll.value = checkedCount === employeeOptions.value.length
|
||||||
|
isIndeterminate.value = checkedCount > 0 && checkedCount < employeeOptions.value.length
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped></style>
|
||||||
156
src/views/Kpi/Appraise/index.vue
Normal file
@@ -0,0 +1,156 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<!-- 搜索工作栏 -->
|
||||||
|
<el-form :model="searchForm" ref="queryFormRef" inline label-width="0">
|
||||||
|
<el-form-item>
|
||||||
|
<el-input v-model="searchForm.examineTarget" placeholder="请输入考核指标" clearable />
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
<el-button @click="handleQuery" v-hasPermi="['kpi:appraise:search']"> 搜索</el-button>
|
||||||
|
<el-button @click="resetQuery" v-hasPermi="['kpi:appraise:reset']"> 重置</el-button>
|
||||||
|
<el-button
|
||||||
|
type="primary"
|
||||||
|
plain
|
||||||
|
@click="openForm('create')"
|
||||||
|
v-hasPermi="['kpi:appraise:add']"
|
||||||
|
>
|
||||||
|
新增
|
||||||
|
</el-button>
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
|
||||||
|
<!-- 列表 -->
|
||||||
|
<el-table v-loading="loading" :data="list" border>
|
||||||
|
<el-table-column prop="examineTarget" label="考核指标" width="180" />
|
||||||
|
<el-table-column prop="typeName" label="分值模式" width="90" :formatter="getKpiModeLabel" />
|
||||||
|
<el-table-column prop="weight" label="权重%" width="90" />
|
||||||
|
<el-table-column label="考核内容">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<div v-dompurify-html="row.examineContent"></div>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="考核规则" prop="examineRule">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<div v-dompurify-html="row.examineRule"></div>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="评分上限" prop="examineScore" width="90" />
|
||||||
|
<el-table-column label="考核人数" width="90">
|
||||||
|
<template #default="{ row }">
|
||||||
|
{{ row.examinedUserIdList?.length || 0 }}
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="生效日期" prop="effectiveDate" width="120" />
|
||||||
|
<el-table-column label="操作" class-name="fixed-width" width="120">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-button
|
||||||
|
link
|
||||||
|
type="primary"
|
||||||
|
@click="openForm('update', row.id)"
|
||||||
|
v-hasPermi="['kpi:appraise:update']"
|
||||||
|
>
|
||||||
|
修改
|
||||||
|
</el-button>
|
||||||
|
<el-button
|
||||||
|
link
|
||||||
|
type="danger"
|
||||||
|
@click="handleDelete(row.id)"
|
||||||
|
v-hasPermi="['kpi:appraise:remove']"
|
||||||
|
>
|
||||||
|
删除
|
||||||
|
</el-button>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
</el-table>
|
||||||
|
<Pagination
|
||||||
|
v-model:limit="searchForm.pageSize"
|
||||||
|
v-model:page="searchForm.pageNo"
|
||||||
|
:total="total"
|
||||||
|
@pagination="getList"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<DialogAppraise ref="formRef" :kpiModeOoptions="kpiModeOoptions" @success="handleQuery" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="KpiContent">
|
||||||
|
import { removeNullField } from '@/utils'
|
||||||
|
import DialogAppraise from './Components/DialogAppraise.vue'
|
||||||
|
import * as KpiApi from '@/api/kpi/index.js'
|
||||||
|
import { getGeneralSysDictData } from '@/api/system/dict/dict.data'
|
||||||
|
|
||||||
|
const { t } = useI18n() // 国际化
|
||||||
|
const message = useMessage() // 消息弹窗
|
||||||
|
const loading = ref(false) // 列表的加载中
|
||||||
|
const list = ref([]) // 列表的数据
|
||||||
|
const searchForm = ref({
|
||||||
|
examineTarget: undefined,
|
||||||
|
pageNo: 1,
|
||||||
|
pageSize: 20
|
||||||
|
})
|
||||||
|
const total = ref(0)
|
||||||
|
|
||||||
|
/** 初始化 **/
|
||||||
|
onMounted(() => {
|
||||||
|
getOptions()
|
||||||
|
handleQuery()
|
||||||
|
})
|
||||||
|
|
||||||
|
const kpiModeOoptions = ref([])
|
||||||
|
function getOptions() {
|
||||||
|
getGeneralSysDictData('examine_type').then((data) => {
|
||||||
|
kpiModeOoptions.value = data
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function getKpiModeLabel(row) {
|
||||||
|
return kpiModeOoptions.value.find((it) => it.value == row.type)?.label
|
||||||
|
}
|
||||||
|
|
||||||
|
const getList = async () => {
|
||||||
|
loading.value = true
|
||||||
|
try {
|
||||||
|
const data = await KpiApi.getKpiPage(removeNullField(searchForm.value))
|
||||||
|
list.value = data.list
|
||||||
|
} finally {
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 搜索按钮操作 */
|
||||||
|
const handleQuery = () => {
|
||||||
|
searchForm.value.pageNo = 1
|
||||||
|
getList()
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 重置按钮操作 */
|
||||||
|
const resetQuery = () => {
|
||||||
|
searchForm.value = {
|
||||||
|
examineTarget: undefined,
|
||||||
|
pageNo: 1,
|
||||||
|
pageSize: 20
|
||||||
|
}
|
||||||
|
handleQuery()
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 添加/修改操作 */
|
||||||
|
const formRef = ref()
|
||||||
|
const openForm = (type, id) => {
|
||||||
|
formRef.value.open(type, id)
|
||||||
|
}
|
||||||
|
|
||||||
|
/** 删除按钮操作 */
|
||||||
|
const handleDelete = async (id) => {
|
||||||
|
try {
|
||||||
|
// 删除的二次确认
|
||||||
|
await message.delConfirm()
|
||||||
|
// 发起删除
|
||||||
|
await KpiApi.deleteKpi(id)
|
||||||
|
message.success(t('common.delSuccess'))
|
||||||
|
// 刷新列表
|
||||||
|
await getList()
|
||||||
|
} catch {}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped></style>
|
||||||
48
src/views/Kpi/Record/DialogScoreDetail.vue
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
<template>
|
||||||
|
<Dialog title="绩效考核详情" v-model="show" width="1200px">
|
||||||
|
<el-table :data="list" border stripe>
|
||||||
|
<el-table-column prop="examineTarget" label="考核指标" width="100" />
|
||||||
|
<el-table-column prop="typeName" label="分值模式" width="90" />
|
||||||
|
<el-table-column prop="weight" label="权重%" width="90" />
|
||||||
|
<el-table-column label="考核内容">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<div v-dompurify-html="row.examineContent"></div>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="考核规则" prop="examineRule">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<div v-dompurify-html="row.examineRule"></div>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="评分上限" prop="examineScore" width="90" />
|
||||||
|
<el-table-column label="考核评分" width="90" prop="score" />
|
||||||
|
<el-table-column label="评分备注" width="90" prop="remark" />
|
||||||
|
<el-table-column label="加权得分" prop="weightSocre" width="90" />
|
||||||
|
</el-table>
|
||||||
|
|
||||||
|
<template #footer>
|
||||||
|
<span>
|
||||||
|
<el-button @click="show = false">关 闭</el-button>
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
</Dialog>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="DialogScoreDetail">
|
||||||
|
import * as KpiApi from '@/api/kpi/score.js'
|
||||||
|
|
||||||
|
const show = ref(false)
|
||||||
|
const list = ref([])
|
||||||
|
|
||||||
|
async function open(id) {
|
||||||
|
show.value = true
|
||||||
|
try {
|
||||||
|
const data = await KpiApi.getScoreDetail({ id })
|
||||||
|
list.value = data.list
|
||||||
|
} finally {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
defineExpose({ open }) // 提供 open 方法,用于打开弹窗
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped></style>
|
||||||
201
src/views/Kpi/Record/index.vue
Normal file
@@ -0,0 +1,201 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<!-- 搜索工作栏 -->
|
||||||
|
<el-form :model="searchForm" inline label-width="0">
|
||||||
|
<el-form-item>
|
||||||
|
<el-date-picker
|
||||||
|
v-model="searchForm.period"
|
||||||
|
type="month"
|
||||||
|
format="YYYY-MM"
|
||||||
|
value-format="YYYY-MM"
|
||||||
|
placeholder="年月"
|
||||||
|
style="width: 100%"
|
||||||
|
@change="handleSearch"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
<el-input
|
||||||
|
v-model.trim="searchForm.dept"
|
||||||
|
placeholder="组织名称"
|
||||||
|
clearable
|
||||||
|
@keyup.enter="handleSearch"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
<el-input
|
||||||
|
v-model.trim="searchForm.examinedUserName"
|
||||||
|
placeholder="员工姓名"
|
||||||
|
clearable
|
||||||
|
@keyup.enter="handleSearch"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
<el-button @click="getList" v-hasPermi="['kpi:score:search']"> 搜索</el-button>
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
<el-table
|
||||||
|
v-loading="loading"
|
||||||
|
:data="tableList"
|
||||||
|
border
|
||||||
|
stripe
|
||||||
|
:default-expand-all="false"
|
||||||
|
row-key="id"
|
||||||
|
:tree-props="{ children: 'examineResultRespVOList', hasChildren: 'hasChildren' }"
|
||||||
|
:span-method="spanMethod"
|
||||||
|
>
|
||||||
|
<el-table-column label="姓名" min-width="90px">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<span v-if="row.examineResultRespVOList && row.examineResultRespVOList.length">
|
||||||
|
<span>{{ row.period }}</span>
|
||||||
|
<span class="ml-20px">考核人数:{{ row.examineNum }}人</span>
|
||||||
|
</span>
|
||||||
|
<span v-else>
|
||||||
|
{{ row.examinedUserName }}
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="period" label="年月" min-width="90px" />
|
||||||
|
<el-table-column prop="dept" label="组织" min-width="90px" />
|
||||||
|
<el-table-column label="考核总分" prop="score" align="center" />
|
||||||
|
<el-table-column label="操作" fixed="right" width="200">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-button
|
||||||
|
type="danger"
|
||||||
|
v-if="!row.examinedUser"
|
||||||
|
:disabled="row.status == 1"
|
||||||
|
style="padding: 0"
|
||||||
|
text
|
||||||
|
v-hasPermi="['home:false-dilegence:sealup']"
|
||||||
|
@click="handleSealup(row)"
|
||||||
|
>
|
||||||
|
<span v-if="row.status == 0">封存</span>
|
||||||
|
<span v-else-if="row.status == 1">已封存</span>
|
||||||
|
</el-button>
|
||||||
|
<el-button
|
||||||
|
type="primary"
|
||||||
|
text
|
||||||
|
v-if="row.examinedUser"
|
||||||
|
style="padding: 0"
|
||||||
|
v-hasPermi="['kpi:record:detail']"
|
||||||
|
@click="handleDetail(row)"
|
||||||
|
>
|
||||||
|
详情
|
||||||
|
</el-button>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
</el-table>
|
||||||
|
|
||||||
|
<Pagination
|
||||||
|
:total="total"
|
||||||
|
v-model:page="searchForm.pageNo"
|
||||||
|
v-model:limit="searchForm.pageSize"
|
||||||
|
@pagination="getList"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<DialogScoreDetail ref="detailDialogRef" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="KpiRecord">
|
||||||
|
import { removeNullField } from '@/utils'
|
||||||
|
import * as KpiApi from '@/api/kpi/score.js'
|
||||||
|
import DialogScoreDetail from './DialogScoreDetail.vue'
|
||||||
|
|
||||||
|
const message = useMessage() // 消息弹窗
|
||||||
|
|
||||||
|
const searchForm = ref({
|
||||||
|
examinedUserName: undefined,
|
||||||
|
period: undefined,
|
||||||
|
dept: undefined,
|
||||||
|
pageNo: 1,
|
||||||
|
pageSize: 20
|
||||||
|
})
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
handleSearch()
|
||||||
|
})
|
||||||
|
|
||||||
|
function handleSearch() {
|
||||||
|
searchForm.value.pageNo = 1
|
||||||
|
getList()
|
||||||
|
}
|
||||||
|
|
||||||
|
const loading = ref(false)
|
||||||
|
const tableList = ref([])
|
||||||
|
const total = ref(0)
|
||||||
|
async function getList() {
|
||||||
|
loading.value = true
|
||||||
|
try {
|
||||||
|
let params = {
|
||||||
|
...searchForm.value
|
||||||
|
}
|
||||||
|
if (params.period) {
|
||||||
|
params = {
|
||||||
|
...params,
|
||||||
|
year: new Date(params.period).getFullYear(),
|
||||||
|
month: new Date(params.period).getMonth() + 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const data = await KpiApi.getScoreRecord(removeNullField(params))
|
||||||
|
if (searchForm.value.examinedUserName) {
|
||||||
|
tableList.value = data.list.reduce((pre, cur) => {
|
||||||
|
return pre.concat(cur.examineResultRespVOList)
|
||||||
|
}, [])
|
||||||
|
} else {
|
||||||
|
tableList.value = data.list.map((it, index) => ({
|
||||||
|
...it,
|
||||||
|
examineResultRespVOList: it.examineResultRespVOList,
|
||||||
|
id: index + 1
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
total.value = data.total
|
||||||
|
} catch (err) {
|
||||||
|
console.log(err)
|
||||||
|
} finally {
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function spanMethod({ row, columnIndex }) {
|
||||||
|
if (row.examineResultRespVOList && row.examineResultRespVOList.length > 0) {
|
||||||
|
if (columnIndex === 0) {
|
||||||
|
return [1, 4]
|
||||||
|
} else if (columnIndex == 4 && row.id) {
|
||||||
|
return [1, 1]
|
||||||
|
} else {
|
||||||
|
return [0, 0]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const detailDialogRef = ref()
|
||||||
|
function handleDetail(row) {
|
||||||
|
detailDialogRef.value.open(row.id)
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleSealup(row) {
|
||||||
|
try {
|
||||||
|
// 二次确认
|
||||||
|
await message.confirm('确认要封存"' + row.period + '"考核记录吗?')
|
||||||
|
// 发起修改状态
|
||||||
|
await KpiApi.saveScoreRecord({
|
||||||
|
period: row.period
|
||||||
|
})
|
||||||
|
message.success('封存成功!')
|
||||||
|
// 刷新列表
|
||||||
|
await getList()
|
||||||
|
} catch (err) {
|
||||||
|
console.log(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
:deep(.el-table__indent) {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
:deep(.el-table__placeholder) {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
213
src/views/Kpi/Score/index.vue
Normal file
@@ -0,0 +1,213 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<!-- 搜索工作栏 -->
|
||||||
|
<el-form :model="searchForm" inline label-width="0">
|
||||||
|
<el-form-item>
|
||||||
|
<el-date-picker
|
||||||
|
v-model="searchForm.period"
|
||||||
|
type="month"
|
||||||
|
format="YYYY-MM"
|
||||||
|
value-format="YYYY-MM"
|
||||||
|
:clearable="false"
|
||||||
|
style="width: 100%"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
<el-button @click="getList" v-hasPermi="['kpi:score:search']"> 搜索</el-button>
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
<el-tabs v-model="employeeId" @tab-click="getList">
|
||||||
|
<el-tab-pane
|
||||||
|
v-for="item in employeeOptions"
|
||||||
|
:key="item.examinedUser"
|
||||||
|
:name="item.examinedUser"
|
||||||
|
>
|
||||||
|
<template #label>
|
||||||
|
<span>{{ item.examinedUserName }}</span>
|
||||||
|
<el-badge :is-dot="item.notExaminedCount > 0" type="danger" />
|
||||||
|
</template>
|
||||||
|
<div class="mb-10px">
|
||||||
|
<el-tag v-if="periodSaved == 1" type="danger">
|
||||||
|
当前周期已封存,不可修改当月绩效考核
|
||||||
|
</el-tag>
|
||||||
|
<el-tag v-else type="warning">
|
||||||
|
当前周期未封存,可选择修改当月绩效考核,两秒内不操作将自动保存考核结果
|
||||||
|
</el-tag>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<el-table
|
||||||
|
v-loading="loading"
|
||||||
|
:data="list"
|
||||||
|
border
|
||||||
|
show-summary
|
||||||
|
:summary-method="getSummaries"
|
||||||
|
>
|
||||||
|
<el-table-column prop="examineTarget" label="考核指标" width="180" />
|
||||||
|
<el-table-column prop="typeName" label="分值模式" width="90" />
|
||||||
|
<el-table-column prop="weight" label="权重%" width="90" />
|
||||||
|
<el-table-column label="考核内容">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<div v-dompurify-html="row.examineContent"></div>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="考核规则" prop="examineRule">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<div v-dompurify-html="row.examineRule"></div>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="评分上限" prop="examineScore" width="90" />
|
||||||
|
<el-table-column label="考核评分" width="120">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<div v-if="periodSaved == 1">{{ row.score }}</div>
|
||||||
|
<el-input-number
|
||||||
|
v-else
|
||||||
|
v-model="row.score"
|
||||||
|
:min="0"
|
||||||
|
:max="row.examineScore"
|
||||||
|
:step="1"
|
||||||
|
:controls="false"
|
||||||
|
size="small"
|
||||||
|
style="width: 100%"
|
||||||
|
@input="handleSave(row)"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="评分备注" width="200">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<div v-if="periodSaved == 1">{{ row.remark }}</div>
|
||||||
|
<el-input
|
||||||
|
v-else
|
||||||
|
v-model="row.remark"
|
||||||
|
type="textarea"
|
||||||
|
:rows="3"
|
||||||
|
size="small"
|
||||||
|
style="width: 100%"
|
||||||
|
@input="handleSave(row)"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="加权得分" prop="weightSocre" width="90" />
|
||||||
|
<el-table-column label="操作" fixed="right" width="100">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-button style="padding: 0" type="danger" text @click="handleDelete(row)">
|
||||||
|
删除
|
||||||
|
</el-button>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
</el-table>
|
||||||
|
</el-tab-pane>
|
||||||
|
</el-tabs>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="KpiScore">
|
||||||
|
import { formatDate } from '@/utils/formatTime'
|
||||||
|
import * as KpiApi from '@/api/kpi/score.js'
|
||||||
|
|
||||||
|
const { t } = useI18n() // 国际化
|
||||||
|
const message = useMessage() // 消息弹窗
|
||||||
|
const loading = ref(false) // 列表的加载中
|
||||||
|
const list = ref([]) // 列表的数据
|
||||||
|
const searchForm = ref({
|
||||||
|
period: undefined
|
||||||
|
})
|
||||||
|
|
||||||
|
const periodSaved = ref(0)
|
||||||
|
|
||||||
|
const employeeOptions = ref([])
|
||||||
|
const employeeId = ref('')
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
// 默认取上个月
|
||||||
|
const today = new Date()
|
||||||
|
searchForm.value.period = formatDate(
|
||||||
|
new Date(today.getFullYear(), today.getMonth() - 1),
|
||||||
|
'YYYY-MM'
|
||||||
|
)
|
||||||
|
|
||||||
|
getList()
|
||||||
|
})
|
||||||
|
|
||||||
|
async function getList() {
|
||||||
|
try {
|
||||||
|
KpiApi.getKpiPeriodStatus({ period: searchForm.value.period }).then((data) => {
|
||||||
|
periodSaved.value = data.status
|
||||||
|
})
|
||||||
|
// 获取当前要考核的人员
|
||||||
|
const arr = await KpiApi.getKpiEmployees(searchForm.value)
|
||||||
|
employeeOptions.value = arr || []
|
||||||
|
if (employeeOptions.value.length > 0) {
|
||||||
|
if (!employeeId.value) {
|
||||||
|
employeeId.value = employeeOptions.value[0].examinedUser
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
message.error('当月无待考核员工!!!')
|
||||||
|
return
|
||||||
|
}
|
||||||
|
list.value =
|
||||||
|
employeeOptions.value.find((it) => it.examinedUser == employeeId.value)
|
||||||
|
?.examineScoreRespVOList || []
|
||||||
|
} finally {
|
||||||
|
loading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let timer = ref(null)
|
||||||
|
const lastId = ref('')
|
||||||
|
function handleSave(row) {
|
||||||
|
try {
|
||||||
|
lastId.value == row.id && clearTimeout(timer.value)
|
||||||
|
timer.value = setTimeout(async () => {
|
||||||
|
lastId.value = row.id
|
||||||
|
// message.success('正在保存数据,请稍后')
|
||||||
|
await KpiApi.saveKpiScore({
|
||||||
|
id: row.id,
|
||||||
|
score: row.score,
|
||||||
|
remark: row.remark
|
||||||
|
})
|
||||||
|
message.success('修改成功!')
|
||||||
|
getList()
|
||||||
|
}, 2000)
|
||||||
|
} catch (err) {
|
||||||
|
console.log(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleDelete(row) {
|
||||||
|
try {
|
||||||
|
// 删除的二次确认
|
||||||
|
await message.delConfirm(
|
||||||
|
`删除后,该员工于${searchForm.value.period}将不再考核此项,是否确认删除?`
|
||||||
|
)
|
||||||
|
// 发起删除
|
||||||
|
await KpiApi.deleteKpiItem({ id: row.id })
|
||||||
|
message.success(t('common.delSuccess'))
|
||||||
|
// 刷新列表
|
||||||
|
await getList()
|
||||||
|
} catch {}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getSummaries({ columns, data }) {
|
||||||
|
let sums = []
|
||||||
|
columns.forEach((column, index) => {
|
||||||
|
if (index == 0) {
|
||||||
|
sums[index] = '考核总分'
|
||||||
|
return
|
||||||
|
}
|
||||||
|
const values = data.map((item) => Number(item[column.property]))
|
||||||
|
if (!values.every((value) => Number.isNaN(value))) {
|
||||||
|
if (column.property == 'weightSocre') {
|
||||||
|
const employee = employeeOptions.value.find((it) => it.examinedUser == employeeId.value)
|
||||||
|
sums[index] = employee.scoreTotal + '分'
|
||||||
|
} else {
|
||||||
|
sums[index] = ''
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
sums[index] = ''
|
||||||
|
}
|
||||||
|
})
|
||||||
|
return sums
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped></style>
|
||||||
@@ -1,54 +1,25 @@
|
|||||||
<template>
|
<template>
|
||||||
<div
|
<div
|
||||||
:class="prefixCls"
|
:class="prefixCls"
|
||||||
class="h-[100%] relative <xl:bg-v-dark <sm:px-10px <xl:px-10px <md:px-10px"
|
class="h-[100%] relative dark:bg-v-dark <sm:px-10px <xl:px-10px <md:px-10px"
|
||||||
>
|
>
|
||||||
<div class="relative h-full flex mx-auto">
|
<div class="relative h-full flex mx-auto">
|
||||||
<div
|
<div :class="`${prefixCls}__left w-[30%] <lg:w-[100%] left-box relative p-30px <xl:p-20px`">
|
||||||
:class="`${prefixCls}__left flex-1 bg-gray-500 bg-opacity-20 relative p-30px <xl:hidden`"
|
|
||||||
>
|
|
||||||
<!-- 左上角的 logo + 系统标题 -->
|
<!-- 左上角的 logo + 系统标题 -->
|
||||||
<div class="flex items-center relative text-white">
|
<div class="flex items-center relative">
|
||||||
<img alt="" class="w-48px h-48px mr-10px" :src="appInfo.instanceIcon" />
|
<h2 class="mb-3 text-2xl font-bold">
|
||||||
<span class="text-20px font-bold">{{ underlineToHump(appInfo.instanceName) }}</span>
|
欢迎使用 {{ underlineToHump(appInfo.instanceName) }}
|
||||||
</div>
|
</h2>
|
||||||
<!-- 左边的背景图 + 欢迎语 -->
|
<!-- <img alt="" class="w-48px h-48px mr-10px" :src="appInfo.instanceIcon" /> -->
|
||||||
<div class="flex justify-center">
|
<!-- <span class="text-20px font-bold">{{ underlineToHump(appInfo.instanceName) }}</span> -->
|
||||||
<TransitionGroup
|
|
||||||
appear
|
|
||||||
enter-active-class="animate__animated animate__bounceInLeft"
|
|
||||||
tag="div"
|
|
||||||
>
|
|
||||||
<img key="1" alt="" class="w-350px" src="@/assets/svgs/login-box-bg.svg" />
|
|
||||||
<!-- <div key="2" class="text-3xl text-white">{{ t('login.welcome') }}</div> -->
|
|
||||||
</TransitionGroup>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="landIn">
|
|
||||||
<transition-group name="text-animation">
|
|
||||||
<span v-for="(char, index) in chars" :key="index">{{ char }}</span>
|
|
||||||
</transition-group>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="flex-1 p-30px <sm:p-10px dark:bg-v-dark relative">
|
|
||||||
<!-- 右上角的主题、语言选择 -->
|
|
||||||
<div class="flex justify-between items-center text-white @2xl:justify-end @xl:justify-end">
|
|
||||||
<div class="flex items-center @2xl:hidden @xl:hidden">
|
|
||||||
<img alt="" class="w-48px h-48px mr-10px" :src="appInfo.instanceIcon" />
|
|
||||||
<span class="text-20px font-bold">{{ underlineToHump(appInfo.instanceName) }}</span>
|
|
||||||
</div>
|
|
||||||
<div class="flex justify-end items-center space-x-10px">
|
|
||||||
<ThemeSwitch />
|
|
||||||
<!-- <LocaleDropdown class="<xl:text-white dark:text-white" /> -->
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
<!-- 右边的登录界面 -->
|
<!-- 右边的登录界面 -->
|
||||||
<Transition appear enter-active-class="animate__animated animate__bounceInRight">
|
<Transition appear enter-active-class="animate__animated animate__bounceInRight">
|
||||||
<div
|
<div
|
||||||
class="h-full flex items-center m-auto w-[100%] @2xl:max-w-500px @xl:max-w-500px @md:max-w-500px @lg:max-w-500px"
|
class="flex items-center m-auto w-[100%] @2xl:max-w-500px @xl:max-w-500px @md:max-w-500px @lg:max-w-500px <xl:mt-20px"
|
||||||
>
|
>
|
||||||
<!-- 账号登录 -->
|
<!-- 账号登录 -->
|
||||||
<LoginForm class="p-20px h-auto m-auto <xl:(rounded-3xl light:bg-white)" />
|
<LoginForm class="p-40px <xl:p-10px h-auto m-auto" />
|
||||||
<!-- 手机登录 -->
|
<!-- 手机登录 -->
|
||||||
<!-- <MobileForm class="p-20px h-auto m-auto <xl:(rounded-3xl light:bg-white)" /> -->
|
<!-- <MobileForm class="p-20px h-auto m-auto <xl:(rounded-3xl light:bg-white)" /> -->
|
||||||
<!-- 二维码登录 -->
|
<!-- 二维码登录 -->
|
||||||
@@ -59,6 +30,39 @@
|
|||||||
<!-- <SSOLoginVue class="p-20px h-auto m-auto <xl:(rounded-3xl light:bg-white)" /> -->
|
<!-- <SSOLoginVue class="p-20px h-auto m-auto <xl:(rounded-3xl light:bg-white)" /> -->
|
||||||
</div>
|
</div>
|
||||||
</Transition>
|
</Transition>
|
||||||
|
<!-- 左边的背景图 + 欢迎语 -->
|
||||||
|
<div class="flex justify-center">
|
||||||
|
<TransitionGroup
|
||||||
|
appear
|
||||||
|
enter-active-class="animate__animated animate__bounceInLeft"
|
||||||
|
tag="div"
|
||||||
|
>
|
||||||
|
<!-- <img key="1" alt="" class="w-350px" src="@/assets/svgs/login-box-bg.svg" /> -->
|
||||||
|
<img key="1" alt="" class="w-[50%] m-auto" src="@/assets/imgs/login2.gif" />
|
||||||
|
</TransitionGroup>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="landIn">
|
||||||
|
<transition-group name="text-animation">
|
||||||
|
<span v-for="(char, index) in chars" :key="index">{{ char }}</span>
|
||||||
|
</transition-group>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="login-box flex-1 p-30px <sm:p-10px dark:bg-v-dark <lg:hidden relative">
|
||||||
|
<!-- 右上角的主题、语言选择 -->
|
||||||
|
<div class="flex justify-between items-center text-white @2xl:justify-end @xl:justify-end">
|
||||||
|
<div class="flex items-center @2xl:hidden @xl:hidden">
|
||||||
|
<!-- <img alt="" class="w-48px h-48px mr-10px" :src="appInfo.instanceIcon" /> -->
|
||||||
|
<!-- <span class="text-20px font-bold text-black">
|
||||||
|
欢迎使用 {{ underlineToHump(appInfo.instanceName) }}
|
||||||
|
</span> -->
|
||||||
|
</div>
|
||||||
|
<div class="flex justify-end items-center space-x-10px">
|
||||||
|
<ThemeSwitch />
|
||||||
|
<!-- <LocaleDropdown class="<xl:text-white dark:text-white" /> -->
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<img src="@/assets/imgs/shisong.jpg" width="80%" alt="" srcset="" />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
@@ -137,7 +141,7 @@ $prefix-cls: #{$namespace}-login;
|
|||||||
z-index: -1;
|
z-index: -1;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
background-image: url('@/assets/svgs/login-bg.svg');
|
// background-image: url('@/assets/svgs/login-bg.svg');
|
||||||
background-position: center;
|
background-position: center;
|
||||||
background-repeat: no-repeat;
|
background-repeat: no-repeat;
|
||||||
content: '';
|
content: '';
|
||||||
@@ -145,6 +149,20 @@ $prefix-cls: #{$namespace}-login;
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// .left-box {
|
||||||
|
// background: #f3f4fb;
|
||||||
|
// background-image: url('@/assets/imgs/login-left-bg.png');
|
||||||
|
// background-repeat: no-repeat;
|
||||||
|
// background-size: 100% 100%;
|
||||||
|
// }
|
||||||
|
|
||||||
|
.login-box {
|
||||||
|
background-image: url('@/assets/imgs/shisong.jpg');
|
||||||
|
background-position: center center;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
background-size: 80% 80%;
|
||||||
|
}
|
||||||
|
|
||||||
.text-animation-enter-active,
|
.text-animation-enter-active,
|
||||||
.text-animation-leave-active {
|
.text-animation-leave-active {
|
||||||
transition: opacity 0.5s;
|
transition: opacity 0.5s;
|
||||||
@@ -160,12 +178,11 @@ $prefix-cls: #{$namespace}-login;
|
|||||||
width: 80%;
|
width: 80%;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-wrap: wrap;
|
flex-wrap: wrap;
|
||||||
font-size: 18px;
|
font-size: 1rem;
|
||||||
line-height: 1.8;
|
line-height: 1.3;
|
||||||
font-family: Lora, serif;
|
font-family: Lora, serif;
|
||||||
white-space: pre;
|
white-space: pre;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
color: #6ee1f5;
|
|
||||||
|
|
||||||
span {
|
span {
|
||||||
animation: landIn 0.8s ease-out both;
|
animation: landIn 0.8s ease-out both;
|
||||||
|
|||||||
@@ -11,10 +11,13 @@
|
|||||||
>
|
>
|
||||||
<el-row style="margin-left: -10px; margin-right: -10px">
|
<el-row style="margin-left: -10px; margin-right: -10px">
|
||||||
<el-col :span="24" style="padding-left: 10px; padding-right: 10px">
|
<el-col :span="24" style="padding-left: 10px; padding-right: 10px">
|
||||||
|
<slot name="title"></slot>
|
||||||
|
</el-col>
|
||||||
|
<!-- <el-col :span="24" style="padding-left: 10px; padding-right: 10px">
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
<LoginFormTitle style="width: 100%" />
|
<LoginFormTitle style="width: 100%" />
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col> -->
|
||||||
<el-col :span="24" style="padding-left: 10px; padding-right: 10px">
|
<el-col :span="24" style="padding-left: 10px; padding-right: 10px">
|
||||||
<el-form-item v-if="loginData.tenantEnable === 'true'" prop="tenantId">
|
<el-form-item v-if="loginData.tenantEnable === 'true'" prop="tenantId">
|
||||||
<el-input
|
<el-input
|
||||||
@@ -107,7 +110,7 @@
|
|||||||
</template>
|
</template>
|
||||||
<script name="LoginForm" setup>
|
<script name="LoginForm" setup>
|
||||||
import { ElLoading } from 'element-plus'
|
import { ElLoading } from 'element-plus'
|
||||||
import LoginFormTitle from './LoginFormTitle.vue'
|
// import LoginFormTitle from './LoginFormTitle.vue'
|
||||||
|
|
||||||
import { useIcon } from '@/hooks/web/useIcon'
|
import { useIcon } from '@/hooks/web/useIcon'
|
||||||
|
|
||||||
|
|||||||
138
src/views/OKR/Management/Components/AllTarget.vue
Normal file
@@ -0,0 +1,138 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<div class="flex items-center justify-between">
|
||||||
|
<el-row>
|
||||||
|
<el-tree-select
|
||||||
|
v-model="searchForm.nodeId"
|
||||||
|
:data="peroidList"
|
||||||
|
:props="defaultProps"
|
||||||
|
:render-after-expand="false"
|
||||||
|
:default-expand-all="false"
|
||||||
|
check-strictly
|
||||||
|
style="width: 400px"
|
||||||
|
@change="nodeChange"
|
||||||
|
/>
|
||||||
|
<el-button class="ml-10px" type="primary" @click="handleAddNode">新建节点</el-button>
|
||||||
|
</el-row>
|
||||||
|
<el-row>
|
||||||
|
<el-button type="info" @click="handleShowOkr(searchForm.nodeId)"> 节点详情 </el-button>
|
||||||
|
<el-button
|
||||||
|
type="warning"
|
||||||
|
v-if="currentUserId == searchForm.creatorId"
|
||||||
|
@click="handleEditOkr(searchForm.nodeId)"
|
||||||
|
>
|
||||||
|
修改当前节点
|
||||||
|
</el-button>
|
||||||
|
<el-button
|
||||||
|
type="success"
|
||||||
|
v-if="searchForm.executor?.includes(currentUserId)"
|
||||||
|
@click="handleUpdateProcess"
|
||||||
|
>更新进度</el-button
|
||||||
|
>
|
||||||
|
</el-row>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<OkrTable ref="okrTableRef" :canEdit="isCurrentLeafNode" />
|
||||||
|
<DialogOkr ref="dialogOkr" @edit="handleEditOkr" />
|
||||||
|
<DialogOkrInfo ref="dialogOkrInfo" @success="handleSearchPeroid" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="AllTarget">
|
||||||
|
import OkrTable from './OkrTable.vue'
|
||||||
|
import DialogOkrInfo from './DialogOkrInfo.vue'
|
||||||
|
import DialogOkr from './DialogOkr.vue'
|
||||||
|
|
||||||
|
import { getAllNodeTree, getAllOkrPage } from '@/api/okr/okr'
|
||||||
|
import { listToTree, findNode } from '@/utils/tree'
|
||||||
|
import { useUserStore } from '@/store/modules/user'
|
||||||
|
|
||||||
|
const message = useMessage()
|
||||||
|
const userStore = useUserStore()
|
||||||
|
const currentUserId = userStore.getUser.id
|
||||||
|
|
||||||
|
const defaultProps = {
|
||||||
|
value: 'nodeId',
|
||||||
|
label: 'nodeName',
|
||||||
|
children: 'children'
|
||||||
|
}
|
||||||
|
|
||||||
|
const okrTableRef = ref(null)
|
||||||
|
const searchForm = ref({
|
||||||
|
nodeId: undefined
|
||||||
|
})
|
||||||
|
|
||||||
|
const peroidList = ref([])
|
||||||
|
|
||||||
|
handleSearchPeroid()
|
||||||
|
|
||||||
|
// 当前是否是叶子节点
|
||||||
|
// 如果不是叶子节点,则表格数据不可修改
|
||||||
|
const isCurrentLeafNode = ref(false)
|
||||||
|
|
||||||
|
function handleSearchPeroid() {
|
||||||
|
getAllNodeTree().then((resp) => {
|
||||||
|
peroidList.value = listToTree(resp.tree, {
|
||||||
|
id: 'nodeId',
|
||||||
|
pid: 'parentId',
|
||||||
|
children: 'children'
|
||||||
|
})
|
||||||
|
nodeChange(resp.nodeId)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function nodeChange(nodeId) {
|
||||||
|
searchForm.value.nodeId = nodeId
|
||||||
|
getOkrList()
|
||||||
|
const currentNode = findNode(peroidList.value, (node) => {
|
||||||
|
return node.nodeId == nodeId
|
||||||
|
})
|
||||||
|
if (!currentNode.children || currentNode.children.length == 0) {
|
||||||
|
isCurrentLeafNode.value = true
|
||||||
|
} else {
|
||||||
|
isCurrentLeafNode.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getOkrList() {
|
||||||
|
getAllOkrPage(searchForm.value).then((resp) => {
|
||||||
|
const list = resp
|
||||||
|
nextTick(() => {
|
||||||
|
okrTableRef.value.prepareData(list)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const dialogOkrInfo = ref(null)
|
||||||
|
function handleAddNode() {
|
||||||
|
dialogOkrInfo.value.open('create', null)
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleEditOkr() {
|
||||||
|
dialogOkr.value.close()
|
||||||
|
dialogOkrInfo.value.open('update', searchForm.value.nodeId)
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleUpdateProcess() {
|
||||||
|
okrTableRef.value.updateProcess(searchForm.value.nodeId).then(() => {
|
||||||
|
message.success('更新成功')
|
||||||
|
getOkrList()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const dialogOkr = ref(null)
|
||||||
|
function handleShowOkr(id) {
|
||||||
|
dialogOkr.value.open({
|
||||||
|
nodeId: id,
|
||||||
|
canEdit: isCurrentLeafNode.value
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
:deep(.el-overlay-dialog) {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
648
src/views/OKR/Management/Components/DialogOkr.vue
Normal file
@@ -0,0 +1,648 @@
|
|||||||
|
<template>
|
||||||
|
<el-dialog
|
||||||
|
v-if="show"
|
||||||
|
v-model="show"
|
||||||
|
width="94vw"
|
||||||
|
class="dialog-okr"
|
||||||
|
:show-close="false"
|
||||||
|
:close-on-click-modal="false"
|
||||||
|
>
|
||||||
|
<template #header>
|
||||||
|
<div class="dialog-okr-header">
|
||||||
|
<div class="flex-1 flex items-center" style="line-height: 50px">
|
||||||
|
<div class="dialog-okr-title">
|
||||||
|
<Icon icon="ep:circle-check-filled" color="#30d1fc" :size="14" />
|
||||||
|
<span>目标</span>
|
||||||
|
</div>
|
||||||
|
<el-divider direction="vertical" />
|
||||||
|
<span class="text-14px ml-0.25">ork落地</span>
|
||||||
|
<div class="ml-20px text-14px">
|
||||||
|
<span>【节点】</span>
|
||||||
|
<span>{{ nodeInfo.allNodeName }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="flex items-center">
|
||||||
|
<el-button v-if="nodeInfo.creatorId == currentUserId" link @click="emit('edit')">
|
||||||
|
<el-tooltip content="编辑" placement="top" effect="dark">
|
||||||
|
<Icon icon="ep:edit" :size="16" />
|
||||||
|
</el-tooltip>
|
||||||
|
</el-button>
|
||||||
|
|
||||||
|
<el-button link @click="show = false">
|
||||||
|
<el-tooltip content="关闭" placement="top" effect="dark">
|
||||||
|
<Icon icon="ep:close" :size="16" />
|
||||||
|
</el-tooltip>
|
||||||
|
</el-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<template #default>
|
||||||
|
<div class="dialog-okr-body">
|
||||||
|
<div class="dialog-okr-content pr-10px">
|
||||||
|
<div class="dialog-okr-content-detail">
|
||||||
|
<div class="detail-basic-title">
|
||||||
|
<div class="basic-title-item">
|
||||||
|
<div class="basic-title-label">创建人</div>
|
||||||
|
<div class="basic-title-value">{{ nodeInfo.creator }}</div>
|
||||||
|
</div>
|
||||||
|
<div class="basic-title-item">
|
||||||
|
<div class="basic-title-label">执行人</div>
|
||||||
|
<div class="basic-title-value">{{ nodeInfo.executorName || '无' }}</div>
|
||||||
|
</div>
|
||||||
|
<div class="basic-title-item">
|
||||||
|
<div class="basic-title-label">目标数</div>
|
||||||
|
<div class="basic-title-value">{{ okrList.length }}</div>
|
||||||
|
</div>
|
||||||
|
<div class="basic-title-item" style="min-width: 200px">
|
||||||
|
<div class="basic-title-label">总体进度</div>
|
||||||
|
<el-progress :percentage="parseInt(nodeInfo.progress) || 0" :stroke-width="8" />
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="flex detail-basic-info">
|
||||||
|
<span>开始日期:{{ nodeInfo.startTime }}</span>
|
||||||
|
<span>截止日期:{{ nodeInfo.endTime }}</span>
|
||||||
|
<span>最新更新时间:{{ nodeInfo.updateTime }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="flex" style="position: relative; flex: 1; height: 100px">
|
||||||
|
<el-tabs v-model="workIndex" style="flex: 1">
|
||||||
|
<el-tab-pane label="目标/关键成果" name="okr">
|
||||||
|
<div class="content-wrap">
|
||||||
|
<OkrTable ref="okrTableRef" :canEdit="canEdit" />
|
||||||
|
</div>
|
||||||
|
</el-tab-pane>
|
||||||
|
</el-tabs>
|
||||||
|
<el-button v-if="canEdit" class="sav-btn" type="primary" @click="handleSaveProcess">
|
||||||
|
保存并更新
|
||||||
|
</el-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="dialog-okr-side pl-10px">
|
||||||
|
<el-tabs v-model="sideIndex" style="flex: 1; height: 100%">
|
||||||
|
<el-tab-pane label="子节点" name="subNode">
|
||||||
|
<div class="overflow-y-auto" style="height: calc(100% - 50px)">
|
||||||
|
<div
|
||||||
|
v-for="item in childNodeList"
|
||||||
|
:key="item.nodeId"
|
||||||
|
class="border-b-1 child-item"
|
||||||
|
style="padding: 10px 5px; cursor: pointer"
|
||||||
|
@click="handleChildItem(item)"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="flex justify-between items-center overflow-hidden text-16px"
|
||||||
|
style="line-height: 30px"
|
||||||
|
>
|
||||||
|
<div class="child-label">【节点】{{ item.nodeName }}</div>
|
||||||
|
<el-progress
|
||||||
|
type="line"
|
||||||
|
:percentage="parseInt(item.progress) || 0"
|
||||||
|
:stroke-width="6"
|
||||||
|
style="width: 120px"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div class="ml-10px flex items-center text-13px" style="line-height: 30px">
|
||||||
|
<span>目标数:</span>
|
||||||
|
<span style="color: #aaa">{{ item.objectiveNum }}</span>
|
||||||
|
<el-divider direction="vertical" style="margin: 0 20px" />
|
||||||
|
<div
|
||||||
|
class="flex-1 overflow-hidden h-30px"
|
||||||
|
style="text-overflow: ellipsis; white-space: nowrap"
|
||||||
|
>
|
||||||
|
<span>执行人:</span>
|
||||||
|
<span style="color: #aaa">{{ item.executorName || '无' }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
class="ml-10px flex items-center text-12px"
|
||||||
|
style="line-height: 20px; color: #aaa"
|
||||||
|
>
|
||||||
|
<span>开始日期:{{ item.startTime }}</span>
|
||||||
|
<el-divider direction="vertical" style="margin: 0 20px" />
|
||||||
|
<span>截止日期:{{ item.endTime }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</el-tab-pane>
|
||||||
|
<el-tab-pane label="评论" name="conclusion">
|
||||||
|
<div class="relative overflow-y-auto" style="height: calc(100% - 50px)">
|
||||||
|
<div class="flex justify-between items-center">
|
||||||
|
<el-select
|
||||||
|
v-if="addNewComment"
|
||||||
|
v-model="form.commentType"
|
||||||
|
filterable
|
||||||
|
size="small"
|
||||||
|
style="width: 120px"
|
||||||
|
@change="getCommentTemplate"
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="it in commentTypeOptions"
|
||||||
|
:label="it.label"
|
||||||
|
:value="it.id"
|
||||||
|
:key="it.id"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
|
||||||
|
<div v-if="addNewComment">
|
||||||
|
<el-button size="small" @click="addNewComment = false"> 取消 </el-button>
|
||||||
|
<el-button type="primary" size="small" @click="handleSaveComment">
|
||||||
|
发布
|
||||||
|
</el-button>
|
||||||
|
</div>
|
||||||
|
<el-button v-else type="primary" size="small" @click="handleInsertComment">
|
||||||
|
新增评论
|
||||||
|
</el-button>
|
||||||
|
</div>
|
||||||
|
<div class="mt-10px" v-if="addNewComment">
|
||||||
|
<Editor
|
||||||
|
v-model:modelValue="form.commentValue"
|
||||||
|
height="300px"
|
||||||
|
:toolbarConfig="toolbarConfig"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<div
|
||||||
|
v-for="(it, index) in commentList"
|
||||||
|
:key="it.commentId"
|
||||||
|
class="border-b-1"
|
||||||
|
style="padding: 10px 5px"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
class="flex items-center justify-between overflow-hidden text-16px"
|
||||||
|
style="line-height: 30px"
|
||||||
|
>
|
||||||
|
<div class="flex items-center">
|
||||||
|
<el-avatar
|
||||||
|
shape="circle"
|
||||||
|
style="
|
||||||
|
background-color: var(--el-color-primary-light-3);
|
||||||
|
width: 30px;
|
||||||
|
height: 30px;
|
||||||
|
"
|
||||||
|
fit="fill"
|
||||||
|
>
|
||||||
|
<span class="text-12px">{{ it.creatorName.slice(-2) }}</span>
|
||||||
|
</el-avatar>
|
||||||
|
<div class="ml-10px text-16px">{{ it.creatorName }}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="ml-10px" v-dompurify-html="it.content"></div>
|
||||||
|
<div
|
||||||
|
class="ml-10px mt-10px flex items-center justify-between text-12px"
|
||||||
|
style="line-height: 20px; color: #aaa"
|
||||||
|
>
|
||||||
|
<div class="flex items-center">
|
||||||
|
<div class="flex items-center mr-50px">
|
||||||
|
<el-button link @click="good(it)">
|
||||||
|
<Icon
|
||||||
|
icon="fa:thumbs-o-up"
|
||||||
|
:size="16"
|
||||||
|
:color="it.currentUserIsLike ? 'var(--el-color-primary)' : '#333'"
|
||||||
|
/>
|
||||||
|
</el-button>
|
||||||
|
<span
|
||||||
|
class="ml-5px"
|
||||||
|
:style="{
|
||||||
|
color: it.currentUserIsLike ? 'var(--el-color-primary)' : '#333'
|
||||||
|
}"
|
||||||
|
>{{ it.likeCount }}</span
|
||||||
|
>
|
||||||
|
</div>
|
||||||
|
<div class="flex items-center mr-50px">
|
||||||
|
<el-button link @click="showComment(index)">
|
||||||
|
<Icon icon="ep:chat-dot-square" :size="16" color="#333" />
|
||||||
|
</el-button>
|
||||||
|
<span class="ml-5px" style="color: #333">{{ it.commentCount }}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="ml-10px text-13px text-gray-400">
|
||||||
|
{{ formatDate(it.createTime, 'YYYY-MM-DD HH:mm') }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- 评论 -->
|
||||||
|
<div
|
||||||
|
v-if="showCommentIndex == index"
|
||||||
|
class="bg-gray-100 pl-10px pr-10px pt-5px pb-5px"
|
||||||
|
style="margin: 10px 10px 0 10px; border-radius: 4px"
|
||||||
|
label="评论"
|
||||||
|
>
|
||||||
|
<div
|
||||||
|
v-for="subComment in it.children.sort((a, b) => a.createTime - b.createTime)"
|
||||||
|
:key="subComment.commentId"
|
||||||
|
class="text-14px"
|
||||||
|
style="line-height: 24px"
|
||||||
|
>
|
||||||
|
<span class="font-bold">{{ subComment.creatorName }}:</span>
|
||||||
|
<span>
|
||||||
|
{{ subComment.content }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
<div class="mt-10px relative">
|
||||||
|
<!-- <el-input
|
||||||
|
v-model="form.commentValue"
|
||||||
|
placeholder="请输入评论"
|
||||||
|
type="textarea"
|
||||||
|
:autosize="{ minRows: 4 }"
|
||||||
|
clearable
|
||||||
|
size="small"
|
||||||
|
style="width: 100%"
|
||||||
|
/> -->
|
||||||
|
<el-mention
|
||||||
|
v-model="form.commentValue"
|
||||||
|
type="textarea"
|
||||||
|
:autosize="{ minRows: 4 }"
|
||||||
|
:options="employeeOptions"
|
||||||
|
style="width: 100%"
|
||||||
|
size="small"
|
||||||
|
whole
|
||||||
|
placeholder="请输入评论"
|
||||||
|
@select="handleMention"
|
||||||
|
>
|
||||||
|
<template #label="{ item }">
|
||||||
|
<div class="flex items-center justify-between h-full">
|
||||||
|
<span class="text-14px text-dark-700">{{ item.name }}</span>
|
||||||
|
<span class="text-12px text-gray-400">{{ item.dept }}</span>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</el-mention>
|
||||||
|
<el-button
|
||||||
|
type="primary"
|
||||||
|
size="small"
|
||||||
|
style="position: absolute; right: 2px; bottom: 2px"
|
||||||
|
@click="handleSendCommnet(index)"
|
||||||
|
>
|
||||||
|
发布
|
||||||
|
</el-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</el-tab-pane>
|
||||||
|
<el-tab-pane label="进度历史" name="history">
|
||||||
|
<div class="overflow-y-auto pl-15px" style="height: calc(100% - 50px)">
|
||||||
|
<el-timeline class="ml-10px">
|
||||||
|
<el-timeline-item
|
||||||
|
v-for="item in nodeRecords"
|
||||||
|
:key="item.recordId"
|
||||||
|
placement="bottom"
|
||||||
|
:timestamp="formatDate(item.createTime, 'YYYY-MM-DD HH:mm:ss')"
|
||||||
|
color="#30d1fc"
|
||||||
|
>
|
||||||
|
<div>{{ item.creator }}</div>
|
||||||
|
<div class="mt-10px text-14px" style="line-height: 24px; color: #666">
|
||||||
|
{{ item.content }}
|
||||||
|
</div>
|
||||||
|
</el-timeline-item>
|
||||||
|
</el-timeline>
|
||||||
|
</div>
|
||||||
|
</el-tab-pane>
|
||||||
|
</el-tabs>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</el-dialog>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="DialogOkr">
|
||||||
|
import { formatDate } from '@/utils/formatTime'
|
||||||
|
import OkrTable from './OkrTable.vue'
|
||||||
|
|
||||||
|
import { getOkrNodeDetail, getOkrNodeHistory } from '@/api/okr/okr'
|
||||||
|
import {
|
||||||
|
getCommentTypeOptions,
|
||||||
|
createComment,
|
||||||
|
getCommentPage,
|
||||||
|
likeComment
|
||||||
|
} from '@/api/okr/comment'
|
||||||
|
import { listToTree } from '@/utils/tree'
|
||||||
|
import { getEmployeeSimpleList } from '@/api/pers/employee'
|
||||||
|
import { useUserStore } from '@/store/modules/user'
|
||||||
|
|
||||||
|
const message = useMessage()
|
||||||
|
const userStore = useUserStore()
|
||||||
|
const currentUserId = userStore.getUser.id
|
||||||
|
const emit = defineEmits(['edit'])
|
||||||
|
|
||||||
|
const show = ref(false)
|
||||||
|
const canEdit = ref(false)
|
||||||
|
|
||||||
|
const toolbarConfig = {
|
||||||
|
toolbarKeys: [
|
||||||
|
'bold', // 加粗
|
||||||
|
'underline', // 下划线
|
||||||
|
'italic', // 斜体
|
||||||
|
'color', // 文字颜色
|
||||||
|
'bgColor', // 背景色
|
||||||
|
'fontSize', // 字号
|
||||||
|
'bulletedList', // 无序列表
|
||||||
|
'numberedList', // 有序列表
|
||||||
|
'insertTable', // 插入表格
|
||||||
|
'insertLink', // 插入链接
|
||||||
|
'undo' // 撤销
|
||||||
|
]
|
||||||
|
}
|
||||||
|
|
||||||
|
const workIndex = ref('okr')
|
||||||
|
const okrList = ref([])
|
||||||
|
const childNodeList = ref([])
|
||||||
|
const sideIndex = ref('subNode')
|
||||||
|
|
||||||
|
const okrTableRef = ref(null)
|
||||||
|
const nodeInfo = ref({})
|
||||||
|
|
||||||
|
const nodeRecords = ref([])
|
||||||
|
const commentTypeOptions = ref([])
|
||||||
|
|
||||||
|
function open(curNode) {
|
||||||
|
canEdit.value = curNode.canEdit
|
||||||
|
nodeInfo.value.nodeId = curNode.nodeId
|
||||||
|
// 获取数据详情
|
||||||
|
searchInfo(curNode)
|
||||||
|
show.value = true
|
||||||
|
}
|
||||||
|
|
||||||
|
const employeeOptions = ref([])
|
||||||
|
|
||||||
|
function searchInfo(curNode) {
|
||||||
|
try {
|
||||||
|
getOkrNodeDetail(curNode.nodeId).then((resp) => {
|
||||||
|
nodeInfo.value = {
|
||||||
|
...resp,
|
||||||
|
executor: resp.executor || []
|
||||||
|
}
|
||||||
|
canEdit.value =
|
||||||
|
canEdit.value && currentUserId == nodeInfo.value.executor.includes(currentUserId)
|
||||||
|
if (resp.objectives) {
|
||||||
|
okrList.value = resp.objectives.map((item) => ({
|
||||||
|
...item,
|
||||||
|
keyResults: item.keyResults || []
|
||||||
|
}))
|
||||||
|
} else {
|
||||||
|
okrList.value = []
|
||||||
|
}
|
||||||
|
childNodeList.value = [...resp.children]
|
||||||
|
nextTick(() => {
|
||||||
|
okrTableRef.value.prepareData(okrList.value)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
getOkrNodeHistory(curNode.nodeId).then((resp) => {
|
||||||
|
nodeRecords.value = resp
|
||||||
|
})
|
||||||
|
getEmployeeSimpleList({ status: 0 }).then((resp) => {
|
||||||
|
employeeOptions.value = resp.map((item) => ({
|
||||||
|
...item,
|
||||||
|
label: item.name,
|
||||||
|
value: item.name
|
||||||
|
}))
|
||||||
|
})
|
||||||
|
getCommentTypeOptions().then((resp) => {
|
||||||
|
commentTypeOptions.value = resp
|
||||||
|
})
|
||||||
|
searchCommentList()
|
||||||
|
} finally {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function close() {
|
||||||
|
show.value = false
|
||||||
|
}
|
||||||
|
|
||||||
|
defineExpose({ open, close })
|
||||||
|
|
||||||
|
function handleMention(item) {
|
||||||
|
form.value.mentionedUserIdList.push(item.id)
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleSaveProcess() {
|
||||||
|
okrTableRef.value.updateProcess(nodeInfo.value.nodeId).then(() => {
|
||||||
|
message.success('更新成功')
|
||||||
|
searchInfo()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleChildItem(item) {
|
||||||
|
const arr = item.executor || []
|
||||||
|
const isExecutor = arr.includes(currentUserId)
|
||||||
|
canEdit.value = isExecutor && (!item.children || item.children.length == 0)
|
||||||
|
searchInfo(item)
|
||||||
|
}
|
||||||
|
|
||||||
|
const addNewComment = ref(false)
|
||||||
|
const form = ref({
|
||||||
|
commentValue: '',
|
||||||
|
commentType: undefined,
|
||||||
|
mentionedUserIdList: []
|
||||||
|
})
|
||||||
|
function handleInsertComment() {
|
||||||
|
addNewComment.value = true
|
||||||
|
const defaultComment = commentTypeOptions.value[0]
|
||||||
|
if (defaultComment) {
|
||||||
|
form.value = {
|
||||||
|
commentType: defaultComment.id,
|
||||||
|
mentionedUserIdList: []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
getCommentTemplate()
|
||||||
|
}
|
||||||
|
|
||||||
|
function getCommentTemplate() {
|
||||||
|
if (form.value.commentType) {
|
||||||
|
form.value.commentValue = commentTypeOptions.value.find(
|
||||||
|
(item) => item.id == form.value.commentType
|
||||||
|
).remark
|
||||||
|
} else {
|
||||||
|
form.value.commentValue = `<h2 style=\"text-align: start;\">一、工作概况</h2><p> </p><h2 style=\"text-align: start;\">二、数据统计</h2><ol><li>呼出电话数:</li><li>有效沟通数:</li><li>销售成果</li></ol><h2 style=\"text-align: start;\">三、问题与改进方案</h2><p><br></p>`
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleSaveComment() {
|
||||||
|
addNewComment.value = false
|
||||||
|
try {
|
||||||
|
const data = {
|
||||||
|
businessType: 1,
|
||||||
|
businessId: nodeInfo.value.nodeId,
|
||||||
|
contentType: 1,
|
||||||
|
content: form.value.commentValue,
|
||||||
|
mentionedUserIdList: form.value.mentionedUserIdList
|
||||||
|
}
|
||||||
|
createComment(data)
|
||||||
|
.then(() => {
|
||||||
|
message.success('评论成功')
|
||||||
|
searchCommentList()
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
form.value.commentValue = ''
|
||||||
|
})
|
||||||
|
} catch (error) {
|
||||||
|
message.error('评论失败')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
const commentList = ref([])
|
||||||
|
|
||||||
|
function searchCommentList() {
|
||||||
|
getCommentPage({
|
||||||
|
businessType: 1,
|
||||||
|
businessId: nodeInfo.value.nodeId,
|
||||||
|
pageNum: 1,
|
||||||
|
pageSize: -1
|
||||||
|
}).then((resp) => {
|
||||||
|
commentList.value = listToTree(resp.list, {
|
||||||
|
id: 'commentId',
|
||||||
|
pid: 'parentId',
|
||||||
|
children: 'children'
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function good(item) {
|
||||||
|
likeComment(item.commentId).then(() => {
|
||||||
|
message.success('点赞成功')
|
||||||
|
searchCommentList()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const showCommentIndex = ref(-1)
|
||||||
|
function showComment(index) {
|
||||||
|
showCommentIndex.value = showCommentIndex.value == index ? -1 : index
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleSendCommnet(idx) {
|
||||||
|
try {
|
||||||
|
// 过滤掉删除的用户,方式为遍历mentionedUserIdList,查找评论中是否有对应的用户名
|
||||||
|
const userList = [...form.value.mentionedUserIdList]
|
||||||
|
const arr = []
|
||||||
|
userList.map((item) => {
|
||||||
|
if (form.value.commentValue.indexOf(`@${item.name}`) != -1) {
|
||||||
|
arr.push(item.id)
|
||||||
|
// 然后移除对应的用户名,防止有多个
|
||||||
|
form.value.commentValue = form.value.commentValue.replace(`@${item.name}`, '')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
const data = {
|
||||||
|
businessType: 1,
|
||||||
|
businessId: nodeInfo.value.nodeId,
|
||||||
|
contentType: 1,
|
||||||
|
content: form.value.commentValue,
|
||||||
|
mentionedUserIdList: arr,
|
||||||
|
parentId: commentList.value[idx].commentId
|
||||||
|
}
|
||||||
|
createComment(data)
|
||||||
|
.then(() => {
|
||||||
|
message.success('评论成功')
|
||||||
|
searchCommentList()
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
form.value.commentValue = ''
|
||||||
|
})
|
||||||
|
} catch (error) {
|
||||||
|
message.error('评论失败')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.dialog-okr {
|
||||||
|
.dialog-okr-header {
|
||||||
|
display: flex;
|
||||||
|
height: 51px;
|
||||||
|
min-height: 51px;
|
||||||
|
vertical-align: middle;
|
||||||
|
justify-content: space-between;
|
||||||
|
-webkit-box-align: center;
|
||||||
|
-ms-flex-align: center;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0 1.25rem;
|
||||||
|
.dialog-okr-title {
|
||||||
|
display: flex;
|
||||||
|
margin-right: 5px;
|
||||||
|
align-items: center;
|
||||||
|
background: rgba(48, 209, 252, 0.1);
|
||||||
|
border-radius: 3px;
|
||||||
|
padding: 0 10px;
|
||||||
|
line-height: 24px;
|
||||||
|
span {
|
||||||
|
color: #2a344b;
|
||||||
|
font-size: 12px;
|
||||||
|
margin-left: 6px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.dialog-okr-body {
|
||||||
|
display: flex;
|
||||||
|
background: #f0f3fa;
|
||||||
|
overflow: hidden;
|
||||||
|
min-width: 1064px;
|
||||||
|
height: calc(94vh - 85px);
|
||||||
|
.dialog-okr-content {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
height: 100%;
|
||||||
|
width: 70%;
|
||||||
|
flex: auto;
|
||||||
|
min-width: 700px;
|
||||||
|
background: #fff;
|
||||||
|
margin: 0 1px 0 0;
|
||||||
|
justify-content: space-between;
|
||||||
|
overflow-y: hidden !important;
|
||||||
|
.dialog-okr-content-detail {
|
||||||
|
position: relative;
|
||||||
|
margin-bottom: 30px;
|
||||||
|
overflow-y: auto;
|
||||||
|
display: -webkit-box;
|
||||||
|
display: -ms-flexbox;
|
||||||
|
display: flex;
|
||||||
|
-webkit-box-orient: vertical;
|
||||||
|
-webkit-box-direction: normal;
|
||||||
|
-ms-flex-direction: column;
|
||||||
|
flex-direction: column;
|
||||||
|
.detail-basic-title {
|
||||||
|
display: flex;
|
||||||
|
font-size: 14px;
|
||||||
|
margin-bottom: 16px;
|
||||||
|
.basic-title-item {
|
||||||
|
margin-right: 50px;
|
||||||
|
.basic-title-label {
|
||||||
|
color: #aaa;
|
||||||
|
margin-bottom: 6px;
|
||||||
|
}
|
||||||
|
.basic-title-value {
|
||||||
|
color: #2a344b;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.detail-basic-info {
|
||||||
|
font-size: 14px;
|
||||||
|
color: #aaa;
|
||||||
|
span {
|
||||||
|
margin-right: 40px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.content-wrap {
|
||||||
|
overflow-y: auto;
|
||||||
|
max-height: calc(100% - 70px);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.dialog-okr-side {
|
||||||
|
padding-left: 10px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
width: 30%;
|
||||||
|
overflow: hidden;
|
||||||
|
min-width: 364px;
|
||||||
|
max-width: 500px;
|
||||||
|
background: #fff;
|
||||||
|
margin: 0;
|
||||||
|
.child-item:hover {
|
||||||
|
background: #f0f3fa;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.sav-btn {
|
||||||
|
position: absolute;
|
||||||
|
right: 10px;
|
||||||
|
top: 0;
|
||||||
|
width: auto;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
587
src/views/OKR/Management/Components/DialogOkrInfo.vue
Normal file
@@ -0,0 +1,587 @@
|
|||||||
|
<template>
|
||||||
|
<el-dialog
|
||||||
|
v-model="show"
|
||||||
|
width="900px"
|
||||||
|
class="dialog-okr"
|
||||||
|
:show-close="false"
|
||||||
|
:close-on-click-modal="false"
|
||||||
|
>
|
||||||
|
<template #header>
|
||||||
|
<div class="dialog-okr-header">
|
||||||
|
<div class="flex-1 flex items-center" style="line-height: 50px"> {{ title }} </div>
|
||||||
|
<div class="flex items-center">
|
||||||
|
<el-button type="primary" link :disabled="formLoading" @click="handleSave">
|
||||||
|
保存
|
||||||
|
</el-button>
|
||||||
|
<el-button link @click="show = false"> 关闭 </el-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<template #default>
|
||||||
|
<div class="dialog-okr-body" v-loading="formLoading">
|
||||||
|
<el-form :model="form" ref="formRef" :rules="rules" label-width="0">
|
||||||
|
<el-row :gutter="10">
|
||||||
|
<el-col :span="4" :offset="0">
|
||||||
|
<el-form-item prop="parentId">
|
||||||
|
<el-tree-select
|
||||||
|
v-model="form.parentId"
|
||||||
|
:data="parentNodeList"
|
||||||
|
check-strictly
|
||||||
|
:props="{ value: 'nodeId', label: 'nodeName', children: 'children' }"
|
||||||
|
:render-after-expand="false"
|
||||||
|
placeholder="父节点"
|
||||||
|
clearable
|
||||||
|
:disabled="!!form.nodeId"
|
||||||
|
@change="handleChangeParent"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="4" :offset="0">
|
||||||
|
<el-form-item prop="nodeName">
|
||||||
|
<el-input v-model="form.nodeName" placeholder="请输入节点名称" />
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="4" :offset="0">
|
||||||
|
<el-form-item prop="startTime">
|
||||||
|
<el-date-picker
|
||||||
|
v-model="form.startTime"
|
||||||
|
type="date"
|
||||||
|
format="YYYY-MM-DD"
|
||||||
|
value-format="YYYY-MM-DD"
|
||||||
|
placeholder="开始日期"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="4" :offset="0">
|
||||||
|
<el-form-item prop="endTime">
|
||||||
|
<el-date-picker
|
||||||
|
v-model="form.endTime"
|
||||||
|
type="date"
|
||||||
|
format="YYYY-MM-DD"
|
||||||
|
value-format="YYYY-MM-DD"
|
||||||
|
placeholder="截止日期"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="4" :offset="0">
|
||||||
|
<el-form-item prop="executor">
|
||||||
|
<el-select
|
||||||
|
v-model="form.executor"
|
||||||
|
placeholder="选择执行人,可多选"
|
||||||
|
multiple
|
||||||
|
collapse-tags
|
||||||
|
collapse-tags-tooltip
|
||||||
|
clearable
|
||||||
|
filterable
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="item in userOptions"
|
||||||
|
:key="item.id"
|
||||||
|
:label="item.name"
|
||||||
|
:value="item.id + ''"
|
||||||
|
:disabled="item.status == 1"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="4" :offset="0">
|
||||||
|
<el-form-item prop="dataScope">
|
||||||
|
<el-select v-model="form.dataScope" placeholder="选择权限" clearable filterable>
|
||||||
|
<el-option label="公开" :value="1" />
|
||||||
|
<el-option label="仅上级可见" :value="2" />
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
</el-form>
|
||||||
|
<div class="relative">
|
||||||
|
<el-button
|
||||||
|
v-if="okrKey == 'okr'"
|
||||||
|
style="position: absolute; right: 10px; top: 10px; z-index: 9"
|
||||||
|
type="primary"
|
||||||
|
size="small"
|
||||||
|
@click="addObjective"
|
||||||
|
>
|
||||||
|
添加目标
|
||||||
|
</el-button>
|
||||||
|
<el-button
|
||||||
|
v-else-if="okrKey == 'childNode'"
|
||||||
|
style="position: absolute; right: 10px; top: 10px; z-index: 9"
|
||||||
|
type="primary"
|
||||||
|
size="small"
|
||||||
|
@click="addChildNode"
|
||||||
|
>
|
||||||
|
添加子节点
|
||||||
|
</el-button>
|
||||||
|
<el-tabs v-model="okrKey">
|
||||||
|
<el-tab-pane label="目标/关键成果" name="okr">
|
||||||
|
<div>
|
||||||
|
<div
|
||||||
|
class="flex items-center w-full"
|
||||||
|
v-for="(item, i) in objectList"
|
||||||
|
:key="item.objectiveId"
|
||||||
|
>
|
||||||
|
<div class="flex-1 w-100px">
|
||||||
|
<div class="flex items-center">
|
||||||
|
<el-tag type="success" class="mr-10px">O{{ i + 1 }}</el-tag>
|
||||||
|
<el-input
|
||||||
|
v-model="item.objectiveName"
|
||||||
|
placeholder="目标名称"
|
||||||
|
class="flex-1 w-150px"
|
||||||
|
:disabled="!!item.objectiveId"
|
||||||
|
/>
|
||||||
|
<el-select
|
||||||
|
v-model="item.executor"
|
||||||
|
placeholder="选择执行人,可多选"
|
||||||
|
multiple
|
||||||
|
clearable
|
||||||
|
filterable
|
||||||
|
style="width: 240px; margin-left: 10px"
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="it in userOptions"
|
||||||
|
:key="it.id"
|
||||||
|
:label="it.name"
|
||||||
|
:value="it.id + ''"
|
||||||
|
:disabled="it.status == 1"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
<el-select
|
||||||
|
v-model="item.dataScope"
|
||||||
|
placeholder="选择权限"
|
||||||
|
clearable
|
||||||
|
filterable
|
||||||
|
style="width: 120px; margin-left: 10px"
|
||||||
|
>
|
||||||
|
<el-option label="公开" :value="1" />
|
||||||
|
<el-option label="仅上级可见" :value="2" />
|
||||||
|
</el-select>
|
||||||
|
<div class="ml-10px">
|
||||||
|
<el-button type="primary" text @click="AddKR(i)">新增关键成果</el-button>
|
||||||
|
<el-button type="danger" text @click="removeObj(i)">删除目标</el-button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div style="background: #f0f3fa; padding-bottom: 15px">
|
||||||
|
<div
|
||||||
|
class="ml-50px"
|
||||||
|
v-for="(kr, index) in item.keyResults"
|
||||||
|
:key="kr.keyResultId"
|
||||||
|
>
|
||||||
|
<div class="flex flex-1 items-center pb-5px pt-5px">
|
||||||
|
<el-tag class="mr-10px">KR{{ index + 1 }}</el-tag>
|
||||||
|
<el-select
|
||||||
|
v-model="kr.keyResultName"
|
||||||
|
placeholder="选择或输入关键成果"
|
||||||
|
filterable
|
||||||
|
:props="{ value: 'keyResultId', label: 'keyResultName' }"
|
||||||
|
allow-create
|
||||||
|
:reserve-keyword="false"
|
||||||
|
class="flex-1 w-150px"
|
||||||
|
@change="krChange(i, index)"
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="it in krOptions"
|
||||||
|
:key="it.value"
|
||||||
|
:label="it.label"
|
||||||
|
:value="it.value"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
<el-button type="danger" text @click="removeKR(i, index)">
|
||||||
|
删除关键成果
|
||||||
|
</el-button>
|
||||||
|
</div>
|
||||||
|
<div class="mt-5px flex items-center">
|
||||||
|
<el-tree-select
|
||||||
|
v-model="kr.sourceId"
|
||||||
|
:data="sourceOptions"
|
||||||
|
:props="defaultProps"
|
||||||
|
check-strictly
|
||||||
|
clearable
|
||||||
|
filterable
|
||||||
|
style="width: 200px"
|
||||||
|
node-key="sourceId"
|
||||||
|
placeholder="请选择渠道"
|
||||||
|
/>
|
||||||
|
<el-radio-group v-model="kr.resultType" class="ml-10px">
|
||||||
|
<el-radio-button :value="1">目标值</el-radio-button>
|
||||||
|
<el-radio-button :value="2">是/否</el-radio-button>
|
||||||
|
</el-radio-group>
|
||||||
|
|
||||||
|
<el-input
|
||||||
|
v-if="kr.resultType == 1"
|
||||||
|
v-model="kr.targetValue"
|
||||||
|
type="number"
|
||||||
|
placeholder="目标值"
|
||||||
|
class="ml-10px"
|
||||||
|
style="width: 120px"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<el-select
|
||||||
|
v-model="kr.executor"
|
||||||
|
placeholder="选择执行人,可多选"
|
||||||
|
multiple
|
||||||
|
clearable
|
||||||
|
filterable
|
||||||
|
collapse-tags
|
||||||
|
collapse-tags-tooltip
|
||||||
|
style="width: 240px; margin-left: 10px"
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="it in userOptions"
|
||||||
|
:key="it.id"
|
||||||
|
:label="it.name"
|
||||||
|
:value="it.id + ''"
|
||||||
|
:disabled="it.status == 1"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
<el-select
|
||||||
|
v-model="kr.dataScope"
|
||||||
|
placeholder="选择权限"
|
||||||
|
clearable
|
||||||
|
filterable
|
||||||
|
style="width: 120px; margin-left: 10px"
|
||||||
|
>
|
||||||
|
<el-option label="公开" :value="1" />
|
||||||
|
<el-option label="仅上级可见" :value="2" />
|
||||||
|
</el-select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- <div class="w-100px"></div> -->
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</el-tab-pane>
|
||||||
|
<el-tab-pane label="子节点" name="childNode">
|
||||||
|
<el-table :data="childNodeList">
|
||||||
|
<el-table-column label="名称">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-input v-model="row.nodeName" placeholder="子节点名称" />
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="开始日期" width="160px">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-date-picker
|
||||||
|
v-model="row.startTime"
|
||||||
|
type="date"
|
||||||
|
format="YYYY-MM-DD"
|
||||||
|
value-format="YYYY-MM-DD"
|
||||||
|
placeholder="选择日期时间"
|
||||||
|
style="width: 130px"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="截止日期" width="160px">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-date-picker
|
||||||
|
v-model="row.endTime"
|
||||||
|
type="date"
|
||||||
|
format="YYYY-MM-DD"
|
||||||
|
value-format="YYYY-MM-DD"
|
||||||
|
placeholder="选择日期时间"
|
||||||
|
style="width: 130px"
|
||||||
|
/>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="权限" width="160px">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-select v-model="row.dataScope" placeholder="请选择" style="width: 130px">
|
||||||
|
<el-option label="公开" :value="1" />
|
||||||
|
<el-option label="仅上级可见" :value="2" />
|
||||||
|
</el-select>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="执行人" width="300px">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-select
|
||||||
|
v-model="row.executor"
|
||||||
|
placeholder="选择执行人,可多选"
|
||||||
|
multiple
|
||||||
|
clearable
|
||||||
|
filterable
|
||||||
|
style="width: 100%"
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="item in userOptions"
|
||||||
|
:key="item.id"
|
||||||
|
:label="item.name"
|
||||||
|
:value="item.id + ''"
|
||||||
|
:disabled="item.status == 1"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column fixed="right" label="删除" :width="100">
|
||||||
|
<template #default="{ $index }">
|
||||||
|
<el-button type="danger" text @click="removeChildNode($index)">删除</el-button>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
</el-table>
|
||||||
|
</el-tab-pane>
|
||||||
|
</el-tabs>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</el-dialog>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="DialogOkrInfo">
|
||||||
|
import {
|
||||||
|
getAllNodeTree,
|
||||||
|
getOkrNodeDetail,
|
||||||
|
createOkrNode,
|
||||||
|
updateOkrNode,
|
||||||
|
getAllOkrPage,
|
||||||
|
getDefaultOkrOptions
|
||||||
|
} from '@/api/okr/okr'
|
||||||
|
import { listToTree } from '@/utils/tree'
|
||||||
|
import { getEmployeeSimpleList } from '@/api/pers/employee'
|
||||||
|
|
||||||
|
const { t } = useI18n() // 国际化
|
||||||
|
const message = useMessage() // 消息弹窗
|
||||||
|
|
||||||
|
const show = ref(false)
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => show.value,
|
||||||
|
(newValue, oldValue) => {
|
||||||
|
if (oldValue && !newValue) {
|
||||||
|
emit('close')
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
const title = ref('')
|
||||||
|
const formType = ref('create')
|
||||||
|
|
||||||
|
const form = ref({})
|
||||||
|
const formLoading = ref(false)
|
||||||
|
const rules = ref({
|
||||||
|
nodeName: { required: true, message: '节点名称不可为空', trigger: 'blur' },
|
||||||
|
startTime: { required: true, message: '开始日期不可为空', trigger: 'change' },
|
||||||
|
endTime: { required: true, message: '截止日期不可为空', trigger: 'change' }
|
||||||
|
})
|
||||||
|
|
||||||
|
const okrKey = ref('okr')
|
||||||
|
|
||||||
|
const parentNodeList = ref([])
|
||||||
|
|
||||||
|
const childNodeList = ref([])
|
||||||
|
|
||||||
|
const userOptions = ref([])
|
||||||
|
const krOptions = ref([])
|
||||||
|
const defaultProps = {
|
||||||
|
children: 'children',
|
||||||
|
label: 'sourceName',
|
||||||
|
value: 'sourceId',
|
||||||
|
isLeaf: 'leaf'
|
||||||
|
}
|
||||||
|
const sourceOptions = ref([])
|
||||||
|
const objectList = ref([])
|
||||||
|
function open(type, val) {
|
||||||
|
show.value = true
|
||||||
|
title.value = type == 'update' ? '修改Okr' : '新增Okr'
|
||||||
|
formType.value = type
|
||||||
|
resetForm()
|
||||||
|
getEmployeeSimpleList().then((data) => {
|
||||||
|
userOptions.value = data
|
||||||
|
})
|
||||||
|
getAllNodeTree().then((resp) => {
|
||||||
|
parentNodeList.value = listToTree(resp.tree, {
|
||||||
|
id: 'nodeId',
|
||||||
|
pid: 'parentId',
|
||||||
|
children: 'children'
|
||||||
|
})
|
||||||
|
})
|
||||||
|
getDefaultOkrOptions().then((resp) => {
|
||||||
|
krOptions.value = resp
|
||||||
|
})
|
||||||
|
if (val) {
|
||||||
|
formLoading.value = true
|
||||||
|
try {
|
||||||
|
getOkrNodeDetail(val).then((resp) => {
|
||||||
|
form.value = resp
|
||||||
|
if (resp.objectives) {
|
||||||
|
objectList.value = resp.objectives.map((item) => ({
|
||||||
|
...item,
|
||||||
|
keyResults: item.keyResults || []
|
||||||
|
}))
|
||||||
|
} else {
|
||||||
|
objectList.value = []
|
||||||
|
}
|
||||||
|
childNodeList.value = [...resp.children]
|
||||||
|
})
|
||||||
|
} finally {
|
||||||
|
formLoading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function resetForm() {
|
||||||
|
form.value = {
|
||||||
|
parentId: undefined,
|
||||||
|
nodeName: undefined,
|
||||||
|
startTime: undefined,
|
||||||
|
endTime: undefined,
|
||||||
|
executor: [],
|
||||||
|
dataScope: 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
defineExpose({ open }) // 提供 open 方法,用于打开弹窗
|
||||||
|
|
||||||
|
const formRef = ref()
|
||||||
|
/** 提交表单 */
|
||||||
|
const emit = defineEmits(['success', 'close']) // 定义 success 事件,用于操作成功后的回调
|
||||||
|
|
||||||
|
function addObjective() {
|
||||||
|
objectList.value.push({
|
||||||
|
objectiveName: '',
|
||||||
|
executor: [],
|
||||||
|
keyResults: [],
|
||||||
|
dataScope: 1
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function AddKR(idx) {
|
||||||
|
if (!objectList.value[idx].keyResults) {
|
||||||
|
objectList.value[idx].keyResults = []
|
||||||
|
}
|
||||||
|
const obj = objectList.value[idx]
|
||||||
|
objectList.value[idx].keyResults.push({
|
||||||
|
keyResultName: undefined,
|
||||||
|
resultType: 1,
|
||||||
|
source: undefined,
|
||||||
|
targetValue: undefined,
|
||||||
|
process: undefined,
|
||||||
|
currentValue: undefined,
|
||||||
|
executor: obj.executor,
|
||||||
|
dataScope: obj.dataScope
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeObj(idx) {
|
||||||
|
objectList.value.splice(idx, 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeKR(oIdx, krIdx) {
|
||||||
|
objectList.value[oIdx].keyResults.splice(krIdx, 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
function addChildNode() {
|
||||||
|
childNodeList.value.push({})
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeChildNode(idx) {
|
||||||
|
childNodeList.value.splice(idx, 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 读取父节点的OKR,并展示出来
|
||||||
|
function handleChangeParent() {
|
||||||
|
// 通过父节点Id,查询父节点的okr,需要继承
|
||||||
|
if (form.value.parentId) {
|
||||||
|
getAllOkrPage({ nodeId: form.value.parentId }).then((resp) => {
|
||||||
|
objectList.value = resp
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function getLastDayOfMonth(year, month) {
|
||||||
|
// 创建一个日期对象,将月份设置为指定月份的下一个月,日期设置为 0
|
||||||
|
const date = new Date(year, month + 1, 0)
|
||||||
|
// 返回该日期对象的日期部分,即指定年月的最后一天
|
||||||
|
return date.getDate()
|
||||||
|
}
|
||||||
|
|
||||||
|
function krChange(idx, krIdx) {
|
||||||
|
const kr = objectList.value[idx].keyResults[krIdx]
|
||||||
|
if (kr.keyResultName) {
|
||||||
|
const obj = krOptions.value.find((item) => item.value == kr.keyResultName)
|
||||||
|
if (obj) {
|
||||||
|
kr.isSys = true
|
||||||
|
} else {
|
||||||
|
kr.isSys = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async function handleSave() {
|
||||||
|
// 校验表单
|
||||||
|
if (!formRef.value) return
|
||||||
|
const valid = await formRef.value.validate()
|
||||||
|
if (!valid) return
|
||||||
|
// 提交请求
|
||||||
|
formLoading.value = true
|
||||||
|
try {
|
||||||
|
form.value.objectives = objectList.value
|
||||||
|
form.value.children = childNodeList.value
|
||||||
|
if (formType.value === 'create') {
|
||||||
|
// 创建的是一级节点,且子节点为空时,自动新增12个月子节点,并且每月添加4周的周子节点
|
||||||
|
if (!form.value.parentId) {
|
||||||
|
if (form.value.children.length == 0) {
|
||||||
|
const defaultTime = new Date(form.value.startTime)
|
||||||
|
for (let month = 0; month < 12; month++) {
|
||||||
|
form.value.children.push({
|
||||||
|
nodeName: `${month + 1}月`,
|
||||||
|
startTime: `${defaultTime.getFullYear()}-${String(month + 1).padStart(2, '0')}-01`,
|
||||||
|
endTime: `${defaultTime.getFullYear()}-${String(month + 1).padStart(
|
||||||
|
2,
|
||||||
|
'0'
|
||||||
|
)}-${getLastDayOfMonth(defaultTime.getFullYear(), month)}`,
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
nodeName: `${month + 1}月第1周`,
|
||||||
|
children: []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
nodeName: `${month + 1}月第2周`,
|
||||||
|
children: []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
nodeName: `${month + 1}月第3周`,
|
||||||
|
children: []
|
||||||
|
},
|
||||||
|
{
|
||||||
|
nodeName: `${month + 1}月第4周`,
|
||||||
|
children: []
|
||||||
|
}
|
||||||
|
]
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
await createOkrNode(form.value)
|
||||||
|
message.success(t('common.createSuccess'))
|
||||||
|
} else {
|
||||||
|
await updateOkrNode(form.value)
|
||||||
|
message.success(t('common.updateSuccess'))
|
||||||
|
}
|
||||||
|
show.value = false
|
||||||
|
// 发送操作成功的事件
|
||||||
|
emit('success')
|
||||||
|
} finally {
|
||||||
|
formLoading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.dialog-okr {
|
||||||
|
.dialog-okr-header {
|
||||||
|
display: flex;
|
||||||
|
height: 51px;
|
||||||
|
min-height: 51px;
|
||||||
|
vertical-align: middle;
|
||||||
|
justify-content: space-between;
|
||||||
|
-webkit-box-align: center;
|
||||||
|
-ms-flex-align: center;
|
||||||
|
align-items: center;
|
||||||
|
padding: 0 1.25rem;
|
||||||
|
}
|
||||||
|
.dialog-okr-body {
|
||||||
|
overflow-x: hidden;
|
||||||
|
overflow-y: auto;
|
||||||
|
min-width: 1064px;
|
||||||
|
height: calc(94vh - 85px);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
347
src/views/OKR/Management/Components/DialogWait.vue
Normal file
@@ -0,0 +1,347 @@
|
|||||||
|
<template>
|
||||||
|
<Dialog :title="title" v-model="show" :width="formType == 'create' ? '750px' : '1100px'">
|
||||||
|
<div class="flex items-baseline">
|
||||||
|
<el-form
|
||||||
|
:model="form"
|
||||||
|
ref="formRef"
|
||||||
|
:disabled="formType == 'do'"
|
||||||
|
:rules="rules"
|
||||||
|
label-width="80px"
|
||||||
|
class="flex-1"
|
||||||
|
>
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="24" :offset="0">
|
||||||
|
<el-form-item label="标题" prop="title">
|
||||||
|
<el-input v-model="form.title" placeholder="请输入待办标题" />
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="24">
|
||||||
|
<el-form-item label="内容" prop="content">
|
||||||
|
<Editor
|
||||||
|
v-model:modelValue="form.content"
|
||||||
|
height="400px"
|
||||||
|
:disabled="formType == 'do'"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="12" :offset="0">
|
||||||
|
<el-form-item label="优先级" prop="priority">
|
||||||
|
<el-select v-model="form.priority" placeholder="选择优先级" filterable>
|
||||||
|
<el-option
|
||||||
|
v-for="item in props.priorityOptions"
|
||||||
|
:key="item.value"
|
||||||
|
:label="item.label"
|
||||||
|
:value="item.value"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="12" :offset="0">
|
||||||
|
<el-form-item label="完成状态" prop="completeStatus">
|
||||||
|
<el-select v-model="form.completeStatus" placeholder="选择完成状态" filterable>
|
||||||
|
<el-option
|
||||||
|
v-for="item in props.statusOptions"
|
||||||
|
:key="item.value"
|
||||||
|
:label="item.label"
|
||||||
|
:value="item.value"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<el-row :gutter="20">
|
||||||
|
<el-col :span="12" :offset="0">
|
||||||
|
<el-form-item label="截止日期" prop="endDate">
|
||||||
|
<el-date-picker
|
||||||
|
v-model="form.endDate"
|
||||||
|
type="date"
|
||||||
|
format="YYYY-MM-DD"
|
||||||
|
value-format="YYYY-MM-DD"
|
||||||
|
placeholder="选择日期时间"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="12" :offset="0">
|
||||||
|
<el-form-item label="执行人" prop="userList">
|
||||||
|
<el-select
|
||||||
|
v-model="form.userList"
|
||||||
|
placeholder="选择执行人,可多选"
|
||||||
|
multiple
|
||||||
|
clearable
|
||||||
|
filterable
|
||||||
|
style="width: 100%"
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="item in employeeOptions"
|
||||||
|
:key="item.id"
|
||||||
|
:disabled="item.status == 1"
|
||||||
|
:label="item.name"
|
||||||
|
:value="item.id"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<el-row :gutter="0">
|
||||||
|
<el-col :span="8" :offset="0">
|
||||||
|
<el-form-item prop="notifyType">
|
||||||
|
<template #label>
|
||||||
|
<el-tooltip
|
||||||
|
content="新增待办时会立刻发送消息通知执行人,若后续无需再次提醒,频率可选择“不提醒”"
|
||||||
|
placement="top"
|
||||||
|
effect="dark"
|
||||||
|
>
|
||||||
|
<Icon icon="ep:info-filled" :size="12" />
|
||||||
|
</el-tooltip>
|
||||||
|
<span>提醒频率</span>
|
||||||
|
</template>
|
||||||
|
<el-select v-model="form.notifyType" placeholder="选择提醒频率">
|
||||||
|
<el-option label="不提醒" value="1" />
|
||||||
|
<el-option label="单次提醒" value="2" />
|
||||||
|
<el-option label="循环提醒" value="3" />
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="8" :offset="0" v-if="form.notifyType != 1">
|
||||||
|
<el-form-item label="提醒时间" prop="remindTime" class="wait-form">
|
||||||
|
<el-input
|
||||||
|
v-if="form.notifyType == 2"
|
||||||
|
v-model="form.notifyBeforeTime"
|
||||||
|
placeholder="输入小时数"
|
||||||
|
min="0"
|
||||||
|
type="number"
|
||||||
|
>
|
||||||
|
<template #prepend>
|
||||||
|
<span>提前</span>
|
||||||
|
</template>
|
||||||
|
<template #append> 小时 </template>
|
||||||
|
</el-input>
|
||||||
|
<el-input
|
||||||
|
v-else-if="form.notifyType == 3"
|
||||||
|
v-model="form.intervalTime"
|
||||||
|
placeholder="输入小时数"
|
||||||
|
min="0"
|
||||||
|
type="number"
|
||||||
|
>
|
||||||
|
<template #prepend>
|
||||||
|
<span>每</span>
|
||||||
|
</template>
|
||||||
|
<template #append> 小时 </template>
|
||||||
|
</el-input>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
<el-col :span="8" :offset="0" v-if="form.notifyType == 3">
|
||||||
|
<el-form-item label="首次提醒" prop="firstNotifyTime">
|
||||||
|
<el-time-picker
|
||||||
|
v-model="form.firstNotifyTime"
|
||||||
|
format="HH:mm"
|
||||||
|
value-format="HH:mm"
|
||||||
|
style="width: 100%"
|
||||||
|
:picker-options="enableTimeRange"
|
||||||
|
placeholder="选择时间点"
|
||||||
|
/>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
</el-form>
|
||||||
|
<div v-if="formType != 'create'" class="w-350px ml-10px h-full">
|
||||||
|
<div class="mb-10px" v-if="formType == 'do'">
|
||||||
|
<Editor
|
||||||
|
v-model:modelValue="form.followUpContent"
|
||||||
|
height="150px"
|
||||||
|
:toolbarConfig="toolbarConfig"
|
||||||
|
/>
|
||||||
|
<el-button class="absolute" style="right: 20px; top: 195px" type="primary" @click="addDo">
|
||||||
|
更新进度
|
||||||
|
</el-button>
|
||||||
|
</div>
|
||||||
|
<div class="overflow-y-auto pl-15px" style="height: calc(100% - 50px)">
|
||||||
|
<el-timeline class="ml-10px">
|
||||||
|
<el-timeline-item
|
||||||
|
v-for="item in followList"
|
||||||
|
:key="item.followId"
|
||||||
|
placement="bottom"
|
||||||
|
:timestamp="formatDate(item.createTime, 'YYYY-MM-DD HH:mm:ss')"
|
||||||
|
color="#30d1fc"
|
||||||
|
>
|
||||||
|
<div class="font-bold" style="font-size: 16px">{{ item.followUserName }}</div>
|
||||||
|
<div v-dompurify-html="item.content"></div>
|
||||||
|
</el-timeline-item>
|
||||||
|
</el-timeline>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<template #footer v-if="formType != 'do'">
|
||||||
|
<span>
|
||||||
|
<el-button @click="show = false">取 消</el-button>
|
||||||
|
<el-button type="primary" :disabled="formLoading" @click="handleSave">保 存</el-button>
|
||||||
|
</span>
|
||||||
|
</template>
|
||||||
|
</Dialog>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="DialogWait">
|
||||||
|
import {
|
||||||
|
createWait,
|
||||||
|
updateWait,
|
||||||
|
getWaitDetail,
|
||||||
|
getFollowWaitPage,
|
||||||
|
followWait
|
||||||
|
} from '@/api/okr/wait'
|
||||||
|
import { getEmployeeSimpleList } from '@/api/pers/employee'
|
||||||
|
import { formatDate } from '@/utils/formatTime'
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
priorityOptions: {
|
||||||
|
type: Array,
|
||||||
|
default: () => [
|
||||||
|
{ label: '高', value: '1' },
|
||||||
|
{ label: '中', value: '2' },
|
||||||
|
{ label: '低', value: '3' }
|
||||||
|
]
|
||||||
|
},
|
||||||
|
statusOptions: {
|
||||||
|
type: Array,
|
||||||
|
default: () => [
|
||||||
|
{ label: '未完成', value: '1', tagType: 'warning' },
|
||||||
|
{ label: '已关闭', value: '2', tagType: 'success' }
|
||||||
|
]
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const { t } = useI18n() // 国际化
|
||||||
|
const message = useMessage() // 消息弹窗
|
||||||
|
|
||||||
|
const toolbarConfig = {
|
||||||
|
toolbarKeys: []
|
||||||
|
}
|
||||||
|
|
||||||
|
const enableTimeRange = {
|
||||||
|
selectableRange: '07:00 - 22:00'
|
||||||
|
}
|
||||||
|
|
||||||
|
const show = ref(false)
|
||||||
|
const title = ref('')
|
||||||
|
const formType = ref('create')
|
||||||
|
|
||||||
|
const form = ref({})
|
||||||
|
const formLoading = ref(false)
|
||||||
|
const rules = ref({
|
||||||
|
title: [{ required: true, message: '请输入标题', trigger: 'blur' }],
|
||||||
|
endDate: [{ required: true, message: '请选择截止日期', trigger: 'change' }],
|
||||||
|
userList: [{ required: true, message: '请选择执行人', trigger: 'change' }]
|
||||||
|
})
|
||||||
|
|
||||||
|
const followList = ref([])
|
||||||
|
|
||||||
|
function open(type, id) {
|
||||||
|
show.value = true
|
||||||
|
title.value = { create: '新增待办', update: '修改待办', do: '更新待办进度' }[type]
|
||||||
|
formType.value = type
|
||||||
|
resetForm()
|
||||||
|
if (id) {
|
||||||
|
formLoading.value = true
|
||||||
|
try {
|
||||||
|
// 表单数据
|
||||||
|
getWaitDetail(id).then((info) => {
|
||||||
|
form.value = {
|
||||||
|
...info,
|
||||||
|
endDate: formatDate(info.endDate, 'YYYY-MM-DD'),
|
||||||
|
userList: info.userIdList
|
||||||
|
}
|
||||||
|
})
|
||||||
|
getFollowList(id)
|
||||||
|
} finally {
|
||||||
|
formLoading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
getOptions()
|
||||||
|
}
|
||||||
|
|
||||||
|
function getFollowList(id) {
|
||||||
|
// 跟进数据
|
||||||
|
getFollowWaitPage({
|
||||||
|
workId: id
|
||||||
|
}).then((data) => {
|
||||||
|
followList.value = data
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function getOptions() {
|
||||||
|
getEmployeeSimpleList().then((data) => {
|
||||||
|
employeeOptions.value = data
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function resetForm() {
|
||||||
|
form.value = {
|
||||||
|
title: undefined,
|
||||||
|
content: undefined,
|
||||||
|
priority: '1',
|
||||||
|
completeStatus: '1',
|
||||||
|
endDate: undefined,
|
||||||
|
userList: [],
|
||||||
|
notifyType: '3',
|
||||||
|
remindTime: 12,
|
||||||
|
firstNotifyTime: '09:00'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
defineExpose({ open }) // 提供 open 方法,用于打开弹窗
|
||||||
|
|
||||||
|
const formRef = ref()
|
||||||
|
|
||||||
|
/** 提交表单 */
|
||||||
|
const emit = defineEmits(['success']) // 定义 success 事件,用于操作成功后的回调
|
||||||
|
async function handleSave() {
|
||||||
|
// 校验表单
|
||||||
|
if (!formRef.value) return
|
||||||
|
const valid = await formRef.value.validate()
|
||||||
|
if (!valid) return
|
||||||
|
// 提交请求
|
||||||
|
formLoading.value = true
|
||||||
|
try {
|
||||||
|
if (formType.value === 'create') {
|
||||||
|
await createWait(form.value)
|
||||||
|
message.success(t('common.createSuccess'))
|
||||||
|
} else {
|
||||||
|
await updateWait(form.value)
|
||||||
|
message.success(t('common.updateSuccess'))
|
||||||
|
}
|
||||||
|
show.value = false
|
||||||
|
// 发送操作成功的事件
|
||||||
|
emit('success')
|
||||||
|
} finally {
|
||||||
|
formLoading.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const employeeOptions = ref([])
|
||||||
|
|
||||||
|
function addDo() {
|
||||||
|
followWait({
|
||||||
|
workId: form.value.workId,
|
||||||
|
content: form.value.followUpContent
|
||||||
|
}).then(() => {
|
||||||
|
message.success('跟进成功')
|
||||||
|
getFollowList(form.value.workId)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.wait-form {
|
||||||
|
:deep(.el-input-group__prepend) {
|
||||||
|
padding: 0 10px;
|
||||||
|
}
|
||||||
|
:deep(.el-input-group__append) {
|
||||||
|
padding: 0 10px;
|
||||||
|
}
|
||||||
|
:deep(.el-input__inner) {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
107
src/views/OKR/Management/Components/MyDuty.vue
Normal file
@@ -0,0 +1,107 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<div class="flex items-center justify-between">
|
||||||
|
<el-row>
|
||||||
|
<el-tree-select
|
||||||
|
v-model="searchForm.nodeId"
|
||||||
|
:data="peroidList"
|
||||||
|
:props="defaultProps"
|
||||||
|
:render-after-expand="false"
|
||||||
|
:default-expand-all="false"
|
||||||
|
style="width: 400px"
|
||||||
|
@change="getOkrList"
|
||||||
|
/>
|
||||||
|
</el-row>
|
||||||
|
<el-row>
|
||||||
|
<el-button type="info" @click="handleShowOkr(searchForm.nodeId)"> 节点详情 </el-button>
|
||||||
|
<el-button type="success" @click="handleUpdateProcess">更新进度</el-button>
|
||||||
|
</el-row>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<OkrTable ref="okrTableRef" canEdit />
|
||||||
|
<DialogOkr ref="dialogOkr" @edit="handleEditOkr" />
|
||||||
|
<DialogOkrInfo ref="dialogOkrInfo" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="MyDuty">
|
||||||
|
import OkrTable from './OkrTable.vue'
|
||||||
|
import DialogOkrInfo from './DialogOkrInfo.vue'
|
||||||
|
import DialogOkr from './DialogOkr.vue'
|
||||||
|
import { listToTree } from '@/utils/tree'
|
||||||
|
|
||||||
|
import { getMyNodeTree, getMyOkrPage } from '@/api/okr/okr'
|
||||||
|
|
||||||
|
const message = useMessage()
|
||||||
|
|
||||||
|
const defaultProps = {
|
||||||
|
value: 'nodeId',
|
||||||
|
label: 'nodeName',
|
||||||
|
children: 'children'
|
||||||
|
}
|
||||||
|
|
||||||
|
const okrTableRef = ref(null)
|
||||||
|
const searchForm = ref({
|
||||||
|
nodeId: undefined
|
||||||
|
})
|
||||||
|
|
||||||
|
const peroidList = ref([])
|
||||||
|
|
||||||
|
// onMounted(() => {
|
||||||
|
handleSearchPeroid()
|
||||||
|
// })
|
||||||
|
|
||||||
|
function handleSearchPeroid() {
|
||||||
|
getMyNodeTree().then((resp) => {
|
||||||
|
peroidList.value = listToTree(resp.tree, {
|
||||||
|
id: 'nodeId',
|
||||||
|
pid: 'parentId',
|
||||||
|
children: 'children'
|
||||||
|
})
|
||||||
|
searchForm.value.nodeId = resp.nodeId
|
||||||
|
getOkrList()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function getOkrList() {
|
||||||
|
getMyOkrPage(searchForm.value).then((resp) => {
|
||||||
|
const list = resp
|
||||||
|
nextTick(() => {
|
||||||
|
okrTableRef.value.prepareData(list)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const dialogOkrInfo = ref(null)
|
||||||
|
// function handleAddNode() {
|
||||||
|
// dialogOkrInfo.value.open('create', null)
|
||||||
|
// }
|
||||||
|
|
||||||
|
function handleEditOkr() {
|
||||||
|
dialogOkr.value.close()
|
||||||
|
dialogOkrInfo.value.open('update', 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleUpdateProcess() {
|
||||||
|
okrTableRef.value.updateProcess(searchForm.value.nodeId).then(() => {
|
||||||
|
message.success('更新成功')
|
||||||
|
getOkrList()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const dialogOkr = ref(null)
|
||||||
|
function handleShowOkr(id) {
|
||||||
|
dialogOkr.value.open({
|
||||||
|
nodeId: id,
|
||||||
|
canEdit: true
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
:deep(.el-overlay-dialog) {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
228
src/views/OKR/Management/Components/ObjectList.vue
Normal file
@@ -0,0 +1,228 @@
|
|||||||
|
<template>
|
||||||
|
<div style="height: 100vh">
|
||||||
|
<div class="flex items-center">
|
||||||
|
<el-popover placement="bottom" width="500px" trigger="click" @show="handleSearchPeroid">
|
||||||
|
<template #reference>
|
||||||
|
<div class="flex items-center border-1px w-300px h-32px p-10px peroid-select">
|
||||||
|
<Icon icon="ep:calendar" style="color: #aaa" />
|
||||||
|
<span class="text-14px ml-10px" style="color: #aaa">
|
||||||
|
{{ searchForm.nodeName ? searchForm.nodeName : '选择周期' }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<div>
|
||||||
|
<div class="mt-10px" style="height: 400px">
|
||||||
|
<el-table :data="peroidList" @row-click="handleSelectPeroid">
|
||||||
|
<el-table-column label="节点名称">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-radio v-model="searchForm.nodeId" :label="row.nodeId">{{
|
||||||
|
row.nodeName
|
||||||
|
}}</el-radio>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column label="开始日期" prop="startTime" width="120" />
|
||||||
|
<el-table-column label="截止日期" prop="endTime" width="120" />
|
||||||
|
</el-table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</el-popover>
|
||||||
|
<el-button class="ml-10px" type="primary" @click="handleAddNode">新建节点</el-button>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<vue3-tree-org
|
||||||
|
:data="dataList"
|
||||||
|
center
|
||||||
|
collapsable
|
||||||
|
:props="treeProps"
|
||||||
|
@on-node-click="handleClickNode"
|
||||||
|
>
|
||||||
|
<template #default="{ node }">
|
||||||
|
<div style="cursor: pointer">
|
||||||
|
<div class="tree-org-node__text node-label">
|
||||||
|
<el-popover placement="right" title="目标" width="270px" trigger="hover">
|
||||||
|
<template #reference>
|
||||||
|
<div>
|
||||||
|
<div style="max-height: 40px; overflow: hidden">{{ node.label }}</div>
|
||||||
|
<div class="tip"> 目标数: {{ getNodePropData(node.id, 'objectiveNum') }}</div>
|
||||||
|
<!-- 执行人 -->
|
||||||
|
<div class="tip">
|
||||||
|
执行人: {{ getNodePropData(node.id, 'executorName') || '无' }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
<template #default>
|
||||||
|
<div style="font-size: 0.875rem; line-height: 1.5">
|
||||||
|
<div v-for="i in getNodePropData(node.id, 'objectives')" :key="i.id" class="mt-5">
|
||||||
|
<span style="color: #aaa">{{ parseInt(i.progress) }}%</span>
|
||||||
|
<span style="margin-left: 0.5rem">
|
||||||
|
{{ i.objectiveName }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</el-popover>
|
||||||
|
</div>
|
||||||
|
<div class="p-5px">
|
||||||
|
<el-progress
|
||||||
|
type="line"
|
||||||
|
:percentage="parseInt(getNodePropData(node.id, 'progress')) || 0"
|
||||||
|
text-inside
|
||||||
|
:stroke-width="12"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</vue3-tree-org>
|
||||||
|
<DialogOkr ref="dialogOkr" @edit="handleEditOkr" />
|
||||||
|
<DialogOkrInfo ref="dialogOkrInfo" @close="openOkr" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="ObjectList">
|
||||||
|
import { Vue3TreeOrg } from 'vue3-tree-org'
|
||||||
|
import 'vue3-tree-org/lib/vue3-tree-org.css'
|
||||||
|
import DialogOkr from './DialogOkr.vue'
|
||||||
|
import DialogOkrInfo from './DialogOkrInfo.vue'
|
||||||
|
|
||||||
|
import { getOkrRelationTree, getOkrRelationTreeChildren } from '@/api/okr/okr'
|
||||||
|
import { listToTree } from '@/utils/tree'
|
||||||
|
|
||||||
|
const dataList = ref({})
|
||||||
|
const helpDataList = ref([])
|
||||||
|
const treeProps = {
|
||||||
|
children: 'children',
|
||||||
|
label: 'nodeName',
|
||||||
|
id: 'nodeId',
|
||||||
|
pid: 'parentId'
|
||||||
|
}
|
||||||
|
|
||||||
|
const searchForm = ref({
|
||||||
|
nodeName: '',
|
||||||
|
nodeId: null
|
||||||
|
})
|
||||||
|
|
||||||
|
const peroidList = ref([])
|
||||||
|
|
||||||
|
handleSearchPeroid()
|
||||||
|
|
||||||
|
function handleSearchPeroid() {
|
||||||
|
getOkrRelationTree().then((resp) => {
|
||||||
|
peroidList.value = resp
|
||||||
|
if (resp && resp.length && !searchForm.value.nodeId) {
|
||||||
|
searchForm.value.nodeId = resp[0].nodeId
|
||||||
|
searchForm.value.nodeName = resp[0].nodeName
|
||||||
|
getOkrList()
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function getOkrList() {
|
||||||
|
getOkrRelationTreeChildren({
|
||||||
|
nodeId: searchForm.value.nodeId
|
||||||
|
}).then((resp) => {
|
||||||
|
const tree = listToTree(resp, {
|
||||||
|
id: 'nodeId',
|
||||||
|
pid: 'parentId',
|
||||||
|
children: 'children'
|
||||||
|
})
|
||||||
|
helpDataList.value = resp
|
||||||
|
if (tree && tree.length) {
|
||||||
|
dataList.value = tree[0]
|
||||||
|
} else {
|
||||||
|
dataList.value = {}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleSelectPeroid(row) {
|
||||||
|
searchForm.value.nodeName = row.nodeName
|
||||||
|
searchForm.value.nodeId = row.nodeId
|
||||||
|
getOkrList()
|
||||||
|
}
|
||||||
|
|
||||||
|
function getNodePropData(nodeId, prop) {
|
||||||
|
const nodeData = helpDataList.value.find((it) => it.nodeId == nodeId)
|
||||||
|
if (nodeData) {
|
||||||
|
return nodeData[prop]
|
||||||
|
} else {
|
||||||
|
return ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function toggleExpand(data, val) {
|
||||||
|
if (Array.isArray(data)) {
|
||||||
|
data.forEach((item) => {
|
||||||
|
item.expand = val
|
||||||
|
if (item.children) {
|
||||||
|
toggleExpand(item.children, val)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
data.expand = val
|
||||||
|
if (data.children) {
|
||||||
|
toggleExpand(data.children, val)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
toggleExpand(dataList.value, true)
|
||||||
|
|
||||||
|
const dialogOkr = ref(null)
|
||||||
|
const clickNode = ref(null)
|
||||||
|
function handleClickNode(node, data) {
|
||||||
|
clickNode.value = data
|
||||||
|
openOkr()
|
||||||
|
}
|
||||||
|
|
||||||
|
function openOkr() {
|
||||||
|
clickNode.value &&
|
||||||
|
dialogOkr.value.open({
|
||||||
|
nodeId: clickNode.value.nodeId,
|
||||||
|
canEdit: !clickNode.value.children || clickNode.value.children.length == 0
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const dialogOkrInfo = ref(null)
|
||||||
|
function handleAddNode() {
|
||||||
|
clickNode.value = null
|
||||||
|
dialogOkrInfo.value.open('create', null)
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleEditOkr() {
|
||||||
|
dialogOkr.value.close()
|
||||||
|
dialogOkrInfo.value.open('update', 1)
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.peroid-select {
|
||||||
|
border-radius: 4px;
|
||||||
|
cursor: pointer;
|
||||||
|
&:hover {
|
||||||
|
border-color: var(--el-color-primary-light-5);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.tree-org-node__text {
|
||||||
|
min-width: 200px;
|
||||||
|
min-height: 80px;
|
||||||
|
text-align: left;
|
||||||
|
font-size: 14px;
|
||||||
|
border-bottom: 1px solid #ccc;
|
||||||
|
padding: 5px;
|
||||||
|
}
|
||||||
|
.tip {
|
||||||
|
position: relative;
|
||||||
|
color: #aaa;
|
||||||
|
font-size: 0.875rem;
|
||||||
|
}
|
||||||
|
:deep(.el-progress-bar__innerText) {
|
||||||
|
display: block;
|
||||||
|
font-size: 0.75rem !important;
|
||||||
|
}
|
||||||
|
:deep(.el-overlay-dialog) {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
181
src/views/OKR/Management/Components/OkrTable.vue
Normal file
@@ -0,0 +1,181 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<el-table
|
||||||
|
ref="tableRef"
|
||||||
|
:data="okrList"
|
||||||
|
default-expand-all
|
||||||
|
row-key="id"
|
||||||
|
size="large"
|
||||||
|
@row-click="handleRowClick"
|
||||||
|
@expand-change="handleExpand"
|
||||||
|
>
|
||||||
|
<el-table-column label="目标/关键成果">
|
||||||
|
<template #default="{ row, $index }">
|
||||||
|
<span
|
||||||
|
v-if="!expandedRows[row.id]"
|
||||||
|
class="line1"
|
||||||
|
:style="{
|
||||||
|
top: row.type == '目标' ? '30px' : 0,
|
||||||
|
height: getHeight(row, $index)
|
||||||
|
}"
|
||||||
|
></span>
|
||||||
|
<span v-if="row.type == '目标'">【目标】{{ row.name }}</span>
|
||||||
|
<template v-else>
|
||||||
|
<span class="line2"></span>
|
||||||
|
<span>
|
||||||
|
【关键成果】{{ row.channelName }} {{ row.name }}
|
||||||
|
<span v-if="row.resultType == 1">达 {{ row.targetValue }}</span>
|
||||||
|
</span>
|
||||||
|
<div class="flex items-center mt-10px ml-50px">
|
||||||
|
<span>当前进度:</span>
|
||||||
|
<span v-if="row.isSys || !props.canEdit">{{ row.currentValue }}</span>
|
||||||
|
<el-input
|
||||||
|
v-else-if="row.resultType == 1"
|
||||||
|
v-model="row.currentValue"
|
||||||
|
clearable
|
||||||
|
size="small"
|
||||||
|
style="width: 200px"
|
||||||
|
/>
|
||||||
|
<el-radio-group
|
||||||
|
v-else-if="row.resultType == 2"
|
||||||
|
v-model="row.currentValue"
|
||||||
|
size="small"
|
||||||
|
>
|
||||||
|
<el-radio :value="1">是</el-radio>
|
||||||
|
<el-radio :value="0">否</el-radio>
|
||||||
|
</el-radio-group>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="progress" label="进度" width="200px">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-progress :percentage="parseInt(row.progress) || 0" />
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="executorName" label="执行人" width="200px" />
|
||||||
|
</el-table>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="OkrTable">
|
||||||
|
import { updateOkrProgress } from '@/api/okr/okr'
|
||||||
|
|
||||||
|
const emit = defineEmits(['rowClick'])
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
canEdit: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const okrList = ref([])
|
||||||
|
const helpList = ref([])
|
||||||
|
|
||||||
|
function prepareData(list) {
|
||||||
|
helpList.value = []
|
||||||
|
expandedRows.value = {}
|
||||||
|
okrList.value = list.map((item) => {
|
||||||
|
let obj = {
|
||||||
|
id: item.objectiveId,
|
||||||
|
name: item.objectiveName,
|
||||||
|
progress: item.progress,
|
||||||
|
executorName: item.executorName,
|
||||||
|
type: '目标'
|
||||||
|
}
|
||||||
|
helpList.value.push(obj)
|
||||||
|
if (item.keyResults) {
|
||||||
|
obj.children = item.keyResults.map((child) => {
|
||||||
|
let kr = {
|
||||||
|
id: child.keyResultId,
|
||||||
|
keyResultId: child.keyResultId,
|
||||||
|
objectiveId: child.objectiveId,
|
||||||
|
nodeId: child.nodeId,
|
||||||
|
isSys: child.isSys,
|
||||||
|
processId: child.id,
|
||||||
|
name: child.keyResultName,
|
||||||
|
progress: child.progress,
|
||||||
|
executorName: child.executorName,
|
||||||
|
type: '关键成果',
|
||||||
|
resultType: child.resultType,
|
||||||
|
targetValue: child.targetValue,
|
||||||
|
currentValue: Number(child.currentValue)
|
||||||
|
}
|
||||||
|
helpList.value.push(kr)
|
||||||
|
return kr
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return obj
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
defineExpose({
|
||||||
|
prepareData,
|
||||||
|
updateProcess
|
||||||
|
})
|
||||||
|
function getHeight(row, index) {
|
||||||
|
if (helpList.value.length - 1 == index || helpList.value[index + 1].type == '目标') {
|
||||||
|
return '22px'
|
||||||
|
} else {
|
||||||
|
return '100%'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const expandedRows = ref({})
|
||||||
|
function handleExpand(row, expanded) {
|
||||||
|
expandedRows.value[row.id] = !expanded
|
||||||
|
}
|
||||||
|
|
||||||
|
function updateProcess(nodeId) {
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
try {
|
||||||
|
let arr = []
|
||||||
|
okrList.value.map((item) => {
|
||||||
|
arr = [...arr, ...(item.children || []).map((it) => ({ ...it, id: it.processId }))]
|
||||||
|
})
|
||||||
|
updateOkrProgress({
|
||||||
|
nodeId,
|
||||||
|
keyResultProgressList: arr
|
||||||
|
}).then((resp) => {
|
||||||
|
resolve(resp)
|
||||||
|
})
|
||||||
|
} catch (error) {
|
||||||
|
reject(error)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleRowClick(row) {
|
||||||
|
emit('rowClick', row)
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
.line1 {
|
||||||
|
position: absolute;
|
||||||
|
width: 1px;
|
||||||
|
left: 21px;
|
||||||
|
border-left: 1px dashed #ccc;
|
||||||
|
}
|
||||||
|
.line2 {
|
||||||
|
position: absolute;
|
||||||
|
width: 26px;
|
||||||
|
height: 1px;
|
||||||
|
left: 21px;
|
||||||
|
top: 22px;
|
||||||
|
border-bottom: 1px dashed #ccc;
|
||||||
|
&::after {
|
||||||
|
position: absolute;
|
||||||
|
left: 25px;
|
||||||
|
top: -3px;
|
||||||
|
display: block;
|
||||||
|
width: 6px;
|
||||||
|
height: 6px;
|
||||||
|
border-radius: 6px;
|
||||||
|
margin: 0 auto;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
content: '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
293
src/views/OKR/Management/Components/WaitTarget.vue
Normal file
@@ -0,0 +1,293 @@
|
|||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<el-form :model="searchForm" inline label-width="0" @submit.prevent="searchList">
|
||||||
|
<el-form-item>
|
||||||
|
<el-input v-model="searchForm.title" placeholder="输入标题" />
|
||||||
|
</el-form-item>
|
||||||
|
<el-form-item>
|
||||||
|
<el-select v-model="searchForm.priority" placeholder="选择优先级" clearable filterable>
|
||||||
|
<el-option
|
||||||
|
v-for="item in priorityOptions"
|
||||||
|
:key="item.value"
|
||||||
|
:label="item.label"
|
||||||
|
:value="item.value"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
<!-- <el-form-item>
|
||||||
|
<el-select
|
||||||
|
v-model="searchForm.completeStatus"
|
||||||
|
placeholder="选择完成状态"
|
||||||
|
clearable
|
||||||
|
filterable
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="item in statusOptions"
|
||||||
|
:key="item.value"
|
||||||
|
:label="item.label"
|
||||||
|
:value="item.value"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item> -->
|
||||||
|
<el-form-item>
|
||||||
|
<el-button @click="searchList"> 搜索</el-button>
|
||||||
|
<el-button type="primary" @click="handleAdd"> 新增</el-button>
|
||||||
|
</el-form-item>
|
||||||
|
</el-form>
|
||||||
|
<el-tabs v-model="tabIndex" @tab-change="searchList">
|
||||||
|
<el-tab-pane :name="1">
|
||||||
|
<template #label>
|
||||||
|
<el-badge :value="tabCount.dayEndAgentWorkNum" :max="99" :show-zero="false">
|
||||||
|
<el-tooltip content="截止日在今日及之前的未关闭事项" placement="top" effect="dark">
|
||||||
|
<Icon icon="ep:question-filled" />
|
||||||
|
</el-tooltip>
|
||||||
|
<span>今日截止</span>
|
||||||
|
</el-badge>
|
||||||
|
</template>
|
||||||
|
</el-tab-pane>
|
||||||
|
<el-tab-pane :name="2">
|
||||||
|
<template #label>
|
||||||
|
<el-tooltip content="需要我执行的全部事项" placement="top" effect="dark">
|
||||||
|
<Icon icon="ep:question-filled" />
|
||||||
|
</el-tooltip>
|
||||||
|
<span>我的待办</span>
|
||||||
|
</template>
|
||||||
|
</el-tab-pane>
|
||||||
|
<el-tab-pane :name="3">
|
||||||
|
<template #label>
|
||||||
|
<el-badge :value="tabCount.urgeAgentWorkNum" :max="99" :show-zero="false">
|
||||||
|
<el-tooltip content="指派他人去办的事项" placement="top" effect="dark">
|
||||||
|
<Icon icon="ep:question-filled" />
|
||||||
|
</el-tooltip>
|
||||||
|
<span>催办事项</span>
|
||||||
|
</el-badge>
|
||||||
|
</template>
|
||||||
|
</el-tab-pane>
|
||||||
|
<el-tab-pane :name="4">
|
||||||
|
<template #label>
|
||||||
|
<el-badge :value="tabCount.notifyNum" :max="99" :show-zero="false">
|
||||||
|
<el-tooltip content="特指OKR中@我的消息" placement="top" effect="dark">
|
||||||
|
<Icon icon="ep:question-filled" />
|
||||||
|
</el-tooltip>
|
||||||
|
<span>通知我的</span>
|
||||||
|
</el-badge>
|
||||||
|
</template>
|
||||||
|
<el-table :data="mentionedList">
|
||||||
|
<el-table-column type="index" width="50" />
|
||||||
|
<el-table-column prop="content" label="okr标题" />
|
||||||
|
<el-table-column prop="createUserName" label="创建人" />
|
||||||
|
<el-table-column prop="createDate" label="创建日期" />
|
||||||
|
<el-table-column fixed="right" label="操作" width="140">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-button style="padding: 0" type="primary" text @click="handleShow(row)">
|
||||||
|
查看
|
||||||
|
</el-button>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
</el-table>
|
||||||
|
</el-tab-pane>
|
||||||
|
</el-tabs>
|
||||||
|
|
||||||
|
<el-table v-if="tabIndex != 4" :data="tableList">
|
||||||
|
<el-table-column type="index" width="50" />
|
||||||
|
<el-table-column prop="title" label="标题" />
|
||||||
|
<el-table-column prop="priority" label="优先级">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-tag :type="priorityFilter(row.priority)">
|
||||||
|
{{ priorityNameFilter(row.priority) }}
|
||||||
|
</el-tag>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="creatorName" label="创建人" />
|
||||||
|
<el-table-column prop="createTime" label="创建日期" :formatter="dateFormatter" />
|
||||||
|
<el-table-column prop="userNameStr" label="执行人" />
|
||||||
|
<el-table-column prop="endDate" label="截止日期" :formatter="dateFormatter" />
|
||||||
|
<el-table-column label="状态">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-tag :type="statusFilter(row.completeStatus)">{{
|
||||||
|
statusNameFilter(row.completeStatus)
|
||||||
|
}}</el-tag>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
<el-table-column prop="completedDate" label="关闭日期" :formatter="dateFormatter" />
|
||||||
|
<el-table-column fixed="right" label="操作" width="140">
|
||||||
|
<template #default="{ row }">
|
||||||
|
<el-button
|
||||||
|
v-if="row.userIdList.includes(currentUserId) && row.completeStatus == 1"
|
||||||
|
style="padding: 0; margin-right: 10px"
|
||||||
|
type="primary"
|
||||||
|
text
|
||||||
|
@click="handleDo(row)"
|
||||||
|
>
|
||||||
|
跟进
|
||||||
|
</el-button>
|
||||||
|
<el-button
|
||||||
|
v-if="!row.userIdList.includes(currentUserId) && row.completeStatus == 1"
|
||||||
|
style="padding: 0; margin-right: 10px; margin-left: 0"
|
||||||
|
type="primary"
|
||||||
|
text
|
||||||
|
@click="handleNotice(row)"
|
||||||
|
>
|
||||||
|
催办
|
||||||
|
</el-button>
|
||||||
|
<el-button
|
||||||
|
v-if="row.creator == currentUserId && row.completeStatus == 1"
|
||||||
|
style="padding: 0; margin-right: 10px; margin-left: 0"
|
||||||
|
type="primary"
|
||||||
|
text
|
||||||
|
@click="handleEdit(row)"
|
||||||
|
>
|
||||||
|
修改
|
||||||
|
</el-button>
|
||||||
|
<el-button
|
||||||
|
v-if="row.creator == currentUserId && row.completeStatus == 1"
|
||||||
|
style="padding: 0; margin-right: 10px; margin-left: 0"
|
||||||
|
type="danger"
|
||||||
|
text
|
||||||
|
@click="handleDelete(row)"
|
||||||
|
>
|
||||||
|
删除
|
||||||
|
</el-button>
|
||||||
|
</template>
|
||||||
|
</el-table-column>
|
||||||
|
</el-table>
|
||||||
|
<Pagination
|
||||||
|
v-model:limit="searchForm.pageSize"
|
||||||
|
v-model:page="searchForm.pageNo"
|
||||||
|
:total="total"
|
||||||
|
@pagination="getList"
|
||||||
|
/>
|
||||||
|
|
||||||
|
<DialogWait ref="waitDialogRef" @success="getList" />
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="WaitTarget">
|
||||||
|
import { useUserStore } from '@/store/modules/user'
|
||||||
|
import { dateFormatter } from '@/utils/formatTime'
|
||||||
|
import DialogWait from './DialogWait.vue'
|
||||||
|
|
||||||
|
import { getWaitPage, deleteWait, getWaitCount, urgeWait } from '@/api/okr/wait'
|
||||||
|
|
||||||
|
const userStore = useUserStore()
|
||||||
|
const currentUserId = userStore.getUser.id
|
||||||
|
|
||||||
|
const message = useMessage()
|
||||||
|
|
||||||
|
const searchForm = ref({
|
||||||
|
title: undefined,
|
||||||
|
completeStatus: undefined,
|
||||||
|
priority: undefined,
|
||||||
|
pageNo: 1,
|
||||||
|
pageSize: 50
|
||||||
|
})
|
||||||
|
|
||||||
|
const statusOptions = [
|
||||||
|
{ label: '未完成', value: 1, tagType: 'warning' },
|
||||||
|
{ label: '已关闭', value: 2, tagType: 'success' }
|
||||||
|
]
|
||||||
|
|
||||||
|
const statusFilter = (status) => {
|
||||||
|
return statusOptions.find((item) => item.value == status)?.tagType
|
||||||
|
}
|
||||||
|
|
||||||
|
const statusNameFilter = (status) => {
|
||||||
|
return statusOptions.find((item) => item.value == status)?.label
|
||||||
|
}
|
||||||
|
|
||||||
|
const priorityOptions = [
|
||||||
|
{ label: '高', value: '1', tagType: 'danger' },
|
||||||
|
{ label: '中', value: '2', tagType: 'warning' },
|
||||||
|
{ label: '低', value: '3', tagType: 'info' }
|
||||||
|
]
|
||||||
|
|
||||||
|
const priorityFilter = (priority) => {
|
||||||
|
return priorityOptions.find((item) => item.value == priority)?.tagType
|
||||||
|
}
|
||||||
|
|
||||||
|
const priorityNameFilter = (priority) => {
|
||||||
|
return priorityOptions.find((item) => item.value == priority)?.label
|
||||||
|
}
|
||||||
|
|
||||||
|
const tabIndex = ref(1)
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
searchList()
|
||||||
|
})
|
||||||
|
|
||||||
|
function searchList() {
|
||||||
|
searchForm.value.pageNo = 1
|
||||||
|
getList()
|
||||||
|
}
|
||||||
|
|
||||||
|
const tabCount = ref({
|
||||||
|
dayEndAgentWorkNum: 0,
|
||||||
|
myAgentWorkNum: 0,
|
||||||
|
urgeAgentWorkNum: 0,
|
||||||
|
notifyNum: 0
|
||||||
|
})
|
||||||
|
function getTabCount() {
|
||||||
|
getWaitCount({
|
||||||
|
title: searchForm.value.title,
|
||||||
|
priority: searchForm.value.priority
|
||||||
|
}).then((res) => {
|
||||||
|
tabCount.value = res
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
const loading = ref(false)
|
||||||
|
const tableList = ref([])
|
||||||
|
const mentionedList = ref([])
|
||||||
|
const total = ref(0)
|
||||||
|
function getList() {
|
||||||
|
loading.value = true
|
||||||
|
try {
|
||||||
|
searchForm.value.queryType = tabIndex.value
|
||||||
|
getWaitPage(searchForm.value).then((res) => {
|
||||||
|
tableList.value = res.list
|
||||||
|
total.value = res.total
|
||||||
|
loading.value = false
|
||||||
|
})
|
||||||
|
getTabCount()
|
||||||
|
} catch (error) {}
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleDo(row) {
|
||||||
|
waitDialogRef.value.open('do', row.workId)
|
||||||
|
}
|
||||||
|
|
||||||
|
const waitDialogRef = ref()
|
||||||
|
function handleAdd() {
|
||||||
|
waitDialogRef.value.open('create')
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleEdit(row) {
|
||||||
|
waitDialogRef.value.open('update', row.workId)
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleDelete(row) {
|
||||||
|
message.confirm('确定删除待办事项吗?').then(() => {
|
||||||
|
deleteWait(row.workId).then(() => {
|
||||||
|
message.success('删除成功')
|
||||||
|
getList()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleNotice(row) {
|
||||||
|
console.log(row)
|
||||||
|
|
||||||
|
message.confirm('即将发送微信通知提醒执行人,是否继续?').then(() => {
|
||||||
|
urgeWait({ workId: row.workId }).then(() => {
|
||||||
|
message.success('发送成功')
|
||||||
|
})
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function handleShow(row) {
|
||||||
|
console.log(row)
|
||||||
|
message.success('打开okr详情页')
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped></style>
|
||||||
62
src/views/OKR/Management/index.vue
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
<template>
|
||||||
|
<div :style="{ height: `${height}px` }">
|
||||||
|
<el-tabs tab-position="left" v-model="tabIndex" style="height: 100%">
|
||||||
|
<el-tab-pane disabled name="-1">
|
||||||
|
<template #label>
|
||||||
|
<div class="w-full">
|
||||||
|
<div style="text-align: left; transform: translateX(-10px)">目标</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</el-tab-pane>
|
||||||
|
<el-tab-pane label="我负责的" name="0">
|
||||||
|
<MyDuty v-if="tabIndex == 0" />
|
||||||
|
</el-tab-pane>
|
||||||
|
<el-tab-pane label="待办事项" name="1">
|
||||||
|
<WaitTarget v-if="tabIndex == 1" />
|
||||||
|
</el-tab-pane>
|
||||||
|
<el-tab-pane label="全部目标" name="2">
|
||||||
|
<AllTarget v-if="tabIndex == 2" />
|
||||||
|
</el-tab-pane>
|
||||||
|
<el-tab-pane label="目标关系树" name="3">
|
||||||
|
<ObjectList v-if="tabIndex == 3" />
|
||||||
|
</el-tab-pane>
|
||||||
|
<el-tab-pane disabled>
|
||||||
|
<template #label>
|
||||||
|
<div class="w-full">
|
||||||
|
<el-divider
|
||||||
|
direction="horizontal"
|
||||||
|
style="width: calc(100% + 10px) !important; transform: translateX(-10px)"
|
||||||
|
/>
|
||||||
|
<div style="text-align: left; transform: translateX(-10px)">成员</div>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</el-tab-pane>
|
||||||
|
<el-tab-pane label="张三">张三</el-tab-pane>
|
||||||
|
<el-tab-pane label="李四">李四</el-tab-pane>
|
||||||
|
<el-tab-pane label="王二">王二</el-tab-pane>
|
||||||
|
</el-tabs>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="OkrManagement">
|
||||||
|
import AllTarget from './Components/AllTarget.vue'
|
||||||
|
import MyDuty from './Components/MyDuty.vue'
|
||||||
|
import ObjectList from './Components/ObjectList.vue'
|
||||||
|
import WaitTarget from './Components/WaitTarget.vue'
|
||||||
|
|
||||||
|
const tabIndex = ref('0')
|
||||||
|
const height = ref(innerHeight - 115)
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
:deep(.el-tabs__content) {
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
:deep(.el-tab-pane) {
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
:deep(.el-popover__title) {
|
||||||
|
font-size: 0.875rem;
|
||||||
|
color: #aaa !important;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
@@ -16,8 +16,8 @@
|
|||||||
<el-col :span="12">
|
<el-col :span="12">
|
||||||
<el-form-item label="用户性别">
|
<el-form-item label="用户性别">
|
||||||
<el-radio-group v-model="formData.sex">
|
<el-radio-group v-model="formData.sex">
|
||||||
<el-radio :label="1"> 男 </el-radio>
|
<el-radio :value="1"> 男 </el-radio>
|
||||||
<el-radio :label="2"> 女 </el-radio>
|
<el-radio :value="2"> 女 </el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
@@ -143,12 +143,35 @@
|
|||||||
<el-input v-model="formData.dingUserId" disabled />
|
<el-input v-model="formData.dingUserId" disabled />
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
|
</el-row>
|
||||||
|
<el-row :gutter="20">
|
||||||
<el-col :span="12">
|
<el-col :span="12">
|
||||||
<el-form-item label="微信号" prop="wxAlias">
|
<el-form-item label="微信号" prop="wxAlias">
|
||||||
<el-input v-model="formData.wxAlias" placeholder="请输入微信号" />
|
<el-input v-model="formData.wxAlias" placeholder="请输入微信号" />
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-col>
|
</el-col>
|
||||||
|
<el-col :span="12">
|
||||||
|
<el-form-item label="绩效考核人" prop="meritsExamineUserList">
|
||||||
|
<el-select
|
||||||
|
v-model="formData.meritsExamineUserList"
|
||||||
|
placeholder="选择绩效考核人"
|
||||||
|
clearable
|
||||||
|
filterable
|
||||||
|
multiple
|
||||||
|
collapse-tags
|
||||||
|
>
|
||||||
|
<el-option
|
||||||
|
v-for="item in employeeOptions"
|
||||||
|
:key="item.id"
|
||||||
|
:disabled="item.status == 1"
|
||||||
|
:label="item.name"
|
||||||
|
:value="item.id"
|
||||||
|
/>
|
||||||
|
</el-select>
|
||||||
|
</el-form-item>
|
||||||
|
</el-col>
|
||||||
</el-row>
|
</el-row>
|
||||||
|
|
||||||
<el-row :gutter="20">
|
<el-row :gutter="20">
|
||||||
<el-col :span="24">
|
<el-col :span="24">
|
||||||
<el-form-item label="备注">
|
<el-form-item label="备注">
|
||||||
@@ -194,6 +217,7 @@ const formRules = ref({
|
|||||||
const formRef = ref() // 表单 Ref
|
const formRef = ref() // 表单 Ref
|
||||||
const attendanceSettingIdOptions = ref([])
|
const attendanceSettingIdOptions = ref([])
|
||||||
const instanceIdsOptions = ref([])
|
const instanceIdsOptions = ref([])
|
||||||
|
const employeeOptions = ref([])
|
||||||
|
|
||||||
/** 打开弹窗 */
|
/** 打开弹窗 */
|
||||||
const open = async (type, id) => {
|
const open = async (type, id) => {
|
||||||
@@ -221,6 +245,9 @@ function getOptions() {
|
|||||||
getPlanSimpleList().then((data) => {
|
getPlanSimpleList().then((data) => {
|
||||||
attendanceSettingIdOptions.value = data
|
attendanceSettingIdOptions.value = data
|
||||||
})
|
})
|
||||||
|
EmployeeApi.getEmployeeSimpleList().then((data) => {
|
||||||
|
employeeOptions.value = data
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
/** 提交表单 */
|
/** 提交表单 */
|
||||||
@@ -267,7 +294,8 @@ const resetForm = () => {
|
|||||||
retireDate: undefined,
|
retireDate: undefined,
|
||||||
attendanceSettingId: undefined,
|
attendanceSettingId: undefined,
|
||||||
instanceIds: [],
|
instanceIds: [],
|
||||||
wxAlias: ''
|
wxAlias: '',
|
||||||
|
meritsExamineUserList: []
|
||||||
}
|
}
|
||||||
formRef.value?.resetFields()
|
formRef.value?.resetFields()
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,8 +11,8 @@
|
|||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
<el-radio-group v-model="searchForm.status" @change="handleQuery">
|
<el-radio-group v-model="searchForm.status" @change="handleQuery">
|
||||||
<el-radio :label="0"> 在职 </el-radio>
|
<el-radio :value="0"> 在职 </el-radio>
|
||||||
<el-radio :label="1"> 离职 </el-radio>
|
<el-radio :value="1"> 离职 </el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item>
|
<el-form-item>
|
||||||
@@ -36,6 +36,7 @@
|
|||||||
<el-table-column label="微信号" prop="wxAlias" min-width="120" />
|
<el-table-column label="微信号" prop="wxAlias" min-width="120" />
|
||||||
<el-table-column label="钉钉号" prop="dingUserId" min-width="120" />
|
<el-table-column label="钉钉号" prop="dingUserId" min-width="120" />
|
||||||
<el-table-column label="考勤方案" prop="attendanceSettingName" width="100" />
|
<el-table-column label="考勤方案" prop="attendanceSettingName" width="100" />
|
||||||
|
<el-table-column label="绩效考核人" prop="meritsExamineUserNames" width="120" />
|
||||||
<el-table-column label="已开通系统" prop="instanceName" min-width="120" />
|
<el-table-column label="已开通系统" prop="instanceName" min-width="120" />
|
||||||
<el-table-column label="在职状态" key="status" width="150">
|
<el-table-column label="在职状态" key="status" width="150">
|
||||||
<template #default="scope">
|
<template #default="scope">
|
||||||
|
|||||||
@@ -2,8 +2,8 @@
|
|||||||
<Form ref="formRef" :labelWidth="80" :rules="rules" :schema="schema">
|
<Form ref="formRef" :labelWidth="80" :rules="rules" :schema="schema">
|
||||||
<template #sex="form">
|
<template #sex="form">
|
||||||
<el-radio-group v-model="form['sex']">
|
<el-radio-group v-model="form['sex']">
|
||||||
<el-radio :label="1">{{ t('profile.user.man') }}</el-radio>
|
<el-radio :value="1">{{ t('profile.user.man') }}</el-radio>
|
||||||
<el-radio :label="2">{{ t('profile.user.woman') }}</el-radio>
|
<el-radio :value="2">{{ t('profile.user.woman') }}</el-radio>
|
||||||
</el-radio-group>
|
</el-radio-group>
|
||||||
</template>
|
</template>
|
||||||
</Form>
|
</Form>
|
||||||
|
|||||||
74
yarn.lock
@@ -1227,7 +1227,7 @@
|
|||||||
resolved "https://registry.yarnpkg.com/@ctrl/tinycolor/-/tinycolor-3.6.1.tgz#b6c75a56a1947cc916ea058772d666a2c8932f31"
|
resolved "https://registry.yarnpkg.com/@ctrl/tinycolor/-/tinycolor-3.6.1.tgz#b6c75a56a1947cc916ea058772d666a2c8932f31"
|
||||||
integrity sha512-SITSV6aIXsuVNV3f3O0f2n/cgyEDWoSqtZMYiAmcsYHydcKrOz3gUxB/iXd/Qf08+IZX4KpgNbvUdMBmWz+kcA==
|
integrity sha512-SITSV6aIXsuVNV3f3O0f2n/cgyEDWoSqtZMYiAmcsYHydcKrOz3gUxB/iXd/Qf08+IZX4KpgNbvUdMBmWz+kcA==
|
||||||
|
|
||||||
"@element-plus/icons-vue@^2.0.6", "@element-plus/icons-vue@^2.1.0":
|
"@element-plus/icons-vue@^2.1.0", "@element-plus/icons-vue@^2.3.1":
|
||||||
version "2.3.1"
|
version "2.3.1"
|
||||||
resolved "https://registry.yarnpkg.com/@element-plus/icons-vue/-/icons-vue-2.3.1.tgz#1f635ad5fdd5c85ed936481525570e82b5a8307a"
|
resolved "https://registry.yarnpkg.com/@element-plus/icons-vue/-/icons-vue-2.3.1.tgz#1f635ad5fdd5c85ed936481525570e82b5a8307a"
|
||||||
integrity sha512-XxVUZv48RZAd87ucGS48jPf6pKu0yV5UCg9f4FFwtrYxXOwWuVJo6wOvSLKEoMQKjv8GsX/mhP6UsC1lRwbUWg==
|
integrity sha512-XxVUZv48RZAd87ucGS48jPf6pKu0yV5UCg9f4FFwtrYxXOwWuVJo6wOvSLKEoMQKjv8GsX/mhP6UsC1lRwbUWg==
|
||||||
@@ -1374,25 +1374,25 @@
|
|||||||
resolved "https://registry.yarnpkg.com/@eslint/js/-/js-8.57.0.tgz#a5417ae8427873f1dd08b70b3574b453e67b5f7f"
|
resolved "https://registry.yarnpkg.com/@eslint/js/-/js-8.57.0.tgz#a5417ae8427873f1dd08b70b3574b453e67b5f7f"
|
||||||
integrity sha512-Ys+3g2TaW7gADOJzPt83SJtCDhMjndcDMFVQ/Tj9iA1BfJzFKD9mAUXT3OenpuPHbI6P/myECxRJrofUsDx/5g==
|
integrity sha512-Ys+3g2TaW7gADOJzPt83SJtCDhMjndcDMFVQ/Tj9iA1BfJzFKD9mAUXT3OenpuPHbI6P/myECxRJrofUsDx/5g==
|
||||||
|
|
||||||
"@floating-ui/core@^1.0.0":
|
"@floating-ui/core@^1.6.0":
|
||||||
version "1.6.0"
|
version "1.6.9"
|
||||||
resolved "https://registry.yarnpkg.com/@floating-ui/core/-/core-1.6.0.tgz#fa41b87812a16bf123122bf945946bae3fdf7fc1"
|
resolved "https://registry.yarnpkg.com/@floating-ui/core/-/core-1.6.9.tgz#64d1da251433019dafa091de9b2886ff35ec14e6"
|
||||||
integrity sha512-PcF++MykgmTj3CIyOQbKA/hDzOAiqI3mhuoN44WRCopIs1sgoDoU4oty4Jtqaj/y3oDU6fnVSm4QG0a3t5i0+g==
|
integrity sha512-uMXCuQ3BItDUbAMhIXw7UPXRfAlOAvZzdK9BWpE60MCn+Svt3aLn9jsPTi/WNGlRUu2uI0v5S7JiIUsbsvh3fw==
|
||||||
dependencies:
|
dependencies:
|
||||||
"@floating-ui/utils" "^0.2.1"
|
"@floating-ui/utils" "^0.2.9"
|
||||||
|
|
||||||
"@floating-ui/dom@^1.0.1":
|
"@floating-ui/dom@^1.0.1":
|
||||||
version "1.6.3"
|
version "1.6.13"
|
||||||
resolved "https://registry.yarnpkg.com/@floating-ui/dom/-/dom-1.6.3.tgz#954e46c1dd3ad48e49db9ada7218b0985cee75ef"
|
resolved "https://registry.yarnpkg.com/@floating-ui/dom/-/dom-1.6.13.tgz#a8a938532aea27a95121ec16e667a7cbe8c59e34"
|
||||||
integrity sha512-RnDthu3mzPlQ31Ss/BTwQ1zjzIhr3lk1gZB1OC56h/1vEtaXkESrOqL5fQVMfXpwGtRwX+YsZBdyHtJMQnkArw==
|
integrity sha512-umqzocjDgNRGTuO7Q8CU32dkHkECqI8ZdMZ5Swb6QAM0t5rnlrN3lGo1hdpscRd3WS8T6DKYK4ephgIH9iRh3w==
|
||||||
dependencies:
|
dependencies:
|
||||||
"@floating-ui/core" "^1.0.0"
|
"@floating-ui/core" "^1.6.0"
|
||||||
"@floating-ui/utils" "^0.2.0"
|
"@floating-ui/utils" "^0.2.9"
|
||||||
|
|
||||||
"@floating-ui/utils@^0.2.0", "@floating-ui/utils@^0.2.1":
|
"@floating-ui/utils@^0.2.9":
|
||||||
version "0.2.1"
|
version "0.2.9"
|
||||||
resolved "https://registry.yarnpkg.com/@floating-ui/utils/-/utils-0.2.1.tgz#16308cea045f0fc777b6ff20a9f25474dd8293d2"
|
resolved "https://registry.yarnpkg.com/@floating-ui/utils/-/utils-0.2.9.tgz#50dea3616bc8191fb8e112283b49eaff03e78429"
|
||||||
integrity sha512-9TANp6GPoMtYzQdt54kfAyMmz1+osLlXdg2ENroU7zzrtflTLrrC/lgrIfaSe+Wu0b89GKccT7vxXA0MoAIO+Q==
|
integrity sha512-MDWhGtE+eHw5JW7lq4qhc5yRLS11ERl1c7Z6Xd0a58DozHES6EnNNwUWbMiG4J9Cgj053Bhk8zvlhFYKVhULwg==
|
||||||
|
|
||||||
"@form-create/component-elm-checkbox@^3.1.29":
|
"@form-create/component-elm-checkbox@^3.1.29":
|
||||||
version "3.1.29"
|
version "3.1.29"
|
||||||
@@ -1901,9 +1901,9 @@
|
|||||||
"@types/lodash" "*"
|
"@types/lodash" "*"
|
||||||
|
|
||||||
"@types/lodash@*", "@types/lodash@^4.14.182":
|
"@types/lodash@*", "@types/lodash@^4.14.182":
|
||||||
version "4.17.0"
|
version "4.17.16"
|
||||||
resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.17.0.tgz#d774355e41f372d5350a4d0714abb48194a489c3"
|
resolved "https://registry.yarnpkg.com/@types/lodash/-/lodash-4.17.16.tgz#94ae78fab4a38d73086e962d0b65c30d816bfb0a"
|
||||||
integrity sha512-t7dhREVv6dbNj0q17X12j7yDG4bD/DHYX7o5/DbDxobP0HnGPgpRz2Ej77aL7TZT3DSw13fqUTj8J4mMnqa7WA==
|
integrity sha512-HX7Em5NYQAXKW+1T+FiuG27NGwzJfCX3s1GjOa7ujxZa52kjJLOr4FUxT+giF6Tgxv1e+/czV/iTtBw27WTU9g==
|
||||||
|
|
||||||
"@types/minimist@^1.2.0", "@types/minimist@^1.2.2":
|
"@types/minimist@^1.2.0", "@types/minimist@^1.2.2":
|
||||||
version "1.2.5"
|
version "1.2.5"
|
||||||
@@ -3362,6 +3362,11 @@ core-js@^3.31.1:
|
|||||||
resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.36.1.tgz#c97a7160ebd00b2de19e62f4bbd3406ab720e578"
|
resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.36.1.tgz#c97a7160ebd00b2de19e62f4bbd3406ab720e578"
|
||||||
integrity sha512-BTvUrwxVBezj5SZ3f10ImnX2oRByMxql3EimVqMysepbC9EeMUOpLwdy6Eoili2x6E4kf+ZUB5k/+Jv55alPfA==
|
integrity sha512-BTvUrwxVBezj5SZ3f10ImnX2oRByMxql3EimVqMysepbC9EeMUOpLwdy6Eoili2x6E4kf+ZUB5k/+Jv55alPfA==
|
||||||
|
|
||||||
|
core-js@^3.6.5:
|
||||||
|
version "3.40.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.40.0.tgz#2773f6b06877d8eda102fc42f828176437062476"
|
||||||
|
integrity sha512-7vsMc/Lty6AGnn7uFpYT56QesI5D2Y/UkgKounk87OP9Z2H9Z8kj6jzcSGAxFmUtDOS0ntK6lbQz+Nsa0Jj6mQ==
|
||||||
|
|
||||||
cors@^2.8.5:
|
cors@^2.8.5:
|
||||||
version "2.8.5"
|
version "2.8.5"
|
||||||
resolved "https://registry.yarnpkg.com/cors/-/cors-2.8.5.tgz#eac11da51592dd86b9f06f6e7ac293b3df875d29"
|
resolved "https://registry.yarnpkg.com/cors/-/cors-2.8.5.tgz#eac11da51592dd86b9f06f6e7ac293b3df875d29"
|
||||||
@@ -3515,7 +3520,12 @@ data-view-byte-offset@^1.0.0:
|
|||||||
es-errors "^1.3.0"
|
es-errors "^1.3.0"
|
||||||
is-data-view "^1.0.1"
|
is-data-view "^1.0.1"
|
||||||
|
|
||||||
dayjs@^1.11.3, dayjs@^1.11.7:
|
dayjs@^1.11.13:
|
||||||
|
version "1.11.13"
|
||||||
|
resolved "https://registry.yarnpkg.com/dayjs/-/dayjs-1.11.13.tgz#92430b0139055c3ebb60150aa13e860a4b5a366c"
|
||||||
|
integrity sha512-oaMBel6gjolK862uaPQOVTA7q3TZhuSvuMQAAglQDOWYO9A91IrAOUJEyKVlqJlHE0vq5p5UXxzdPfMH/x6xNg==
|
||||||
|
|
||||||
|
dayjs@^1.11.7:
|
||||||
version "1.11.10"
|
version "1.11.10"
|
||||||
resolved "https://registry.yarnpkg.com/dayjs/-/dayjs-1.11.10.tgz#68acea85317a6e164457d6d6947564029a6a16a0"
|
resolved "https://registry.yarnpkg.com/dayjs/-/dayjs-1.11.10.tgz#68acea85317a6e164457d6d6947564029a6a16a0"
|
||||||
integrity sha512-vjAczensTgRcqDERK0SR2XMwsF/tSvnvlv6VcF2GIhg6Sx4yOIt/irsr1RDJsKiIyBzJDpCoXiWWq28MqH2cnQ==
|
integrity sha512-vjAczensTgRcqDERK0SR2XMwsF/tSvnvlv6VcF2GIhg6Sx4yOIt/irsr1RDJsKiIyBzJDpCoXiWWq28MqH2cnQ==
|
||||||
@@ -3827,20 +3837,20 @@ electron-to-chromium@^1.4.668:
|
|||||||
resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.736.tgz#ecb4348f4d5c70fb1e31c347e5bad6b751066416"
|
resolved "https://registry.yarnpkg.com/electron-to-chromium/-/electron-to-chromium-1.4.736.tgz#ecb4348f4d5c70fb1e31c347e5bad6b751066416"
|
||||||
integrity sha512-Rer6wc3ynLelKNM4lOCg7/zPQj8tPOCB2hzD32PX9wd3hgRRi9MxEbmkFCokzcEhRVMiOVLjnL9ig9cefJ+6+Q==
|
integrity sha512-Rer6wc3ynLelKNM4lOCg7/zPQj8tPOCB2hzD32PX9wd3hgRRi9MxEbmkFCokzcEhRVMiOVLjnL9ig9cefJ+6+Q==
|
||||||
|
|
||||||
element-plus@2.3.4:
|
element-plus@2.9.4:
|
||||||
version "2.3.4"
|
version "2.9.4"
|
||||||
resolved "https://registry.yarnpkg.com/element-plus/-/element-plus-2.3.4.tgz#4319f6b0a1096cfac9c53b69da2fe23ac5aea63e"
|
resolved "https://registry.yarnpkg.com/element-plus/-/element-plus-2.9.4.tgz#e4b75055f0f1b1c1f33186321eb3d7a6424051e5"
|
||||||
integrity sha512-SQr0J9z7N4z48WYk/l9NE2tizl8Q7j2OhqlpTc42k4pGncry3+rVX6dsmcsglFynn6vt3NzYxWJqmLFyDKQq+g==
|
integrity sha512-sGnW0wd9zf6lEGixXV2gfwx3X6VTMkP52qTkX7zbURJ2oariyslrKTBh2txt1sdn1pUvj2l0KY3OfSXoZGmDOw==
|
||||||
dependencies:
|
dependencies:
|
||||||
"@ctrl/tinycolor" "^3.4.1"
|
"@ctrl/tinycolor" "^3.4.1"
|
||||||
"@element-plus/icons-vue" "^2.0.6"
|
"@element-plus/icons-vue" "^2.3.1"
|
||||||
"@floating-ui/dom" "^1.0.1"
|
"@floating-ui/dom" "^1.0.1"
|
||||||
"@popperjs/core" "npm:@sxzz/popperjs-es@^2.11.7"
|
"@popperjs/core" "npm:@sxzz/popperjs-es@^2.11.7"
|
||||||
"@types/lodash" "^4.14.182"
|
"@types/lodash" "^4.14.182"
|
||||||
"@types/lodash-es" "^4.17.6"
|
"@types/lodash-es" "^4.17.6"
|
||||||
"@vueuse/core" "^9.1.0"
|
"@vueuse/core" "^9.1.0"
|
||||||
async-validator "^4.2.5"
|
async-validator "^4.2.5"
|
||||||
dayjs "^1.11.3"
|
dayjs "^1.11.13"
|
||||||
escape-html "^1.0.3"
|
escape-html "^1.0.3"
|
||||||
lodash "^4.17.21"
|
lodash "^4.17.21"
|
||||||
lodash-es "^4.17.21"
|
lodash-es "^4.17.21"
|
||||||
@@ -8149,7 +8159,12 @@ vue-amap@^0.5.10:
|
|||||||
dependencies:
|
dependencies:
|
||||||
uppercamelcase "^1.1.0"
|
uppercamelcase "^1.1.0"
|
||||||
|
|
||||||
vue-demi@*, vue-demi@>=0.14.5, vue-demi@>=0.14.7:
|
vue-demi@*:
|
||||||
|
version "0.14.10"
|
||||||
|
resolved "https://registry.yarnpkg.com/vue-demi/-/vue-demi-0.14.10.tgz#afc78de3d6f9e11bf78c55e8510ee12814522f04"
|
||||||
|
integrity sha512-nMZBOwuzabUO0nLgIcc6rycZEebF6eeUfaiQx9+WSk8e29IbLvPU9feI6tqW4kTo3hvoYAJkMh8n8D0fuISphg==
|
||||||
|
|
||||||
|
vue-demi@>=0.14.5, vue-demi@>=0.14.7:
|
||||||
version "0.14.7"
|
version "0.14.7"
|
||||||
resolved "https://registry.yarnpkg.com/vue-demi/-/vue-demi-0.14.7.tgz#8317536b3ef74c5b09f268f7782e70194567d8f2"
|
resolved "https://registry.yarnpkg.com/vue-demi/-/vue-demi-0.14.7.tgz#8317536b3ef74c5b09f268f7782e70194567d8f2"
|
||||||
integrity sha512-EOG8KXDQNwkJILkx/gPcoL/7vH+hORoBaKgGe+6W7VFMvCYJfmF2dGbvgDroVnI8LU7/kTu8mbjRZGBU1z9NTA==
|
integrity sha512-EOG8KXDQNwkJILkx/gPcoL/7vH+hORoBaKgGe+6W7VFMvCYJfmF2dGbvgDroVnI8LU7/kTu8mbjRZGBU1z9NTA==
|
||||||
@@ -8215,6 +8230,13 @@ vue-types@^5.0.3:
|
|||||||
dependencies:
|
dependencies:
|
||||||
is-plain-object "5.0.0"
|
is-plain-object "5.0.0"
|
||||||
|
|
||||||
|
vue3-tree-org@^4.2.2:
|
||||||
|
version "4.2.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/vue3-tree-org/-/vue3-tree-org-4.2.2.tgz#6b711dc6e7c5aac99d9b7ff169a432435080cb4b"
|
||||||
|
integrity sha512-AG2SykyD6dw0jIyqBm8iuF9j9GWli6KrwudxR1RjULCCBTDFsoNm7MmP/weKT7wowN/sPk+e2RsnvEJMw2OJMw==
|
||||||
|
dependencies:
|
||||||
|
core-js "^3.6.5"
|
||||||
|
|
||||||
vue@3.3.4:
|
vue@3.3.4:
|
||||||
version "3.3.4"
|
version "3.3.4"
|
||||||
resolved "https://registry.yarnpkg.com/vue/-/vue-3.3.4.tgz#8ed945d3873667df1d0fcf3b2463ada028f88bd6"
|
resolved "https://registry.yarnpkg.com/vue/-/vue-3.3.4.tgz#8ed945d3873667df1d0fcf3b2463ada028f88bd6"
|
||||||
|
|||||||