index.vue 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596
  1. <template>
  2. <div class="p-2">
  3. <transition :enter-active-class="proxy?.animate.searchAnimate.enter" :leave-active-class="proxy?.animate.searchAnimate.leave">
  4. <div v-show="showSearch" class="mb-[10px]">
  5. <el-card shadow="hover">
  6. <el-form ref="queryFormRef" :model="queryParams" :inline="true">
  7. <el-form-item label="订单编号" prop="orderNo">
  8. <el-input v-model="queryParams.orderNo" placeholder="请输入订单编号" clearable @keyup.enter="handleQuery" />
  9. </el-form-item>
  10. <el-form-item label="客户编号" prop="customerCode">
  11. <el-input v-model="queryParams.customerCode" placeholder="请输入客户编号" clearable @keyup.enter="handleQuery" />
  12. </el-form-item>
  13. <el-form-item label="客户名称" prop="customerName">
  14. <el-input v-model="queryParams.customerName" placeholder="请输入客户名称" clearable @keyup.enter="handleQuery" />
  15. </el-form-item>
  16. <el-form-item label="订单来源" prop="orderSource">
  17. <el-select v-model="queryParams.orderSource" placeholder="请选择订单来源" clearable>
  18. <el-option v-for="dict in order_source" :key="dict.value" :label="dict.label" :value="dict.value" />
  19. </el-select>
  20. </el-form-item>
  21. <el-form-item label="订单状态" prop="orderStatus">
  22. <el-select v-model="queryParams.orderStatus" placeholder="请选择订单状态" clearable>
  23. <el-option v-for="dict in order_status" :key="dict.value" :label="dict.label" :value="dict.value" />
  24. </el-select>
  25. </el-form-item>
  26. <el-form-item label="提交时间" prop="orderTime">
  27. <el-date-picker
  28. v-model="dateRange"
  29. type="daterange"
  30. range-separator="至"
  31. start-placeholder="开始时间"
  32. end-placeholder="结束时间"
  33. value-format="YYYY-MM-DD"
  34. />
  35. </el-form-item>
  36. <el-form-item>
  37. <el-button type="primary" icon="Search" @click="handleQuery()">搜索</el-button>
  38. <el-button icon="Refresh" @click="resetQuery">重置</el-button>
  39. </el-form-item>
  40. </el-form>
  41. </el-card>
  42. </div>
  43. </transition>
  44. <el-card shadow="never">
  45. <template #header>
  46. <el-row :gutter="10" class="mb8">
  47. <el-col :span="24"> 待审核订单列表 </el-col>
  48. </el-row>
  49. </template>
  50. <el-table v-loading="loading" border :data="orderMainList" @selection-change="handleSelectionChange">
  51. <el-table-column type="selection" width="55" align="center" />
  52. <el-table-column label="订单时间" align="center" prop="orderTime" />
  53. <el-table-column label="订单编号" align="center" prop="orderNo" />
  54. <el-table-column label="客户编号" align="center" prop="customerCode" />
  55. <el-table-column label="订单总金额" align="center" prop="totalAmount" />
  56. <el-table-column label="支付方式" align="center" prop="payType">
  57. <template #default="scope">
  58. <dict-tag :options="pay_method" :value="scope.row.payType" />
  59. </template>
  60. </el-table-column>
  61. <el-table-column label="业务员" align="center" prop="businessStaff" />
  62. <el-table-column label="客服" align="center" prop="customerService" />
  63. <el-table-column label="归属部门" align="center" prop="businessDept" />
  64. <el-table-column label="订单来源" align="center" prop="orderSource">
  65. <template #default="scope">
  66. <dict-tag :options="order_source" :value="scope.row.orderSource" />
  67. </template>
  68. </el-table-column>
  69. <el-table-column label="审核状态" align="center" prop="checkStatus">
  70. <template #default="scope">
  71. <dict-tag :options="order_check_status" :value="scope.row.checkStatus" />
  72. </template>
  73. </el-table-column>
  74. <el-table-column label="订单状态" align="center" prop="orderStatus">
  75. <template #default="scope">
  76. <dict-tag :options="order_status" :value="scope.row.orderStatus" />
  77. </template>
  78. </el-table-column>
  79. <el-table-column label="操作" align="center" class-name="small-padding fixed-width">
  80. <template #default="scope">
  81. <template v-for="btn in getActionButtons(scope.row)" :key="btn.label">
  82. <el-button link type="primary" @click="btn.handler(scope.row)">{{ btn.label }}</el-button>
  83. </template>
  84. </template>
  85. </el-table-column>
  86. </el-table>
  87. <pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" v-model:limit="queryParams.pageSize" @pagination="getList" />
  88. </el-card>
  89. </div>
  90. </template>
  91. <script setup name="OrderMain" lang="ts">
  92. import {
  93. listOrderMain,
  94. getOrderMain,
  95. delOrderMain,
  96. addOrderMain,
  97. updateOrderMain,
  98. closeOrderMain,
  99. queryOrderStatusStats,
  100. changeStatus,
  101. changeCheckStatus
  102. } from '@/api/order/orderMain';
  103. import { OrderMainVO, OrderMainQuery, OrderMainForm } from '@/api/order/orderMain/types';
  104. import DeliverDialog from './deliverDialog.vue';
  105. import LogisticsDetail from './logisticsDetail.vue';
  106. const { proxy } = getCurrentInstance() as ComponentInternalInstance;
  107. const { order_status, order_check_status, pay_method, order_source } = toRefs<any>(
  108. proxy?.useDict('order_status', 'order_check_status', 'pay_method', 'order_source')
  109. );
  110. const orderMainList = ref<OrderMainVO[]>([]);
  111. const buttonLoading = ref(false);
  112. const loading = ref(true);
  113. const showSearch = ref(true);
  114. const ids = ref<Array<string | number>>([]);
  115. const single = ref(true);
  116. const multiple = ref(true);
  117. const total = ref(0);
  118. const router = useRouter();
  119. const queryFormRef = ref<ElFormInstance>();
  120. const orderMainFormRef = ref<ElFormInstance>();
  121. const checkStatusValue = ref('1'); // 审核状态选择值
  122. // 发货对话框
  123. const showDeliverDialog = ref(false);
  124. const currentOrderId = ref<string | number>();
  125. const currentOrderNo = ref<string>();
  126. const dateRange = ref<[DateModelType, DateModelType]>(['', '']);
  127. const showLogisticsDialog = ref(false);
  128. const logisticsOrderId = ref<string | number>();
  129. const dialog = reactive<DialogOption>({
  130. visible: false,
  131. title: ''
  132. });
  133. const orderStatusStats = ref({
  134. pendingPaymentCount: 0, // 待支付
  135. pendingShipmentCount: 0, // 待发货
  136. shippedCount: 0, // 已发货(发货完成)
  137. completedCount: 0, // 已完成
  138. closedCount: 0, // 已关闭
  139. totalCount: 0 // 全部订单
  140. });
  141. const initFormData: OrderMainForm = {
  142. id: undefined,
  143. orderNo: undefined,
  144. shipmentNo: undefined,
  145. subOrderNo: undefined,
  146. companyId: undefined,
  147. customerId: undefined,
  148. customerCode: undefined,
  149. isSplitChild: undefined,
  150. userId: undefined,
  151. shippingAddressId: undefined,
  152. purchaseReason: undefined,
  153. invoiceType: undefined,
  154. payType: undefined,
  155. warehouseId: undefined,
  156. creditLimit: undefined,
  157. expectedDeliveryTime: undefined,
  158. businessStaff: undefined,
  159. customerService: undefined,
  160. businessDept: undefined,
  161. userDept: undefined,
  162. productQuantity: undefined,
  163. shippingFee: undefined,
  164. totalAmount: undefined,
  165. payableAmount: undefined,
  166. paymentStatus: undefined,
  167. orderSource: undefined,
  168. orderStatus: undefined,
  169. orderTime: undefined,
  170. confirmTime: undefined,
  171. shippingTime: undefined,
  172. receivingTime: undefined,
  173. shippedQuantity: undefined,
  174. unshippedQuantity: undefined,
  175. packageCount: undefined,
  176. signedQuantity: undefined,
  177. afterSaleCompleted: undefined,
  178. afterSalePending: undefined,
  179. deliveryDesc: undefined,
  180. pushStatus: undefined,
  181. attachmentPath: undefined,
  182. deliveryType: undefined,
  183. orderCategory: undefined,
  184. productCode: undefined,
  185. cancelReason: undefined,
  186. expenseType: undefined,
  187. userNo: undefined,
  188. status: undefined,
  189. remark: undefined,
  190. orderProductBos: [],
  191. customerSalesInfoVo: {}
  192. };
  193. const data = reactive<PageData<OrderMainForm, OrderMainQuery>>({
  194. form: { ...initFormData },
  195. queryParams: {
  196. pageNum: 1,
  197. pageSize: 10,
  198. orderNo: undefined,
  199. shipmentNo: undefined,
  200. subOrderNo: undefined,
  201. companyId: undefined,
  202. customerCode: undefined,
  203. customerId: undefined,
  204. userId: undefined,
  205. shippingAddressId: undefined,
  206. purchaseReason: undefined,
  207. invoiceType: undefined,
  208. payType: undefined,
  209. warehouseId: undefined,
  210. creditLimit: undefined,
  211. expectedDeliveryTime: undefined,
  212. businessStaff: undefined,
  213. customerService: undefined,
  214. businessDept: undefined,
  215. userDept: undefined,
  216. productQuantity: undefined,
  217. shippingFee: undefined,
  218. totalAmount: undefined,
  219. payableAmount: undefined,
  220. paymentStatus: undefined,
  221. orderSource: undefined,
  222. orderStatus: undefined,
  223. orderTime: undefined,
  224. confirmTime: undefined,
  225. shippingTime: undefined,
  226. receivingTime: undefined,
  227. shippedQuantity: undefined,
  228. unshippedQuantity: undefined,
  229. packageCount: undefined,
  230. signedQuantity: undefined,
  231. afterSaleCompleted: undefined,
  232. afterSalePending: undefined,
  233. deliveryDesc: undefined,
  234. pushStatus: undefined,
  235. attachmentPath: undefined,
  236. deliveryType: undefined,
  237. orderCategory: undefined,
  238. productCode: undefined,
  239. cancelReason: undefined,
  240. expenseType: undefined,
  241. status: undefined,
  242. platformCode: undefined,
  243. checkStatus: '0',
  244. params: {}
  245. },
  246. rules: {
  247. customerId: [{ required: true, message: '客户ID不能为空', trigger: 'blur' }],
  248. payType: [{ required: true, message: '支付方式不能为空', trigger: 'change' }],
  249. warehouseId: [{ required: true, message: '发货仓库不能为空', trigger: 'change' }],
  250. expectedDeliveryTime: [{ required: true, message: '预计送达时间不能为空', trigger: 'blur' }],
  251. shippingFee: [{ required: true, message: '运费不能为空', trigger: 'blur' }],
  252. confirmTime: [{ required: true, message: '确认时间不能为空', trigger: 'blur' }],
  253. shippingTime: [{ required: true, message: '发货时间不能为空', trigger: 'blur' }],
  254. receivingTime: [{ required: true, message: '收货时间不能为空', trigger: 'blur' }],
  255. cancelReason: [{ required: true, message: '取消或异常原因不能为空', trigger: 'blur' }],
  256. expenseType: [{ required: true, message: '费用类型不能为空', trigger: 'change' }]
  257. }
  258. });
  259. const { queryParams, form, rules } = toRefs(data);
  260. /** 查询订单主信息列表 */
  261. const getList = async () => {
  262. loading.value = true;
  263. const res = await listOrderMain(proxy?.addDateRange(queryParams.value, dateRange.value));
  264. orderMainList.value = res.rows;
  265. total.value = res.total;
  266. loading.value = false;
  267. queryOrderStatusStatsMethod();
  268. };
  269. const queryOrderStatusStatsMethod = async () => {
  270. const res = await queryOrderStatusStats();
  271. orderStatusStats.value = res as any;
  272. };
  273. /** 取消按钮 */
  274. const cancel = () => {
  275. reset();
  276. dialog.visible = false;
  277. };
  278. /** 表单重置 */
  279. const reset = () => {
  280. form.value = { ...initFormData };
  281. orderMainFormRef.value?.resetFields();
  282. };
  283. /** 搜索按钮操作 */
  284. const handleQuery = (orderStatus?: string) => {
  285. if (orderStatus) {
  286. queryParams.value.orderStatus = orderStatus;
  287. } else {
  288. queryParams.value.orderStatus = undefined;
  289. }
  290. queryParams.value.pageNum = 1;
  291. getList();
  292. };
  293. /** 重置按钮操作 */
  294. const resetQuery = () => {
  295. dateRange.value = ['', ''];
  296. queryFormRef.value?.resetFields();
  297. handleQuery();
  298. };
  299. /** 多选框选中数据 */
  300. const handleSelectionChange = (selection: OrderMainVO[]) => {
  301. ids.value = selection.map((item) => item.id);
  302. single.value = selection.length != 1;
  303. multiple.value = !selection.length;
  304. };
  305. /** 查看物流按钮操作 */
  306. const handleViewLogistics = (row?: OrderMainVO) => {
  307. if (!row?.id) {
  308. proxy?.$modal.msgWarning('订单ID不能为空');
  309. return;
  310. }
  311. logisticsOrderId.value = row.id;
  312. showLogisticsDialog.value = true;
  313. };
  314. const handleReview = (row?: OrderMainVO) => {
  315. router.push({
  316. path: '/order-manage/order-sendDetail',
  317. query: { id: row.id }
  318. });
  319. };
  320. /** 审核按钮操作 */
  321. const handleCheck = async (row?: OrderMainVO) => {
  322. const oldValue = row.checkStatus; // 保存旧值
  323. // 弹出审核选择对话框
  324. try {
  325. await ElMessageBox({
  326. title: '审核',
  327. message: h('div', { style: 'padding: 10px 0' }, [
  328. h('p', { style: 'margin-bottom: 15px; font-size: 14px' }, '请选择审核结果:'),
  329. h('div', { style: 'display: flex; flex-direction: column; gap: 12px' }, [
  330. h(
  331. 'label',
  332. {
  333. style: 'display: flex; align-items: center; cursor: pointer; font-size: 14px',
  334. onClick: () => {
  335. checkStatusValue.value = '1';
  336. const radio = document.querySelector('input[name="checkStatus"][value="1"]') as HTMLInputElement;
  337. if (radio) radio.checked = true;
  338. }
  339. },
  340. [
  341. h('input', {
  342. type: 'radio',
  343. name: 'checkStatus',
  344. value: '1',
  345. checked: true,
  346. style: 'margin-right: 8px; cursor: pointer',
  347. onChange: () => {
  348. checkStatusValue.value = '1';
  349. }
  350. }),
  351. h('span', null, '审核通过')
  352. ]
  353. ),
  354. h(
  355. 'label',
  356. {
  357. style: 'display: flex; align-items: center; cursor: pointer; font-size: 14px',
  358. onClick: () => {
  359. checkStatusValue.value = '2';
  360. const radio = document.querySelector('input[name="checkStatus"][value="2"]') as HTMLInputElement;
  361. if (radio) radio.checked = true;
  362. }
  363. },
  364. [
  365. h('input', {
  366. type: 'radio',
  367. name: 'checkStatus',
  368. value: '2',
  369. style: 'margin-right: 8px; cursor: pointer',
  370. onChange: () => {
  371. checkStatusValue.value = '2';
  372. }
  373. }),
  374. h('span', null, '驳回')
  375. ]
  376. )
  377. ])
  378. ]),
  379. showCancelButton: true,
  380. confirmButtonText: '确定',
  381. cancelButtonText: '取消',
  382. beforeClose: (action, instance, done) => {
  383. if (action === 'confirm') {
  384. done();
  385. } else {
  386. done();
  387. }
  388. }
  389. });
  390. // 调用接口,传入用户选择的值
  391. await changeCheckStatus(row.id, checkStatusValue.value);
  392. getList();
  393. proxy?.$modal.msgSuccess('操作成功');
  394. } catch (error) {
  395. // 用户取消或操作失败
  396. if (error !== 'cancel' && error !== 'close') {
  397. row.checkStatus = oldValue; // 失败回滚
  398. proxy?.$modal.msgError('操作失败,请重试');
  399. }
  400. }
  401. };
  402. /** 审核按钮操作 */
  403. const handleCancel = async (row?: OrderMainVO) => {
  404. const oldValue = row.checkStatus; // 保存旧值
  405. // 弹出审核选择对话框
  406. try {
  407. // 调用接口,传入用户选择的值
  408. await changeStatus(row.id, '7');
  409. getList();
  410. proxy?.$modal.msgSuccess('操作成功');
  411. } catch (error) {
  412. // 用户取消或操作失败
  413. if (error !== 'cancel' && error !== 'close') {
  414. row.checkStatus = oldValue; // 失败回滚
  415. proxy?.$modal.msgError('操作失败,请重试');
  416. }
  417. }
  418. };
  419. /** 发货按钮操作 */
  420. const handleDeliver = (row?: OrderMainVO) => {
  421. if (!row?.id) {
  422. proxy?.$modal.msgWarning('订单ID不能为空');
  423. return;
  424. }
  425. currentOrderId.value = row.id;
  426. currentOrderNo.value = row.orderNo;
  427. showDeliverDialog.value = true;
  428. };
  429. /** 发货成功回调 */
  430. const handleDeliverSuccess = () => {
  431. getList();
  432. };
  433. /** 修改按钮操作 */
  434. const handleUpdate = async (row?: OrderMainVO) => {
  435. reset();
  436. const _id = row?.id || ids.value[0];
  437. const res = await getOrderMain(_id);
  438. Object.assign(form.value, res.data);
  439. dialog.visible = true;
  440. dialog.title = '修改订单主信息';
  441. };
  442. /** 提交按钮 */
  443. const submitForm = () => {
  444. orderMainFormRef.value?.validate(async (valid: boolean) => {
  445. if (valid) {
  446. buttonLoading.value = true;
  447. if (form.value.id) {
  448. await updateOrderMain(form.value).finally(() => (buttonLoading.value = false));
  449. } else {
  450. await addOrderMain(form.value).finally(() => (buttonLoading.value = false));
  451. }
  452. proxy?.$modal.msgSuccess('操作成功');
  453. dialog.visible = false;
  454. await getList();
  455. }
  456. });
  457. };
  458. /** 删除按钮操作 */
  459. const handleDelete = async (row?: OrderMainVO) => {
  460. const _ids = row?.id || ids.value;
  461. await proxy?.$modal.confirm('是否确认删除订单主信息编号为"' + _ids + '"的数据项?').finally(() => (loading.value = false));
  462. await delOrderMain(_ids);
  463. proxy?.$modal.msgSuccess('删除成功');
  464. await getList();
  465. };
  466. /** 关闭订单操作 */
  467. const handleCloseOrder = async (row?: OrderMainVO) => {
  468. const _ids = row?.id || ids.value;
  469. await proxy?.$modal.confirm('是否确认关闭订单主信息编号为"' + _ids + '"的数据项?').finally(() => (loading.value = false));
  470. await closeOrderMain(_ids);
  471. proxy?.$modal.msgSuccess('关闭成功');
  472. await getList();
  473. };
  474. // 订单状态枚举
  475. enum OrderStatus {
  476. PENDING_PAYMENT = '0', // 待支付
  477. PENDING_CONFIRM = '1', // 待确认
  478. PENDING_SHIPMENT = '2', // 待发货
  479. PARTIAL_SHIPMENT = '3', // 部分发货
  480. SHIPMENT_COMPLETED = '4', // 发货完成
  481. COMPLETED = '5', // 已完成
  482. CLOSED = '6', // 已关闭
  483. CANCELLED = '7' // 已取消
  484. }
  485. // 审核状态枚举
  486. enum CheckStatus {
  487. PENDING = '0', // 待审核
  488. APPROVED = '1', // 审核通过
  489. REJECTED = '2' // 已驳回
  490. }
  491. // 按钮配置类型
  492. interface ActionButton {
  493. label: string;
  494. handler: (row: OrderMainVO) => void;
  495. }
  496. // 按钮配置映射:根据订单状态和审核状态返回可用按钮
  497. const getButtonsByStatus = (orderStatus: string, checkStatus: string): ActionButton[] => {
  498. const buttons: ActionButton[] = [];
  499. // 待支付状态:显示查看按钮
  500. if (orderStatus === OrderStatus.PENDING_PAYMENT) {
  501. buttons.push({ label: '查看', handler: handleReview });
  502. // buttons.push({ label: '确认订单', handler: handleAffirm });
  503. }
  504. // 待确认状态:显示确认订单按钮
  505. if (orderStatus === OrderStatus.PENDING_CONFIRM) {
  506. buttons.push({ label: '查看', handler: handleReview });
  507. // buttons.push({ label: '确认订单', handler: handleAffirm });
  508. }
  509. // 待审核:显示审核按钮
  510. if (checkStatus === CheckStatus.PENDING) {
  511. buttons.push({ label: '审核', handler: handleCheck });
  512. }
  513. // 待发货、部分发货、发货完成、已完成:显示查看订单信息按钮
  514. if (
  515. [OrderStatus.PENDING_SHIPMENT, OrderStatus.PARTIAL_SHIPMENT, OrderStatus.SHIPMENT_COMPLETED, OrderStatus.COMPLETED].includes(
  516. orderStatus as OrderStatus
  517. )
  518. ) {
  519. buttons.push({ label: '查看订单信息', handler: handleReview });
  520. }
  521. // 非已取消和已关闭状态:显示取消订单按钮
  522. if (orderStatus !== OrderStatus.CANCELLED && orderStatus !== OrderStatus.CLOSED) {
  523. // buttons.push({ label: '取消订单', handler: handleCancel });
  524. }
  525. return buttons;
  526. };
  527. // 根据订单状态获取可用的操作按钮
  528. const getActionButtons = (row: OrderMainVO): ActionButton[] => {
  529. return getButtonsByStatus(row.orderStatus, row.checkStatus);
  530. };
  531. /** 导出按钮操作 */
  532. const handleExport = () => {
  533. proxy?.download(
  534. 'system/orderMain/export',
  535. {
  536. ...queryParams.value
  537. },
  538. `orderMain_${new Date().getTime()}.xlsx`
  539. );
  540. };
  541. onMounted(() => {
  542. getList();
  543. });
  544. onActivated(() => {
  545. getList();
  546. });
  547. </script>