Files
cdTestPlant3/cdTMP/src/views/project/design-demand/index.vue

356 lines
12 KiB
Vue
Raw Normal View History

2023-06-15 20:13:46 +08:00
<template>
<div class="ma-content-block lg:flex justify-between p-4">
<div class="lg:w-full w-full lg:ml-4 mt-5 lg:mt-0">
<!-- CRUD组件 -->
2023-08-24 19:24:00 +08:00
<ma-crud :options="crudOptions" :columns="crudColumns" ref="crudRef">
2023-08-23 20:45:44 +08:00
<template #ident="{ record }">
{{ showType(record) }}
</template>
2024-03-05 16:28:47 +08:00
<!-- 表格前置扩展槽:添加关联按钮 -->
<template #tableAfterButtons>
<a-button type="outline" status="warning" @click="handleOpenRelationCSX">
<template #icon>
<icon-tags />
</template>
关联测试项
</a-button>
</template>
2024-07-22 18:57:12 +08:00
<!-- 版本字段的插槽 -->
<template #inputPrepend-ident> XQ_XX_ </template>
2023-08-23 20:45:44 +08:00
</ma-crud>
2023-06-15 20:13:46 +08:00
</div>
2024-03-05 16:28:47 +08:00
<!-- 关联的modal组件 -->
<a-modal v-model:visible="visible" width="700px" draggable :on-before-ok="handleRelatedOk">
<template #title>关联测试项</template>
<div class="pb-3">已存在的关联项:</div>
<a-typography-paragraph>
<ol class="ol-reset">
<li v-for="item in computedRelatedData">{{ item }}</li>
</ol>
</a-typography-paragraph>
<div class="pb-3">选择关联的测试需求项:</div>
<a-cascader
:options="options"
multiple
allow-search
placeholder="暂无关联测试项,请选择..."
:loading="cascaderLoading"
v-model:model-value="relatedData"
/>
</a-modal>
2023-06-15 20:13:46 +08:00
</div>
</template>
<script setup>
2024-03-05 16:28:47 +08:00
import { ref, computed } from "vue"
2024-07-22 18:57:12 +08:00
import { useRoute } from "vue-router"
2023-06-15 20:13:46 +08:00
import testDemandApi from "@/api/project/testDemand"
2023-08-15 17:15:52 +08:00
import { useTreeDataStore } from "@/store"
2023-08-23 20:45:44 +08:00
import commonApi from "@/api/common"
2023-08-24 17:05:21 +08:00
import PinYinMatch from "pinyin-match"
2024-03-05 16:28:47 +08:00
import { Message } from "@arco-design/web-vue"
2023-08-24 17:05:21 +08:00
2023-08-15 17:15:52 +08:00
const treeDataStore = useTreeDataStore()
2023-06-15 20:13:46 +08:00
const route = useRoute()
2023-08-24 19:24:00 +08:00
const crudRef = ref()
2023-06-15 20:13:46 +08:00
// 根据传参获取key分别为轮次、设计需求的key
const roundNumber = route.query.key.split("-")[0]
2023-08-02 13:38:09 +08:00
const dutNumber = route.query.key.split("-")[1]
2023-06-19 19:51:12 +08:00
const designDemandNumber = route.query.key.split("-")[2]
2023-08-15 17:15:52 +08:00
const projectId = ref(route.query.id)
2024-03-05 16:28:47 +08:00
// ~~~~~关联相关变量和函数~~~~~
// 定义关联弹窗变量函数
const visible = ref(false)
const relatedData = ref([])
const computedRelatedData = computed(() => {
const labelResultList = []
options.value.forEach((item) => {
if (item.children) {
item.children.forEach((child) => {
if (relatedData.value.includes(child.value)) {
labelResultList.push(child.label)
}
})
}
})
return labelResultList
})
// 定义cascader的加载圈
const cascaderLoading = ref(false)
// 点击关联测试项-button
const handleOpenRelationCSX = async () => {
// 请求接口获取数据
cascaderLoading.value = true
visible.value = true
// 点击进入时清除关联
relatedData.value = []
const res = await testDemandApi.getRelatedTestDemand({ id: projectId.value, round: roundNumber })
options.value = res.data
// 找出本设计需求design对应已关联的测试项
const res_exist = await testDemandApi.getExistRelatedTestDemand({
project_id: projectId.value,
roundNumber,
dutNumber,
designDemandNumber
})
relatedData.value = res_exist.data
cascaderLoading.value = false
}
// 点击关联确定按钮
const handleRelatedOk = async () => {
// 获取级联数据
const relationDestItemIds = relatedData.value
if (relationDestItemIds.length > 0) {
const res = await testDemandApi.solveRelatedTestDemand({
data: relationDestItemIds,
project_id: projectId.value,
roundNumber,
dutNumber,
designDemandNumber
})
if (res.code == 200) {
Message.success(res.message)
return true
}
} else {
const res = await testDemandApi.solveRelatedTestDemand({
data: [],
project_id: projectId.value,
roundNumber,
dutNumber,
designDemandNumber
})
if (res.code == 200) {
Message.success(res.message)
return true
}
}
return false
}
// 级联cascade组件options
const options = ref([])
// ~~~~~~~~~~~~~~~~~~~~~~~~~~~
2023-08-23 20:45:44 +08:00
// 标识显示字段
const testTypeDict = ref([])
!(function () {
commonApi.getDict("testType").then((res) => {
testTypeDict.value = res
})
})()
const showType = (record) => {
let len = testTypeDict.value.data.length
for (let i = 0; i < len; i++) {
if (testTypeDict.value.data[i].key === record.testType) {
let item = testTypeDict.value.data[i]
2024-05-11 18:11:56 +08:00
return "XQ-" + item.show_title + "-" + record.ident
2023-08-23 20:45:44 +08:00
}
}
}
2023-06-15 20:13:46 +08:00
// crud组件
const crudOptions = ref({
api: testDemandApi.getTestDemandList,
2024-04-19 18:53:52 +08:00
add: { show: true, api: testDemandApi.save, text: "新增测试项" },
2024-04-26 19:06:14 +08:00
edit: { show: true, api: testDemandApi.update, text: "修改测试项" },
2023-08-24 17:05:21 +08:00
delete: { show: true, api: testDemandApi.delete },
2024-05-11 18:11:56 +08:00
showTools: false,
2023-08-24 19:24:00 +08:00
beforeOpenAdd: function () {
2024-03-05 16:28:47 +08:00
let key_split = route.query.key.split("-")
2023-08-25 13:28:24 +08:00
let round_key = key_split[0]
let dut_key = key_split[1]
let design_key = key_split[2]
2023-08-24 19:24:00 +08:00
let td = treeDataStore.treeData
crudRef.value.crudFormRef.actionTitle = `${route.query.ident} > ${td[round_key].title} > ${td[round_key].children[dut_key].title} > ${td[round_key].children[dut_key].children[design_key].title} > 测试项-`
return true
},
beforeOpenEdit: function (record) {
2024-03-05 16:28:47 +08:00
let key_split = route.query.key.split("-")
2023-08-25 13:28:24 +08:00
let round_key = key_split[0]
let dut_key = key_split[1]
let design_key = key_split[2]
2023-08-24 19:24:00 +08:00
let td = treeDataStore.treeData
crudRef.value.crudFormRef.actionTitle = `${route.query.ident} > ${td[round_key].title} > ${td[round_key].children[dut_key].title} > ${td[round_key].children[dut_key].children[design_key].title} >测试项[${record.name}]-`
return true
},
2023-08-15 17:15:52 +08:00
afterAdd: (res) => {
let id = projectId.value
treeDataStore.updateTestDemandTreeData(res.data, id)
},
afterEdit: (res) => {
let id = projectId.value
treeDataStore.updateTestDemandTreeData(res.data, id)
},
afterDelete: (res, record) => {
let id = projectId.value
2024-05-11 18:11:56 +08:00
if (!record) {
record = { key: route.query.key + "-X" }
}
2023-08-15 17:15:52 +08:00
treeDataStore.updateTestDemandTreeData(record, id)
},
2023-06-15 20:13:46 +08:00
parameters: {
projectId: route.query.id,
round: roundNumber,
2023-06-19 19:51:12 +08:00
dut: dutNumber,
2023-06-15 20:13:46 +08:00
designDemand: designDemandNumber
},
showIndex: false,
rowSelection: { showCheckedAll: true },
searchColNumber: 3,
2023-07-25 20:03:06 +08:00
tablePagination: false,
2023-06-15 20:13:46 +08:00
operationColumn: true,
2024-07-22 18:57:12 +08:00
operationColumnAlign:'center',
2023-06-15 20:13:46 +08:00
formOption: {
width: 1200
}
})
const crudColumns = ref([
{
title: "ID",
align: "center",
2024-07-22 18:57:12 +08:00
hide: true,
2023-06-15 20:13:46 +08:00
dataIndex: "id"
},
{
2024-05-11 18:11:56 +08:00
title: "测项标识",
2023-08-24 17:05:21 +08:00
width: 150,
2023-06-15 20:13:46 +08:00
dataIndex: "ident",
2023-08-21 19:57:49 +08:00
sortable: { sortDirections: ["ascend"] },
2023-06-15 20:13:46 +08:00
align: "center",
search: true,
2024-05-11 18:11:56 +08:00
validateTrigger: "blur",
placeholder: "请填写测试项的标识,注意标识不能重复",
2024-07-22 18:57:12 +08:00
commonRules: [{ required: true, message: "测试项标识必填" }],
openPrepend: true
2023-06-15 20:13:46 +08:00
},
{
title: "名称",
dataIndex: "name",
width: 120,
align: "center",
search: true,
commonRules: [{ required: true, message: "名称是必填" }],
validateTrigger: "blur"
},
{
title: "优先级",
dataIndex: "priority",
search: true,
formType: "radio",
align: "center",
addDefaultValue: "1",
dict: {
name: "priority",
props: { label: "title", value: "key" },
translation: true,
tagColors: { 1: "red", 2: "blue", 3: "green" }
}
},
{
title: "测试类型",
dataIndex: "testType",
search: true,
align: "center",
formType: "select",
sortable: { sortDirections: ["ascend", "descend"] },
2024-06-13 19:41:57 +08:00
addDefaultValue: "4",
2023-06-15 20:13:46 +08:00
maxLength: 200,
commonRules: [{ required: true, message: "测试类型必选" }],
dict: { name: "testType", translation: true, props: { label: "title", value: "key" } },
2023-08-24 17:05:21 +08:00
extra: "请保证测试类型选择正确",
2024-06-13 19:41:57 +08:00
// 这是arco的属性所以在ma-crud和ma-form可以直接使用arco属性和事件事件+onXXX
2023-08-24 17:05:21 +08:00
filterOption: function (inputValue, selectedOption) {
2023-08-24 19:24:00 +08:00
if (inputValue) {
let matchRes = PinYinMatch.match(selectedOption.label, inputValue)
if (matchRes) {
2023-08-24 17:05:21 +08:00
return true
}
}
2024-07-22 18:57:12 +08:00
}
2023-06-15 20:13:46 +08:00
},
{
2024-05-11 18:11:56 +08:00
title: "测试手段",
align: "center",
dataIndex: "testMethod",
formType: "select",
multiple: true,
dict: { name: "testMethod", props: { label: "title", value: "key" }, translation: true }
},
{
title: "充分性要求",
2023-06-15 20:13:46 +08:00
hide: true,
2023-06-19 19:51:12 +08:00
addDefaultValue: "覆盖需求相关功能",
2023-06-15 20:13:46 +08:00
dataIndex: "adequacy",
2023-08-24 17:05:21 +08:00
formType: "textarea",
maxLength: 256,
2024-05-11 18:11:56 +08:00
commonRules: [{ required: true, message: "充分性描述必填" }],
2023-06-15 20:13:46 +08:00
addDefaultValue:
2024-05-11 18:11:56 +08:00
"测试用例覆盖XX子项名称1、XX子项名称2、XX子项名称3子项要求的全部内容。\n所有用例执行完毕对于未执行的用例说明未执行原因。"
2023-06-15 20:13:46 +08:00
},
{
2024-05-11 18:11:56 +08:00
title: "测试子项",
2023-06-15 20:13:46 +08:00
hide: true,
2023-06-19 19:51:12 +08:00
dataIndex: "testContent",
2024-05-11 18:11:56 +08:00
commonRules: [{ required: true, message: "测试方法是必填的" }],
2023-06-15 20:13:46 +08:00
formType: "children-form",
formList: [
{
2024-05-11 18:11:56 +08:00
title: "子项名称",
dataIndex: "subName",
placeholder: "对应测试项描述标题,和测试方法的标题",
rules: [{ required: true, message: "测试子项名称必填" }],
onChange: (ev) => {
// 取出子项的对象数组
const subItemFormData = crudRef.value.getFormData().testContent
// 取出充分性条件字段字符串
const mapRes = subItemFormData.map((subItem) => subItem.subName)
2024-06-13 19:41:57 +08:00
crudRef.value.getFormData().adequacy = `测试用例覆盖${mapRes.join(
"、"
)}子项要求的全部内容\n所有用例执行完毕对于未执行的用例说明未执行原因`
2024-05-11 18:11:56 +08:00
}
2023-06-15 20:13:46 +08:00
},
{
2024-05-11 18:11:56 +08:00
title: "子项描述",
dataIndex: "subDesc",
placeholder: "对应大纲测试项表格的测试项描述",
rules: [{ required: true, message: "测试子项描述必填" }]
},
{
title: "条件",
dataIndex: "condition",
placeholder: "在什么环境和前置条件下"
},
{
title: "操作",
dataIndex: "operation",
placeholder: "通过xxx操作"
},
{
title: "观察",
dataIndex: "observe",
placeholder: "查看xxx内容"
},
{
title: "期望",
dataIndex: "expect",
placeholder: "xxx结果正确"
2023-06-15 20:13:46 +08:00
}
]
}
])
2024-05-11 18:11:56 +08:00
// 暴露给route-view的刷新表格函数
const refreshCrudTable = () => {
crudRef.value.refresh()
}
defineExpose({ refreshCrudTable })
2024-07-22 18:57:12 +08:00
defineOptions({
name: "designDemand"
})
2023-06-15 20:13:46 +08:00
</script>
2024-03-05 16:28:47 +08:00
<style lang="less" scoped>
.ol-reset {
list-style: auto;
}
</style>