<!-- 任务单管理列表 -->
<script lang="ts" setup name="OrderList">
import type { Ref } from 'vue'
import { getCurrentInstance, ref } from 'vue'
import { ElLoading, ElMessage, ElMessageBox } from 'element-plus'
import type { DateModelType } from 'element-plus'
import dayjs from 'dayjs'
import backReasonDialog from './dialog/backReasonDialog.vue'
import type { IList, IListQuery } from './order-interface'
import selectFieldTestApprovalDialog from './dialog/selectFieldTestApprovalDialog.vue'
import type { TableColumn } from '@/components/NormalTable/table_interface'
import { usePrintList } from '@/commonMethods/usePrintList'
import { exportFile } from '@/utils/exportUtils'
import type { dictType } from '@/global'
import { getDictByCode } from '@/api/system/dict'
import { uploadApi } from '@/api/system/notice'
import {
bindFieldTestApproval,
exportOrderList,
getOrderDetail,
getOrderList,
receiveOrder,
updateOrder,
} from '@/api/business/manager/order'
import { getUserList } from '@/api/system/user'
import useUserStore from '@/store/modules/user'
const user = useUserStore() // 用户信息
const $router = useRouter()
const operateWidth = ref('160') // 操作栏的宽度
// 查询条件
const listQuery: Ref<IListQuery> = ref({
orderNo: '', // 任务单编号
customerId: '', // 委托方id
customerName: '', // 委托方名称
createUserName: '', // 创建人
deliverer: '', // 送样人
createStartTime: '', // 创建开始时间
createEndTime: '', // 创建结束时间
receiveStatus: '', // 接收状态
dataSource: '', // 任务单来源
isOnSiteCheck: '', // 是否现场检定(1/0)
measureCompany: '', // 检定(校准)单位(字典value,西昌实验室/海口实验室)
offset: 1,
limit: 20,
})
const backReasonDialogRef = ref() // 退回弹出框ref
const columns = ref<TableColumn[]>([ // 表头
{ text: '任务单编号', value: 'orderNo', align: 'center', width: '160' },
{ text: '委托方', value: 'customerName', align: 'center' },
{ text: '受检设备数量', value: 'sampleCount', align: 'center' },
{ text: '创建人', value: 'createUserName', align: 'center' },
{ text: '创建时间', value: 'createTime', align: 'center', width: '180' },
{ text: '送检人', value: 'deliverer', align: 'center' },
{ text: '接收状态', value: 'receiveStatusName', align: 'center', width: '100' },
{ text: '是否现场检定', value: 'isOnSiteCheckValue', align: 'center', width: '100' },
{ text: '任务单来源', value: 'dataSource', align: 'center', width: '120' },
{ text: '有效期', value: 'requireOverTime', align: 'center', width: '120' },
])
const list = ref<IList[]>([]) // 表格数据
const total = ref(0) // 数据总数
const loadingTable = ref(false) // 表格加载状态
const checkoutIdList = ref<IList[]>([]) // 选中的内容
const checkoutList = ref<IList[]>([]) // 选中的内容
const dateRange = ref<[DateModelType, DateModelType]>(['', '']) // 筛选时间段数据
// ------------------------------------------字典----------------------------------------------
const receiveStatusList = ref<dictType[]>([])// 任务单接收状态
const isOnSiteCheckList = ref<dictType[]>([])// 是否现场检定
const dataSourceList = ref<dictType[]>([])// 任务单来源
const userList = ref<{ [key: string]: string }[]>([]) // 用户列表
async function getDict() {
// 获取用户列表
const res = await getUserList({ offset: 1, limit: 999999 })
userList.value = res.data.rows
// 任务单接收状态
getDictByCode('orderStatus').then((response) => {
receiveStatusList.value = response.data
})
// 是否现场检定
isOnSiteCheckList.value = [
{
id: '1',
name: '是',
value: '1',
},
{
id: '0',
name: '否',
value: '0',
},
]
// 任务单来源
dataSourceList.value = [
{
id: '1',
name: '计量业务系统',
value: '1',
},
{
id: '2',
name: '受检设备系统',
value: '2',
},
{
id: '3',
name: '自动检定系统',
value: '3',
},
]
}
getDict()
// ---------------------------------------------------------------------------------------------
// -------------------------------------------文件上传\表格右上角几个图标方法-----------------------------------------------
const fileRef = ref() // 文件上传input
// 上传文件/批量导入
const onFileChange = (event: any) => {
// todo: 校验文件类型,调用上传接口
if (event.target.files[0].type === 'application/pdf') {
if (event.target.files?.length !== 0) {
// 创建formdata对象
const fd = new FormData()
fd.append('multipartFile', event.target.files[0])
uploadApi(fd).then((res) => {
if (res.code === 200) {
ElMessage.success('上传成功')
}
else {
ElMessage.error(res.message)
}
})
}
}
else {
ElMessage.error('请上传pdf格式')
}
}
// 点击批量导入
const uploadAll = () => {
fileRef.value.click()
}
// 导出
const exportAll = () => {
const loading = ElLoading.service({
lock: true,
text: '下载中请稍后',
background: 'rgba(255, 255, 255, 0.8)',
})
if (list.value.length > 0) {
const params = {
orderNo: listQuery.value.orderNo, // 任务单编号
customerId: listQuery.value.customerId, // 委托方id
customerName: listQuery.value.customerName, // 委托方名称
createUserName: listQuery.value.createUserName, // 创建人
deliverer: listQuery.value.deliverer, // 送样人
createStartTime: listQuery.value.createStartTime, // 创建开始时间
createEndTime: listQuery.value.createEndTime, // 创建结束时间
receiveStatus: listQuery.value.receiveStatus, // 接收状态
dataSource: listQuery.value.dataSource, // 任务单来源
isOnSiteCheck: listQuery.value.isOnSiteCheck, // 是否现场检定(1/0)
measureCompany: listQuery.value.measureCompany, // 检定(校准)单位(字典value,西昌实验室/海口实验室)
offset: 1,
limit: 20,
ids: checkoutIdList.value,
}
exportOrderList(params).then((res) => {
const blob = new Blob([res.data])
loading.close()
exportFile(blob, '任务单列表.xlsx')
})
}
else {
loading.close()
ElMessage.warning('无数据可导出数据')
}
}
// 打印列表
function printList() {
usePrintList(list.value, columns.value, checkoutIdList.value, '任务单列表')
}
// 模板下载
const templateDownload = () => {}
// 点击新建
const add = () => {
$router.push('/manager/add')
}
// -------------------------------------------------------------------------------------------------
// 数据查询
function fetchData(isNowPage = false) {
loadingTable.value = true
if (!isNowPage) {
// 是否显示当前页,否则跳转第一页
listQuery.value.offset = 1
}
listQuery.value.measureCompany = user.bizLabCode === 'X' ? '西昌实验室' : user.bizLabCode === 'H' ? '海口实验室' : ''
getOrderList(listQuery.value).then((response) => {
list.value = response.data.rows.map((item: { deliverer: string; delivererName: string; requireOverTime: string }) => {
const index = userList.value.findIndex(i => i.id === item.deliverer)
if (index !== -1) {
item.delivererName = userList.value[index].name
}
item.requireOverTime = item.requireOverTime ? dayjs(item.requireOverTime).format('YYYY-MM-DD') : item.requireOverTime // 有效期(要求检完时间)
return item
})
total.value = parseInt(response.data.total)
loadingTable.value = false
})
}
// 点击搜索
const searchList = () => {
fetchData(true)
}
// 点击重置
const clearList = () => {
listQuery.value = {
orderNo: '', // 任务单编号
customerId: '', // 委托方id
customerName: '', // 委托方名称
createUserName: '', // 创建人
deliverer: '', // 送样人
createStartTime: '', // 创建开始时间
createEndTime: '', // 创建结束时间
receiveStatus: '', // 接收状态
dataSource: '', // 任务单来源
isOnSiteCheck: '', // 是否现场检定(1/0)
measureCompany: '', // 检定(校准)单位(字典value,西昌实验室/海口实验室)
offset: 1,
limit: 20,
}
dateRange.value = ['', ''] // 时间清空
fetchData(true)
}
// 多选发生改变时
function handleSelectionChange(e: any) {
checkoutList.value = e
checkoutIdList.value = e.map((item: { id: string }) => item.id)
}
// 页数发生变化后的操作,可能是页码变化,可能是每页容量变化,此函数必写
const changePage = (val: { size?: number; page?: number }) => {
if (val && val.size) {
listQuery.value.limit = val.size
}
if (val && val.page) {
listQuery.value.offset = val.page
}
fetchData(true)
}
// --------------------------------------------操作------------------------------------------
// 退回成功
const backSuccess = () => {
fetchData(true)
}
// 操作
const handleEdit = (row: IList, type: string, title: string) => {
if (type === 'receive') { // 接收
ElMessageBox.confirm(
`确认${title}吗?`,
'提示',
{
confirmButtonText: '确认',
cancelButtonText: '取消',
type: 'warning',
},
)
.then(() => {
const loading = ElLoading.service({
lock: true,
background: 'rgba(255, 255, 255, 0.8)',
})
let form
receiveOrder({ id: row.id }).then(() => {
ElMessage.success(`已${title}`)
if (row.dataSource === '受检设备系统') { // 数据来源于受检系统
getOrderDetail({ id: row.id }).then((res) => {
form = res.data.data
updateOrder({
...form,
undertakerId: user.id, // 承接人id
undertakerName: user.name, // 承接人
undertakeTime: dayjs().format('YYYY-MM-DD HH-mm:ss'), // 承接时间
}).then(() => {
fetchData(true)
loading.close()
}).catch(() => {
fetchData(true)
loading.close()
})
})
}
else { // 数据来源于业务系统
loading.close()
fetchData(true)
}
})
})
}
else if (type === 'back') { // 退回
backReasonDialogRef.value.initDialog(row)
}
}
// 点击编辑详情
const goEdit = (row: IList, pageType: 'edit' | 'detail') => {
$router.push({
path: `/manager/${pageType}/${row.id}`,
// query: {
// statusName: row.receiveStatusName,
// },
})
}
// --------------------------------------------绑定现场审批单-------------------------------------------
const selectFieldTestApprovalRef = ref() // 选择现场测试、校准或检定审批 组件ref
const siteExecutiveId = ref() // 选择的现场审批单id
const siteExecutiveNo = ref() // 选择的现场审批单编号
const table = ref() // 表格组件ref
// 点击绑定现场审批单
const bind = () => {
if (!checkoutIdList.value.length) {
ElMessage.warning('请选择任务单')
return false
}
const index = checkoutList.value.findIndex(item => item.isOnSiteCheck === 0)
if (index !== -1) {
ElMessage.warning(`任务单【${checkoutList.value[0].orderNo}】非现场检定,不允许绑定`)
return false
}
selectFieldTestApprovalRef.value.initDialog()
}
// 绑定操作
const handleBind = (flag = 0) => {
console.log('checkoutIdList.value')
console.log(checkoutIdList.value)
const orderIds = checkoutIdList.value
const params = {
flag, // 确定解除原有绑定并进行本次绑定(1/0)
orderIds, // 任务单id列表
siteExecutiveId: siteExecutiveId.value, // 现场测试、校准或检定审批单id
siteExecutiveNo: siteExecutiveNo.value, // 现场测试、校准或检定审批单编号
}
const loading = ElLoading.service({
lock: true,
text: '加载中...',
background: 'rgba(255, 255, 255, 0.6)',
})
bindFieldTestApproval(params).then((res) => {
ElMessage.success('绑定成功')
table.value.clearMulti() // 清除多选
fetchData(true)
loading.close()
}).catch((res) => {
console.log(res.message)
loading.close()
ElMessageBox.confirm(
`${res.message}, 是否进行更新替换?`,
'提示',
{
confirmButtonText: '确认',
cancelButtonText: '取消',
type: 'warning',
},
)
.then(() => {
handleBind(1)
})
})
}
// 确定选择
const confirmSelectedFieldTestApproval = (val: any) => {
siteExecutiveId.value = val[0].id
siteExecutiveNo.value = val[0].siteExecutiveNo // 现场测试、校 准或检定审批单编号
handleBind()
}
// ----------------------------------------------钩子------------------------------------------------------
watch(dateRange, (val) => { // 监听创建时间改变
if (val) {
listQuery.value.createStartTime = `${val[0]}`
listQuery.value.createEndTime = `${val[1]}`
}
else {
listQuery.value.createStartTime = ''
listQuery.value.createEndTime = ''
}
})
onMounted(async () => {
getDict().then(() => {
fetchData(true) // 获取列表数据
})
})
</script>
<template>
<app-container>
<search-area :need-clear="true" @search="searchList" @clear="clearList">
<search-item>
<el-input
v-model.trim="listQuery.orderNo"
placeholder="任务单编号"
class="short-input"
clearable
/>
</search-item>
<search-item>
<el-input
v-model.trim="listQuery.customerName"
placeholder="委托方"
class="short-input"
clearable
/>
</search-item>
<search-item>
<el-input
v-model.trim="listQuery.createUserName"
placeholder="创建人"
class="short-input"
clearable
/>
</search-item>
<search-item>
<el-date-picker
v-model="dateRange"
type="datetimerange"
range-separator="至"
format="YYYY-MM-DD HH:mm:ss"
value-format="YYYY-MM-DD HH:mm:ss"
start-placeholder="创建时间(开始)"
end-placeholder="创建时间(结束)"
/>
</search-item>
<search-item>
<el-select
v-model="listQuery.receiveStatus"
class="short-input"
placeholder="接收状态"
filterable
clearable
>
<el-option v-for="item in receiveStatusList" :key="item.id" :label="item.name" :value="item.value" />
</el-select>
</search-item>
<search-item>
<el-select
v-model="listQuery.isOnSiteCheck"
class="short-input"
placeholder="是否现场检定"
filterable
clearable
>
<el-option v-for="item in isOnSiteCheckList" :key="item.id" :label="item.name" :value="item.value" />
</el-select>
</search-item>
<search-item>
<el-select
v-model="listQuery.dataSource"
class="short-input"
placeholder="任务单来源"
filterable
clearable
>
<el-option v-for="item in dataSourceList" :key="item.id" :label="item.name" :value="item.name" />
</el-select>
</search-item>
</search-area>
<table-container>
<template #btns-right>
<!-- <icon-button icon="icon-import" title="批量导入" type="primary" @click="uploadAll" /> -->
<!-- <icon-button icon="icon-template" title="模板下载" type="primary" @click="templateDownload" /> -->
<icon-button icon="icon-bind" title="绑定现场审批单" type="primary" @click="bind" />
<icon-button icon="icon-add" title="新建" type="primary" @click="add" />
<icon-button icon="icon-export" title="导出" type="primary" @click="exportAll" />
<icon-button icon="icon-print" title="打印" type="primary" @click="printList" />
</template>
<input v-show="false" ref="fileRef" type="file" accept="pdf/*" @change="onFileChange">
<normal-table
ref="table" :data="list" :total="total" :columns="columns" :query="listQuery"
:list-loading="loadingTable" is-showmulti-select @change="changePage" @multi-select="handleSelectionChange"
>
<template #preColumns>
<el-table-column label="序号" width="55" align="center">
<template #default="scope">
{{ (listQuery.offset - 1) * listQuery.limit + scope.$index + 1 }}
</template>
</el-table-column>
</template>
<template #columns>
<el-table-column label="操作" align="center" fixed="right" :width="operateWidth">
<template #default="{ row }">
<el-button
size="small"
link
type="primary"
@click="goEdit(row, 'detail')"
>
详情
</el-button>
<el-button
v-if="row.receiveStatusName !== '检测完成' && row.receiveStatusName !== '已取消' && row.receiveStatusName !== '已退回'"
size="small"
type="primary"
link
@click="goEdit(row, 'edit')"
>
编辑
</el-button>
<el-button
v-if="row.receiveStatusName !== '检测完成' && row.receiveStatusName !== '已取消' && row.receiveStatusName !== '已接收' && row.receiveStatusName !== '已退回'"
size="small"
link
type="primary"
@click="handleEdit(row, 'receive', '接收')"
>
接收
</el-button>
<el-button
v-if="row.receiveStatusName !== '检测完成' && row.receiveStatusName !== '已取消' && row.receiveStatusName !== '已接收' && row.receiveStatusName !== '已退回'"
size="small"
link
type="primary"
@click="handleEdit(row, 'back', '退回')"
>
退回
</el-button>
</template>
</el-table-column>
</template>
</normal-table>
</table-container>
<back-reason-dialog ref="backReasonDialogRef" @back-success="backSuccess" />
<!-- 选择现场测试、校准或检定审批 -->
<select-field-test-approval-dialog ref="selectFieldTestApprovalRef" @confirm="confirmSelectedFieldTestApproval" />
</app-container>
</template>
<style lang="scss" scoped>
// 样式
.short-input {
width: 180px !important;
}
</style>