OrderList.vue
25.7 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
<script setup lang="ts">
import { ref, reactive } from 'vue'
import { useRouter } from 'vue-router'
import { Search, Download } from '@element-plus/icons-vue'
import type { Order, OrderQuery, DetailRecord, ApiResponse } from '../types/order'
import { qualityApi } from '../api'
import { ElMessage, ElMessageBox } from 'element-plus'
// --- State ---
const loading = ref(false)
const tableData = ref<Order[]>([])
const cities = [
{ code: '3201', name: '南京市' },
{ code: '3202', name: '无锡市' },
{ code: '3203', name: '徐州市' },
{ code: '3204', name: '常州市' },
{ code: '3205', name: '苏州市' },
{ code: '3206', name: '南通市' },
{ code: '3207', name: '连云港市' },
{ code: '3208', name: '淮安市' },
{ code: '3209', name: '盐城市' },
{ code: '3210', name: '扬州市' },
{ code: '3211', name: '镇江市' },
{ code: '3212', name: '泰州市' },
{ code: '3213', name: '宿迁市' }
]
const router = useRouter()
// Search Form
const queryForm = reactive<OrderQuery>({
businessAccount: '',
applyId: '',
orderId: '',
workerId: '',
city: '',
status: '',
dateRange: undefined,
noPhotoCountMin: undefined,
manualInputCountMin: undefined,
envAbnormalCountMin: undefined,
isAbnormal: '',
page: 1,
pageSize: 10
})
const total = ref(0)
// --- Methods ---
const fetchData = async () => {
loading.value = true
try {
// 构建接口参数,将前端筛选条件映射到后端接口参数
const params: any = {
pageNum: queryForm.page,
pageSize: queryForm.pageSize
}
// 业务账号 -> accNbr
if (queryForm.businessAccount) {
params.accNbr = queryForm.businessAccount
}
// Apply_id
if (queryForm.applyId) {
params.applyId = queryForm.applyId
}
// 工单ID -> orderId
if (queryForm.orderId) {
params.orderId = queryForm.orderId
}
// 师傅工号 -> campaignId (根据实际业务逻辑,这里假设师傅工号对应 campaignId)
if (queryForm.workerId) {
params.campaignId = queryForm.workerId
}
// 所属地市 -> areaType
if (queryForm.city) {
params.areaType = queryForm.city
}
// 质检状态 -> checkStatus
if (queryForm.status) {
params.checkStatus = queryForm.status
}
// 质检时间范围 -> startTime, endTime
if (queryForm.dateRange && queryForm.dateRange.length === 2) {
params.startTime = queryForm.dateRange[0]
params.endTime = queryForm.dateRange[1]
}
// 无法拍摄次数 -> noShowNumType
if (queryForm.noPhotoCountMin !== undefined && queryForm.noPhotoCountMin > 0) {
params.noShowNumType = queryForm.noPhotoCountMin
}
// 手动输入次数 -> manualInputNumType
if (queryForm.manualInputCountMin !== undefined && queryForm.manualInputCountMin > 0) {
params.manualInputNumType = queryForm.manualInputCountMin
}
// 环境异常次数 -> cheatNumType
if (queryForm.envAbnormalCountMin !== undefined && queryForm.envAbnormalCountMin > 0) {
params.cheatNumType = queryForm.envAbnormalCountMin
}
// 是否异常 -> isCheat
if (queryForm.isAbnormal === 'abnormal') {
params.isCheat = 1
} else if (queryForm.isAbnormal === 'normal') {
params.isCheat = 0
}
const response = await qualityApi.getQualityCheckList(params) as ApiResponse<any>
if (response.code === 200 || response.code === 0) {
const data = response.data || { records: [], total: 0 }
// 将后端数据映射到前端 Order 类型
tableData.value = (data.records || []).map((item: any) => ({
id: item.applyId, // 使用 applyId 作为唯一 id
applyId: item.applyId,
workerId: item.campaignId,
businessAccount: item.accNbr,
orderIds: [], // 工单ID列表需要单独查询或从其他字段获取
city: item.areaName,
status: item.checkStatusDesc || getStatusText(item.checkStatus),
cannotQcReason: item.failReason || '',
startTime: item.startTime || '',
endTime: item.endTime || '',
noPhotoCount: item.noShowNum || 0,
manualInputCount: item.manualInputNum || 0,
envAbnormalCount: item.cheatNum || 0,
isCheating: item.isCheat === 1,
cheatingReason: '',
cheatingRemark: '',
cheatingTime: ''
}))
total.value = data.total || 0
}
} catch (error) {
console.error('获取质检工单列表失败:', error)
// 错误提示已在 request.ts 中统一处理
} finally {
loading.value = false
}
}
// 辅助函数:将数字状态转换为文字
const getStatusText = (status: number): string => {
const statusMap: Record<number, string> = {
0: '未开始',
1: '未开始',
2: '进行中',
3: '已完成',
4: '无法质检'
}
return statusMap[status] || '未知'
}
const handleSearch = () => {
queryForm.page = 1
fetchData()
}
const handleReset = () => {
// Reset all form fields
queryForm.businessAccount = ''
queryForm.applyId = ''
queryForm.orderId = ''
queryForm.workerId = ''
queryForm.city = ''
queryForm.status = ''
queryForm.dateRange = undefined
queryForm.noPhotoCountMin = undefined
queryForm.manualInputCountMin = undefined
queryForm.envAbnormalCountMin = undefined
queryForm.isAbnormal = ''
queryForm.page = 1
handleSearch()
}
const handleExport = async () => {
try {
loading.value = true
// 使用相同的查询参数进行导出
const params: any = {
pageNum: queryForm.page,
pageSize: queryForm.pageSize
}
if (queryForm.businessAccount) params.accNbr = queryForm.businessAccount
if (queryForm.applyId) params.applyId = queryForm.applyId
if (queryForm.orderId) params.orderId = queryForm.orderId
if (queryForm.workerId) params.campaignId = queryForm.workerId
if (queryForm.city) params.areaType = queryForm.city
if (queryForm.status) params.checkStatus = queryForm.status
if (queryForm.dateRange && queryForm.dateRange.length === 2) {
params.startTime = queryForm.dateRange[0]
params.endTime = queryForm.dateRange[1]
}
if (queryForm.noPhotoCountMin !== undefined && queryForm.noPhotoCountMin > 0) {
params.noShowNumType = queryForm.noPhotoCountMin
}
if (queryForm.manualInputCountMin !== undefined && queryForm.manualInputCountMin > 0) {
params.manualInputNumType = queryForm.manualInputCountMin
}
if (queryForm.envAbnormalCountMin !== undefined && queryForm.envAbnormalCountMin > 0) {
params.cheatNumType = queryForm.envAbnormalCountMin
}
if (queryForm.isAbnormal === 'abnormal') {
params.isCheat = 1
} else if (queryForm.isAbnormal === 'normal') {
params.isCheat = 0
}
const response = await qualityApi.exportQualityCheckList(params)
// 处理文件下载
const blob = new Blob([response], { type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' })
const url = window.URL.createObjectURL(blob)
const link = document.createElement('a')
link.href = url
link.download = `质检工单列表_${new Date().getTime()}.xlsx`
document.body.appendChild(link)
link.click()
document.body.removeChild(link)
window.URL.revokeObjectURL(url)
ElMessage.success('导出成功')
} catch (error) {
console.error('导出失败:', error)
// 错误提示已在 request.ts 中统一处理
} finally {
loading.value = false
}
}
// Table Handlers
const handlePageChange = (val: number) => {
queryForm.page = val
fetchData()
}
// Dialogs State
const dialogVisible = reactive({
orderIds: false,
cannotQc: false,
markCheating: false,
cancelCheating: false,
details: false,
cheatingInfo: false
})
const currentOrder = ref<Order | null>(null)
const detailsType = ref<'noPhoto' | 'manual' | 'env'>('noPhoto')
const detailsTitle = ref('')
const detailsData = ref<DetailRecord[]>([])
// Forms
const cannotQcForm = reactive({
type: '',
reason: ''
})
const markCheatingForm = reactive({
reason: '',
remark: ''
})
// --- Actions ---
// 1. Order IDs
const openIdsDialog = async (row: Order) => {
currentOrder.value = row
try {
// 调用接口获取工单设备列表
const response = await qualityApi.getDevicesByApplyId({
applyId: row.applyId
}) as ApiResponse<any>
if (response.code === 200 || response.code === 0) {
// 更新当前订单的工单设备列表
const devices = response.data || []
// 将设备列表映射为包含工单号和设备串号的对象数组
currentOrder.value.orderIds = devices.map((device: any) => ({
orderCode: device.orderCode || '',
deviceNumber: device.deviceNumber || ''
}))
}
} catch (error) {
console.error('获取工单设备列表失败:', error)
// 错误提示已在 request.ts 中统一处理
}
dialogVisible.orderIds = true
}
// 2. Numeric Details Popup
const openDetailsDialog = async (row: Order, type: 'noPhoto' | 'manual' | 'env') => {
currentOrder.value = row
detailsType.value = type
// 设置标题
if (type === 'noPhoto') {
detailsTitle.value = '无法拍摄明细'
} else if (type === 'manual') {
detailsTitle.value = '手动输入明细'
} else {
detailsTitle.value = '环境异常明细'
}
try {
// 调用接口获取流程明细列表
const response = await qualityApi.getProcessDetailList({
applyId: row.applyId
}) as ApiResponse<any[]>
if (response.code === 200 || response.code === 0) {
const data = response.data || []
// 根据类型过滤数据
// 这里假设后端返回的数据包含 type 字段来区分不同类型
// 实际字段名需要根据后端接口文档调整
detailsData.value = data.map((item: any, index: number) => ({
id: index + 1,
process: item.processName || item.process || '',
reason: item.reason || '',
time: item.createTime || item.time || ''
}))
} else {
detailsData.value = []
}
} catch (error) {
console.error('获取流程明细失败:', error)
detailsData.value = []
// 错误提示已在 request.ts 中统一处理
}
dialogVisible.details = true
}
// 3. Cheating Info Popup
const openCheatingInfo = async (row: Order) => {
currentOrder.value = row
try {
// 调用接口获取作弊标记详情
const response = await qualityApi.getCheatMarkDetail({
applyId: row.applyId
}) as ApiResponse<any>
if (response.code === 200 || response.code === 0) {
// 更新当前订单的作弊信息
if (response.data && currentOrder.value) {
currentOrder.value.cheatingReason = response.data.cause || row.cheatingReason
currentOrder.value.cheatingRemark = response.data.causeOther || row.cheatingRemark
currentOrder.value.cheatingTime = response.data.createTime || row.cheatingTime
}
}
} catch (error) {
console.error('获取作弊详情失败:', error)
// 即使获取失败也显示弹窗,使用现有数据
}
dialogVisible.cheatingInfo = true
}
// 4. Cannot QC Action
const openCannotQcDialog = (row: Order) => {
currentOrder.value = row
cannotQcForm.type = ''
cannotQcForm.reason = ''
dialogVisible.cannotQc = true
}
const submitCannotQc = async () => {
if (!cannotQcForm.type || !cannotQcForm.reason) {
ElMessage.warning('请填写完整信息')
return
}
if (!currentOrder.value) {
ElMessage.warning('未选择工单')
return
}
try {
// 将原因类型映射为对应的 checkStatus
// 个人原因: 2, 用户原因: 3, 其他: 4
let checkStatus = 4 // 默认为"其他"
if (cannotQcForm.type === '个人原因') {
checkStatus = 2
} else if (cannotQcForm.type === '用户原因') {
checkStatus = 3
}
await qualityApi.unCheckByOpt({
applyId: currentOrder.value.applyId,
checkStatus,
failReason: cannotQcForm.reason
})
ElMessage.success('提交成功')
dialogVisible.cannotQc = false
// 刷新列表
fetchData()
} catch (error) {
console.error('无法质检提交失败:', error)
// 错误提示已在 request.ts 中统一处理
}
}
// 5. Mark Cheating Action
const openMarkCheating = (row: Order) => {
currentOrder.value = row
markCheatingForm.reason = ''
markCheatingForm.remark = ''
dialogVisible.markCheating = true
}
const submitMarkCheating = async () => {
if (!markCheatingForm.reason) {
ElMessage.warning('请选择异常原因')
return
}
if (!currentOrder.value) {
ElMessage.warning('未选择工单')
return
}
try {
// 调用标记作弊接口, markType=1 表示标记作弊
await qualityApi.markCheat({
applyId: currentOrder.value.applyId,
markType: 1,
cause: markCheatingForm.reason,
causeOther: markCheatingForm.remark || undefined
})
ElMessage.success('标记成功')
dialogVisible.markCheating = false
// 刷新列表
fetchData()
} catch (error) {
console.error('标记作弊失败:', error)
// 错误提示已在 request.ts 中统一处理
}
}
// 6. Cancel Cheating Action
const openCancelCheating = async (row: Order) => {
try {
await ElMessageBox.confirm(
'确认要取消该工单的作弊标记吗?',
'取消作弊',
{
confirmButtonText: '确认',
cancelButtonText: '取消',
type: 'warning',
}
)
// 调用取消作弊接口, markType=2 表示取消作弊标记
await qualityApi.markCheat({
applyId: row.applyId,
markType: 2
})
ElMessage.success('取消成功')
// 刷新列表
fetchData()
} catch (error: any) {
// 用户取消操作时不显示错误
if (error !== 'cancel') {
console.error('取消作弊失败:', error)
// 错误提示已在 request.ts 中统一处理
}
}
}
// Initial Fetch
fetchData()
</script>
<template>
<div class="p-4">
<!-- Search Area -->
<el-card shadow="never" class="mb-4">
<el-form :model="queryForm" label-width="90px" class="flex flex-wrap">
<el-form-item label="业务账号">
<el-input v-model="queryForm.businessAccount" placeholder="手机号/固话" clearable class="!w-48" />
</el-form-item>
<el-form-item label="Apply_id">
<el-input v-model="queryForm.applyId" placeholder="请输入" clearable class="!w-48" />
</el-form-item>
<el-form-item label="工单ID">
<el-input v-model="queryForm.orderId" placeholder="请输入" clearable class="!w-48" />
</el-form-item>
<el-form-item label="师傅工号">
<el-input v-model="queryForm.workerId" placeholder="请输入" clearable class="!w-48" />
</el-form-item>
<el-form-item label="所属地市">
<el-select v-model="queryForm.city" placeholder="请选择" clearable class="!w-48">
<el-option v-for="city in cities" :key="city.code" :label="city.name" :value="city.code" />
</el-select>
</el-form-item>
<el-form-item label="质检状态">
<el-select v-model="queryForm.status" placeholder="请选择" clearable class="!w-48">
<el-option label="全部" value="" />
<el-option label="未开始" :value="1" />
<el-option label="进行中" :value="2" />
<el-option label="已完成" :value="3" />
<el-option label="无法质检" :value="4" />
</el-select>
</el-form-item>
<el-form-item label="质检时间">
<el-date-picker
v-model="queryForm.dateRange"
type="datetimerange"
range-separator="至"
start-placeholder="开始时间"
end-placeholder="结束时间"
value-format="YYYY-MM-DD HH:mm:ss"
class="!w-80"
/>
</el-form-item>
<el-form-item label="无法拍摄">
<el-input-number v-model="queryForm.noPhotoCountMin" :min="1" controls-position="right" class="!w-32" placeholder=">=1" />
</el-form-item>
<el-form-item label="手动输入">
<el-input-number v-model="queryForm.manualInputCountMin" :min="1" controls-position="right" class="!w-32" placeholder=">=1" />
</el-form-item>
<el-form-item label="环境异常">
<el-input-number v-model="queryForm.envAbnormalCountMin" :min="1" controls-position="right" class="!w-32" placeholder=">=1" />
</el-form-item>
<el-form-item label="是否异常">
<el-select v-model="queryForm.isAbnormal" placeholder="请选择" clearable class="!w-32">
<el-option label="全部" value="" />
<el-option label="异常" value="abnormal" />
<el-option label="正常" value="normal" />
</el-select>
</el-form-item>
<el-form-item class="ml-auto">
<el-button type="primary" :icon="Search" @click="handleSearch">查询</el-button>
<el-button @click="handleReset">重置</el-button>
<el-button type="success" :icon="Download" @click="handleExport">导出</el-button>
</el-form-item>
</el-form>
</el-card>
<!-- Table Area -->
<el-card shadow="never">
<el-table :data="tableData" v-loading="loading" border style="width: 100%">
<el-table-column prop="applyId" label="Apply_id" min-width="120" show-overflow-tooltip />
<el-table-column prop="workerId" label="师傅工号" width="100" show-overflow-tooltip />
<el-table-column prop="businessAccount" label="业务账号" min-width="120" show-overflow-tooltip />
<el-table-column label="工单ID" width="100">
<template #default="{ row }">
<el-button link type="primary" @click="openIdsDialog(row)">查看</el-button>
</template>
</el-table-column>
<el-table-column prop="city" label="所属地市" width="100" show-overflow-tooltip />
<el-table-column prop="status" label="质检状态" width="100">
<template #default="{ row }">
<el-tag :type="row.status === '已完成' ? 'success' : row.status === '无法质检' ? 'info' : ''">{{ row.status }}</el-tag>
</template>
</el-table-column>
<el-table-column prop="cannotQcReason" label="无法质检原因" min-width="120" show-overflow-tooltip />
<el-table-column prop="startTime" label="开始时间" width="160" show-overflow-tooltip />
<el-table-column prop="endTime" label="结束时间" width="160" show-overflow-tooltip />
<!-- Numeric Columns with Sort -->
<el-table-column prop="noPhotoCount" label="无法拍摄" sortable width="110">
<template #default="{ row }">
<span class="text-blue-500 cursor-pointer font-bold hover:underline"
v-if="row.noPhotoCount > 0"
@click="openDetailsDialog(row, 'noPhoto')">
{{ row.noPhotoCount }}
</span>
<span v-else>0</span>
</template>
</el-table-column>
<el-table-column prop="manualInputCount" label="手动输入" sortable width="110">
<template #default="{ row }">
<span class="text-blue-500 cursor-pointer font-bold hover:underline"
v-if="row.manualInputCount > 0"
@click="openDetailsDialog(row, 'manual')">
{{ row.manualInputCount }}
</span>
<span v-else>0</span>
</template>
</el-table-column>
<el-table-column prop="envAbnormalCount" label="环境异常" sortable width="110">
<template #default="{ row }">
<span class="text-blue-500 cursor-pointer font-bold hover:underline"
v-if="row.envAbnormalCount > 0"
@click="openDetailsDialog(row, 'env')">
{{ row.envAbnormalCount }}
</span>
<span v-else>0</span>
</template>
</el-table-column>
<el-table-column label="疑似作弊" width="100">
<template #default="{ row }">
<span v-if="row.isCheating" class="text-red-500 cursor-pointer hover:underline" @click="openCheatingInfo(row)">是</span>
<span v-else>否</span>
</template>
</el-table-column>
<el-table-column label="操作" width="220" fixed="right">
<template #default="{ row }">
<el-button link type="primary" @click="router.push({ path: `/order-detail/${row.applyId}`, query: { data: JSON.stringify(row) } })">详情</el-button>
<el-button link type="warning" v-if="row.status !== '已完成' && row.status !== '无法质检'" @click="openCannotQcDialog(row)">无法质检</el-button>
<el-button link type="danger" v-if="!row.isCheating" @click="openMarkCheating(row)">标记作弊</el-button>
<el-button link type="info" v-else @click="openCancelCheating(row)">取消作弊</el-button>
</template>
</el-table-column>
</el-table>
<div class="mt-4 flex justify-end">
<el-pagination
v-model:current-page="queryForm.page"
v-model:page-size="queryForm.pageSize"
:page-sizes="[10, 20, 50]"
:total="total"
layout="total, sizes, prev, pager, next, jumper"
@current-change="handlePageChange"
/>
</div>
</el-card>
<!-- Dialog: Order IDs -->
<el-dialog v-model="dialogVisible.orderIds" title="工单设备列表" width="600px">
<div v-if="currentOrder">
<p class="mb-2"><strong>Apply_id:</strong> {{ currentOrder.applyId }}</p>
<p class="mb-4"><strong>业务账号:</strong> {{ currentOrder.businessAccount }}</p>
<el-table :data="currentOrder.orderIds" border stripe max-height="300">
<el-table-column type="index" label="序号" width="60" align="center" />
<el-table-column prop="orderCode" label="工单号" show-overflow-tooltip />
<el-table-column prop="deviceNumber" label="设备串号" show-overflow-tooltip />
</el-table>
</div>
</el-dialog>
<!-- Dialog: Numeric Details (NoPhoto/Manual/Env) -->
<el-dialog v-model="dialogVisible.details" :title="detailsTitle" width="600px">
<el-table :data="detailsData" border stripe>
<el-table-column type="index" label="序号" width="60" align="center" />
<el-table-column prop="process" label="流程" />
<el-table-column prop="reason" label="原因" v-if="detailsType !== 'manual'" />
<el-table-column prop="time" label="提交时间" />
</el-table>
</el-dialog>
<!-- Dialog: Cheating Info -->
<el-dialog v-model="dialogVisible.cheatingInfo" title="疑似作弊详情" width="400px">
<div v-if="currentOrder">
<p class="mb-2"><strong>疑似作弊原因:</strong> {{ currentOrder.cheatingReason || '无' }}</p>
<p class="mb-2"><strong>备注:</strong> {{ currentOrder.cheatingRemark || '无' }}</p>
<p class="mb-2"><strong>标记时间:</strong> {{ currentOrder.cheatingTime || '无' }}</p>
</div>
</el-dialog>
<!-- Dialog: Cannot QC Form -->
<el-dialog v-model="dialogVisible.cannotQc" title="无法质检处理" width="500px">
<el-form label-width="80px">
<el-form-item label="原因类型" required>
<el-radio-group v-model="cannotQcForm.type">
<el-radio value="个人原因">个人原因</el-radio>
<el-radio value="用户原因">用户原因</el-radio>
<el-radio value="其他">其他</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item label="具体原因" required>
<el-input type="textarea" v-model="cannotQcForm.reason" placeholder="请填写具体原因" :rows="3" />
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible.cannotQc = false">取消</el-button>
<el-button type="primary" @click="submitCannotQc">提交</el-button>
</template>
</el-dialog>
<!-- Dialog: Mark Cheating Form -->
<el-dialog v-model="dialogVisible.markCheating" title="标记作弊" width="500px">
<div v-if="currentOrder" class="mb-4 p-2 bg-gray-50 rounded">
<p class="text-sm text-gray-600">Apply_id: {{ currentOrder.applyId }}</p>
<p class="text-sm text-gray-600">业务账号: {{ currentOrder.businessAccount }}</p>
</div>
<el-form label-width="80px">
<el-form-item label="异常原因" required>
<el-select v-model="markCheatingForm.reason" class="w-full">
<el-option value="疑似不在用户家质检" label="疑似不在用户家质检" />
<el-option value="一直点击无法拍摄跳过" label="一直点击无法拍摄跳过" />
<el-option value="识别屏幕/识别白纸" label="识别屏幕/识别白纸" />
<el-option value="其他" label="其他" />
</el-select>
</el-form-item>
<el-form-item label="备注">
<el-input type="textarea" v-model="markCheatingForm.remark" placeholder="选填" :rows="3" />
</el-form-item>
</el-form>
<template #footer>
<el-button @click="dialogVisible.markCheating = false">取消</el-button>
<el-button type="primary" @click="submitMarkCheating">提交</el-button>
</template>
</el-dialog>
</div>
</template>
<style scoped>
.el-form-item {
margin-right: 16px;
margin-bottom: 16px;
}
/* 表格单元格不换行,超出显示省略号 */
:deep(.el-table) {
.el-table__cell {
/* 防止单元格内容换行 */
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
/* 固定表格行高 */
.el-table__row {
height: 50px;
}
/* 单元格内容垂直居中 */
.cell {
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
line-height: 50px;
}
}
</style>