Files
td_official/src/views/materials/usageMaterials/purchase/index.vue
2025-08-30 00:01:14 +08:00

486 lines
14 KiB
Vue

<template>
<div style="padding: 20px">
<el-card class="mb-5" shadow="never">
<el-row :gutter="10" class="mb8">
<el-col :span="1.5">
<el-button type="primary" plain icon="Plus" @click="handleAdd">新增</el-button>
</el-col>
<right-toolbar @queryTable="refreshData"></right-toolbar>
</el-row>
</el-card>
<!-- 数据表格 -->
<div class="bg-white rounded-lg shadow-sm overflow-hidden transition-all duration-300 hover:shadow-md">
<el-table
v-loading="loading"
:data="tableData"
border
stripe
style="width: 100%; margin-bottom: 20px; height: calc(100vh - 305px)"
:header-cell-style="{ 'background-color': '#f5f7fa', 'font-weight': 'bold' }"
:row-class-name="tableRowClassName"
>
<!-- <el-table-column prop="id" label="ID" width="180" align="center"></el-table-column> -->
<el-table-column type="index" label="序号" align="center" width="60"></el-table-column>
<el-table-column prop="name" label="材料名称" align="center"></el-table-column>
<el-table-column prop="specification" label="规格" align="center"></el-table-column>
<el-table-column prop="supplier" label="供应商" align="center"></el-table-column>
<el-table-column prop="findType" label="类型" align="center">
<template #default="scope">
<el-tag :type="scope.row.findType === '1' ? 'success' : 'info'">
{{ scope.row.findType === '1' ? '采购' : '材料' }}
</el-tag>
</template>
</el-table-column>
<el-table-column prop="installationQuantity" label="安装量" align="center"></el-table-column>
<el-table-column prop="contractSigning" label="合同签订时间" width="180" align="center">
<template #default="scope">
{{ formatDate(scope.row.contractSigning) }}
</template>
</el-table-column>
<el-table-column prop="createTime" label="创建时间" width="180" align="center">
<template #default="scope">
{{ formatDate(scope.row.createTime) }}
</template>
</el-table-column>
<el-table-column label="操作" width="300" align="center" fixed="right">
<template #default="scope">
<el-button
v-hasPermi="['cailiaoshebei:physicalsupplySon:add']"
size="small"
icon="Plus"
type="primary"
link
@click="handleAddSon(scope.row)"
>添加</el-button
>
<el-button type="primary" v-hasPermi="['cailiaoshebei:physicalsupply:edit']" size="small" icon="Edit" link @click="handleEdit(scope.row)"
>修改</el-button
>
<el-button
type="primary"
v-hasPermi="['cailiaoshebei:physicalsupplySon:list']"
size="small"
icon="View"
link
@click="jumpRouter(scope.row)"
>查看</el-button
>
<el-button
v-hasPermi="['cailiaoshebei:physicalsupply:remove']"
size="small"
icon="Delete"
link
type="primary"
@click="handleDelete(scope.row)"
>删除</el-button
>
</template>
</el-table-column>
</el-table>
<!-- 分页 -->
<pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize" @pagination="fetchData" />
</div>
<!-- 新增/编辑对话框 -->
<el-dialog
v-model="dialogVisible"
:title="dialogType === 'add' ? '新增记录' : '编辑记录'"
:width="dialogWidth"
:fullscreen="isFullscreen"
:before-close="handleDialogClose"
>
<el-form ref="formRef" :model="formData" :rules="formRules" label-width="120px" class="space-y-4">
<el-row :gutter="20">
<el-col :span="12">
<el-form-item label="材料名称" prop="name">
<el-input v-model="formData.name" placeholder="请输入材料名称"></el-input>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="规格" prop="specification">
<el-input v-model="formData.specification" placeholder="请输入规格"></el-input>
</el-form-item>
</el-col>
</el-row>
<el-row :gutter="20">
<el-col :span="12">
<el-form-item label="供应商" prop="supplier">
<el-input v-model="formData.supplier" placeholder="请输入供应商"></el-input>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="运算周期(天)" prop="executionCycle">
<el-input v-model.number="formData.executionCycle" placeholder="请输入运算周期" type="number"></el-input>
</el-form-item>
</el-col>
</el-row>
<el-row :gutter="20">
<el-col :span="12">
<el-form-item label="安装量" prop="installationQuantity">
<el-input v-model="formData.installationQuantity" placeholder="请输入安装量"></el-input>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="安装比例" prop="installationRatio">
<el-input v-model="formData.installationRatio" placeholder="请输入安装比例" suffix="%"></el-input>
</el-form-item>
</el-col>
</el-row>
<el-row :gutter="20">
<el-col :span="12">
<el-form-item label="合同签订时间" prop="contractSigning">
<el-date-picker
v-model="formData.contractSigning"
type="datetime"
placeholder="选择合同签订时间"
value-format="YYYY-MM-DD HH:mm:ss"
></el-date-picker>
</el-form-item>
</el-col>
<el-col :span="12">
<el-form-item label="生产周期(天)" prop="productionPhase">
<el-input v-model.number="formData.productionPhase" placeholder="请输入生产周期" type="number"></el-input>
</el-form-item>
</el-col>
</el-row>
<el-row :gutter="20">
<el-col :span="24">
<el-form-item label="供货要求" prop="supplyRequirements">
<el-input v-model="formData.supplyRequirements" placeholder="请输入供货要求" type="textarea" :rows="3"></el-input>
</el-form-item>
</el-col>
</el-row>
<el-row :gutter="20">
<el-col :span="24">
<el-form-item label="备注" prop="remark">
<el-input v-model="formData.remark" placeholder="请输入备注信息" type="textarea" :rows="3"></el-input>
</el-form-item>
</el-col>
</el-row>
</el-form>
<template #footer>
<div class="flex justify-end gap-2">
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="handleSave" :loading="saveLoading"> 保存 </el-button>
</div>
</template>
</el-dialog>
</div>
</template>
<script setup>
import { ref, reactive, onMounted, computed, toRaw } from 'vue';
import { ElMessage, ElMessageBox } from 'element-plus';
import { useUserStoreHook } from '@/store/modules/user';
import { useRouter } from 'vue-router';
const userStore = useUserStoreHook();
const router = useRouter();
const currentProject = computed(() => userStore.selectedProject);
import { useMaterialsQueryList, newMaterialsAdd, materialsEdit, materialsDel } from '@/api/materials/usageMaterials/index';
// 表格数据相关
const tableData = ref([]);
const total = ref(0);
const currentPage = ref(1);
const pageSize = ref(10);
const loading = ref(false);
const saveLoading = ref(false);
// 搜索表单
const searchForm = reactive({
findType: '3', // 默认查询所有
keyword: ''
});
// 对话框相关
const dialogVisible = ref(false);
const dialogType = ref('add'); // add 或 edit
const dialogWidth = ref('70%');
const isFullscreen = ref(false);
const deleteDialogVisible = ref(false);
const formRef = ref(null);
const deleteLoading = ref(false);
const currentRow = ref(null);
// 表单数据
const formData = reactive({
id: '',
name: '',
specification: '',
supplier: '',
findType: 2, // 默认采购
installationQuantity: '',
installationRatio: '',
contractSigning: '',
productionPhase: null,
executionCycle: null,
projectId: currentProject.value?.id,
supplyRequirements: '',
// purchaseSubmission: '',
// submissionMaterials: '',
remark: '',
createTime: '',
createBy: null,
updateTime: '',
updateBy: null
});
// 表单验证规则
const formRules = reactive({
name: [
{ required: true, message: '请输入材料名称', trigger: 'blur' },
{ max: 50, message: '材料名称不能超过50个字符', trigger: 'blur' }
],
supplier: [
{ required: true, message: '请输入供应商', trigger: 'blur' },
{ max: 100, message: '供应商名称不能超过100个字符', trigger: 'blur' }
],
findType: [{ required: true, message: '请选择类型', trigger: 'change' }]
});
const queryParams = reactive({
pageSize: 10,
pageNum: 1,
projectId: currentProject.value?.id
});
// 格式化日期
const formatDate = (dateString) => {
if (!dateString) return '-';
const date = new Date(dateString);
return date
.toLocaleString('zh-CN', {
year: 'numeric',
month: '2-digit',
day: '2-digit',
hour: '2-digit',
minute: '2-digit',
second: '2-digit'
})
.replace(',', ' ');
};
// 表格行样式
const tableRowClassName = ({ row, rowIndex }) => {
return rowIndex % 2 === 0 ? 'bg-white' : 'bg-gray-50';
};
// 获取数据
const fetchData = async () => {
loading.value = true;
try {
const res = await useMaterialsQueryList(queryParams);
tableData.value = res.rows;
total.value = res.total;
} catch (error) {
ElMessage.error('获取数据失败:' + error.message);
console.error(error);
} finally {
loading.value = false; //
}
};
// 搜索
const handleSearch = () => {
currentPage.value = 1; // 重置到第一页
fetchData();
};
// 刷新数据
const refreshData = () => {
fetchData();
ElMessage.success('数据已刷新');
};
// 分页大小改变
const handleSizeChange = (val) => {
pageSize.value = val;
currentPage.value = 1;
fetchData();
};
// 当前页改变
const handleCurrentChange = (val) => {
currentPage.value = val;
fetchData();
};
// 新增
const handleAdd = () => {
dialogType.value = 'add';
resetForm();
dialogVisible.value = true;
};
//
// 编辑
const handleEdit = (row) => {
dialogType.value = 'edit';
currentRow.value = row;
resetForm();
// 填充表单数据
Object.keys(formData).forEach((key) => {
if (row.hasOwnProperty(key)) {
formData[key] = row[key];
}
});
dialogVisible.value = true;
};
const handleAddSon = (row) => {
ElMessageBox.confirm('确认提交', '提示', {
confirmButtonText: '确认',
cancelButtonText: '取消',
type: 'warning'
})
.then(() => {
materialsEdit({ id: row.id, purchaseSubmission: '1' }).then((res) => {
let { code } = res;
if (code === 200) {
ElMessage({
type: 'success',
message: '提交成功'
});
}
});
})
.catch(() => {
ElMessage({
type: 'info',
message: '已取消提交'
});
});
};
// 删除
const handleDelete = (row) => {
currentRow.value = row;
ElMessageBox.confirm('确定要删除这条记录吗?此操作不可撤销,请谨慎操作', '提示', {
confirmButtonText: '确认',
cancelButtonText: '取消',
type: 'warning'
})
.then(() => {
confirmDelete();
})
.catch(() => {
ElMessage({
type: 'info',
message: '已取消删除'
});
});
};
// 确认删除
const confirmDelete = async () => {
if (!currentRow.value) return;
deleteLoading.value = true;
try {
// 模拟API请求
const res = await materialsDel(currentRow.value.id);
const { code } = res;
if (code === 200) {
ElMessage.success('删除成功');
deleteDialogVisible.value = false;
fetchData();
} else {
ElMessage.error('删除失败');
}
} catch (error) {
ElMessage.error('删除失败:' + error.message);
console.error(error);
} finally {
deleteLoading.value = false;
}
};
//
// 保存
const handleSave = async () => {
// 表单验证
if (!formRef.value) return;
const valid = await formRef.value.validate();
if (!valid) return;
saveLoading.value = true;
try {
// 模拟API请求
const form = toRaw(formData);
if (dialogType.value === 'add') {
// 新增
formData.projectId = currentProject.value?.id;
const res = await newMaterialsAdd(formData);
let { code } = res;
if (code === 200) {
ElMessage.success('新增成功');
fetchData();
}
} else {
// 编辑
const res = await materialsEdit(formData);
let { code } = res;
if (code === 200) {
ElMessage.success('保存成功');
fetchData();
}
}
dialogVisible.value = false;
} catch (error) {
ElMessage.error('保存失败:' + error.message);
console.error(error);
} finally {
saveLoading.value = false;
}
};
// 重置表单
const resetForm = () => {
if (formRef.value) {
formRef.value.resetFields();
}
// 重置表单数据
Object.keys(formData).forEach((key) => {
formData[key] = '';
});
// 设置默认值
formData.findType = 1;
formData.id = '';
};
// 关闭对话框
const handleDialogClose = () => {
resetForm();
dialogVisible.value = false;
};
// 跳转
const jumpRouter = (row) => {
router.push({
path: `/materials/usageMaterials/purchaseIndexSon`,
query: {
id: row.id,
type: 'update'
}
});
};
// 初始化页面
onMounted(() => {
fetchData();
});
</script>
<style scoped></style>